this repo has no description
1package pages 2 3import ( 4 "embed" 5 "fmt" 6 "html/template" 7 "io" 8 "io/fs" 9 "log" 10 "net/http" 11 "path" 12 "strings" 13 14 "github.com/dustin/go-humanize" 15 "github.com/sotangled/tangled/appview/auth" 16 "github.com/sotangled/tangled/appview/db" 17 "github.com/sotangled/tangled/types" 18) 19 20//go:embed templates/* static/* 21var files embed.FS 22 23type Pages struct { 24 t map[string]*template.Template 25} 26 27func funcMap() template.FuncMap { 28 return template.FuncMap{ 29 "split": func(s string) []string { 30 return strings.Split(s, "\n") 31 }, 32 "add": func(a, b int) int { 33 return a + b 34 }, 35 "sub": func(a, b int) int { 36 return a - b 37 }, 38 "cond": func(cond interface{}, a, b string) string { 39 if cond == nil { 40 return b 41 } 42 43 if boolean, ok := cond.(bool); boolean && ok { 44 return a 45 } 46 47 return b 48 }, 49 "didOrHandle": func(did, handle string) string { 50 if handle != "" { 51 return fmt.Sprintf("@%s", handle) 52 } else { 53 return did 54 } 55 }, 56 "assoc": func(values ...string) ([][]string, error) { 57 if len(values)%2 != 0 { 58 return nil, fmt.Errorf("invalid assoc call, must have an even number of arguments") 59 } 60 pairs := make([][]string, 0) 61 for i := 0; i < len(values); i += 2 { 62 pairs = append(pairs, []string{values[i], values[i+1]}) 63 } 64 return pairs, nil 65 }, 66 "append": func(s []string, values ...string) []string { 67 for _, v := range values { 68 s = append(s, v) 69 } 70 return s 71 }, 72 "timeFmt": humanize.Time, 73 "length": func(v []string) int { 74 return len(v) 75 }, 76 "splitN": func(s, sep string, n int) []string { 77 return strings.SplitN(s, sep, n) 78 }, 79 "unescapeHtml": func(s string) template.HTML { 80 return template.HTML(s) 81 }, 82 } 83} 84 85func NewPages() *Pages { 86 templates := make(map[string]*template.Template) 87 88 // Walk through embedded templates directory and parse all .html files 89 err := fs.WalkDir(files, "templates", func(path string, d fs.DirEntry, err error) error { 90 if err != nil { 91 return err 92 } 93 94 if !d.IsDir() && strings.HasSuffix(path, ".html") { 95 name := strings.TrimPrefix(path, "templates/") 96 name = strings.TrimSuffix(name, ".html") 97 98 if !strings.HasPrefix(path, "templates/layouts/") { 99 // Add the page template on top of the base 100 tmpl, err := template.New(name). 101 Funcs(funcMap()). 102 ParseFS(files, "templates/layouts/*.html", path) 103 if err != nil { 104 return fmt.Errorf("setting up template: %w", err) 105 } 106 107 templates[name] = tmpl 108 log.Printf("loaded template: %s", name) 109 } 110 111 return nil 112 } 113 return nil 114 }) 115 if err != nil { 116 log.Fatalf("walking template dir: %v", err) 117 } 118 119 log.Printf("total templates loaded: %d", len(templates)) 120 121 return &Pages{ 122 t: templates, 123 } 124} 125 126type LoginParams struct { 127} 128 129func (p *Pages) execute(name string, w io.Writer, params any) error { 130 return p.t[name].ExecuteTemplate(w, "layouts/base", params) 131} 132 133func (p *Pages) executePlain(name string, w io.Writer, params any) error { 134 return p.t[name].Execute(w, params) 135} 136 137func (p *Pages) executeRepo(name string, w io.Writer, params any) error { 138 return p.t[name].ExecuteTemplate(w, "layouts/repobase", params) 139} 140 141func (p *Pages) Login(w io.Writer, params LoginParams) error { 142 return p.executePlain("user/login", w, params) 143} 144 145type TimelineParams struct { 146 LoggedInUser *auth.User 147} 148 149func (p *Pages) Timeline(w io.Writer, params TimelineParams) error { 150 return p.execute("timeline", w, params) 151} 152 153type SettingsParams struct { 154 LoggedInUser *auth.User 155 PubKeys []db.PublicKey 156} 157 158func (p *Pages) Settings(w io.Writer, params SettingsParams) error { 159 return p.execute("settings/keys", w, params) 160} 161 162type KnotsParams struct { 163 LoggedInUser *auth.User 164 Registrations []db.Registration 165} 166 167func (p *Pages) Knots(w io.Writer, params KnotsParams) error { 168 return p.execute("knots", w, params) 169} 170 171type KnotParams struct { 172 LoggedInUser *auth.User 173 Registration *db.Registration 174 Members []string 175 IsOwner bool 176} 177 178func (p *Pages) Knot(w io.Writer, params KnotParams) error { 179 return p.execute("knot", w, params) 180} 181 182type NewRepoParams struct { 183 LoggedInUser *auth.User 184 Knots []string 185} 186 187func (p *Pages) NewRepo(w io.Writer, params NewRepoParams) error { 188 return p.execute("repo/new", w, params) 189} 190 191type ProfilePageParams struct { 192 LoggedInUser *auth.User 193 UserDid string 194 UserHandle string 195 Repos []db.Repo 196} 197 198func (p *Pages) ProfilePage(w io.Writer, params ProfilePageParams) error { 199 return p.execute("user/profile", w, params) 200} 201 202type RepoInfo struct { 203 Name string 204 OwnerDid string 205 OwnerHandle string 206 Description string 207 SettingsAllowed bool 208} 209 210func (r RepoInfo) OwnerWithAt() string { 211 if r.OwnerHandle != "" { 212 return fmt.Sprintf("@%s", r.OwnerHandle) 213 } else { 214 return r.OwnerDid 215 } 216} 217 218func (r RepoInfo) FullName() string { 219 return path.Join(r.OwnerWithAt(), r.Name) 220} 221 222func (r RepoInfo) GetTabs() [][]string { 223 tabs := [][]string{ 224 {"overview", "/"}, 225 {"issues", "/issues"}, 226 {"pulls", "/pulls"}, 227 } 228 229 if r.SettingsAllowed { 230 tabs = append(tabs, []string{"settings", "/settings"}) 231 } 232 233 return tabs 234} 235 236type RepoIndexParams struct { 237 LoggedInUser *auth.User 238 RepoInfo RepoInfo 239 Active string 240 types.RepoIndexResponse 241} 242 243func (p *Pages) RepoIndexPage(w io.Writer, params RepoIndexParams) error { 244 params.Active = "overview" 245 return p.executeRepo("repo/index", w, params) 246} 247 248type RepoLogParams struct { 249 LoggedInUser *auth.User 250 RepoInfo RepoInfo 251 types.RepoLogResponse 252} 253 254func (p *Pages) RepoLog(w io.Writer, params RepoLogParams) error { 255 return p.execute("repo/log", w, params) 256} 257 258type RepoCommitParams struct { 259 LoggedInUser *auth.User 260 RepoInfo RepoInfo 261 types.RepoCommitResponse 262} 263 264func (p *Pages) RepoCommit(w io.Writer, params RepoCommitParams) error { 265 return p.executeRepo("repo/commit", w, params) 266} 267 268type RepoTreeParams struct { 269 LoggedInUser *auth.User 270 RepoInfo RepoInfo 271 Active string 272 BreadCrumbs [][]string 273 BaseTreeLink string 274 BaseBlobLink string 275 types.RepoTreeResponse 276} 277 278func (p *Pages) RepoTree(w io.Writer, params RepoTreeParams) error { 279 params.Active = "overview" 280 return p.execute("repo/tree", w, params) 281} 282 283type RepoBranchesParams struct { 284 LoggedInUser *auth.User 285 RepoInfo RepoInfo 286 types.RepoBranchesResponse 287} 288 289func (p *Pages) RepoBranches(w io.Writer, params RepoBranchesParams) error { 290 return p.executeRepo("repo/branches", w, params) 291} 292 293type RepoTagsParams struct { 294 LoggedInUser *auth.User 295 RepoInfo RepoInfo 296 types.RepoTagsResponse 297} 298 299func (p *Pages) RepoTags(w io.Writer, params RepoTagsParams) error { 300 return p.executeRepo("repo/tags", w, params) 301} 302 303type RepoBlobParams struct { 304 LoggedInUser *auth.User 305 RepoInfo RepoInfo 306 Active string 307 BreadCrumbs [][]string 308 types.RepoBlobResponse 309} 310 311func (p *Pages) RepoBlob(w io.Writer, params RepoBlobParams) error { 312 params.Active = "overview" 313 return p.executeRepo("repo/blob", w, params) 314} 315 316type RepoSettingsParams struct { 317 LoggedInUser *auth.User 318 RepoInfo RepoInfo 319 Collaborators [][]string 320 Active string 321 IsCollaboratorInviteAllowed bool 322} 323 324func (p *Pages) RepoSettings(w io.Writer, params RepoSettingsParams) error { 325 params.Active = "settings" 326 return p.executeRepo("repo/settings", w, params) 327} 328 329func (p *Pages) Static() http.Handler { 330 sub, err := fs.Sub(files, "static") 331 if err != nil { 332 log.Fatalf("no static dir found? that's crazy: %v", err) 333 } 334 return http.StripPrefix("/static/", http.FileServer(http.FS(sub))) 335} 336 337func (p *Pages) Error500(w io.Writer) error { 338 return p.execute("errors/500", w, nil) 339} 340 341func (p *Pages) Error404(w io.Writer) error { 342 return p.execute("errors/404", w, nil) 343} 344 345func (p *Pages) Error503(w io.Writer) error { 346 return p.execute("errors/503", w, nil) 347}