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