Monorepo for Tangled
tangled.org
1package pages
2
3import (
4 "crypto/sha256"
5 "embed"
6 "encoding/hex"
7 "fmt"
8 "html/template"
9 "io"
10 "io/fs"
11 "log/slog"
12 "net/http"
13 "os"
14 "path/filepath"
15 "strings"
16 "sync"
17 "time"
18
19 "tangled.org/core/api/tangled"
20 "tangled.org/core/appview/cache"
21 "tangled.org/core/appview/commitverify"
22 "tangled.org/core/appview/config"
23 "tangled.org/core/appview/db"
24 "tangled.org/core/appview/models"
25 "tangled.org/core/appview/oauth"
26 "tangled.org/core/appview/pages/markup"
27 "tangled.org/core/appview/pages/repoinfo"
28 "tangled.org/core/appview/pagination"
29 "tangled.org/core/idresolver"
30 "tangled.org/core/types"
31
32 "github.com/bluesky-social/indigo/atproto/identity"
33 "github.com/bluesky-social/indigo/atproto/syntax"
34 "github.com/go-git/go-git/v5/plumbing"
35)
36
37//go:embed templates/* static legal
38var Files embed.FS
39
40type Pages struct {
41 mu sync.RWMutex
42 cache *TmplCache[string, *template.Template]
43
44 avatar config.AvatarConfig
45 pdsCfg config.PdsConfig
46 resolver *idresolver.Resolver
47 db *db.DB
48 rdb *cache.Cache
49 dev bool
50 embedFS fs.FS
51 templateDir string // Path to templates on disk for dev mode
52 rctx *markup.RenderContext
53 logger *slog.Logger
54}
55
56func NewPages(config *config.Config, res *idresolver.Resolver, database *db.DB, rdb *cache.Cache, logger *slog.Logger) *Pages {
57 // initialized with safe defaults, can be overridden per use
58 rctx := &markup.RenderContext{
59 IsDev: config.Core.Dev,
60 Hostname: config.Core.AppviewHost,
61 CamoUrl: config.Camo.Host,
62 CamoSecret: config.Camo.SharedSecret,
63 Sanitizer: markup.NewSanitizer(),
64 Files: Files,
65 }
66
67 p := &Pages{
68 mu: sync.RWMutex{},
69 cache: NewTmplCache[string, *template.Template](),
70 dev: config.Core.Dev,
71 avatar: config.Avatar,
72 pdsCfg: config.Pds,
73 rctx: rctx,
74 resolver: res,
75 db: database,
76 rdb: rdb,
77 templateDir: "appview/pages",
78 logger: logger,
79 }
80
81 if p.dev {
82 p.embedFS = os.DirFS(p.templateDir)
83 } else {
84 p.embedFS = Files
85 }
86
87 return p
88}
89
90// reverse of pathToName
91func (p *Pages) nameToPath(s string) string {
92 return "templates/" + s + ".html"
93}
94
95// FuncMap returns the template function map for use by external template consumers.
96func (p *Pages) FuncMap() template.FuncMap {
97 return p.funcMap()
98}
99
100// FragmentPaths returns all fragment template paths from the embedded FS.
101func (p *Pages) FragmentPaths() ([]string, error) {
102 return p.fragmentPaths()
103}
104
105// EmbedFS returns the embedded filesystem containing templates and static assets.
106func (p *Pages) EmbedFS() fs.FS {
107 return p.embedFS
108}
109
110// ParseWith parses the base layout together with all appview fragments and
111// an additional template from extraFS identified by extraPath (relative to
112// extraFS root). The returned template is ready to ExecuteTemplate with
113// "layouts/base" -- primarily for use with the blog.
114func (p *Pages) ParseWith(extraFS fs.FS, extraPath string) (*template.Template, error) {
115 fragmentPaths, err := p.fragmentPaths()
116 if err != nil {
117 return nil, err
118 }
119
120 funcs := p.funcMap()
121 tpl, err := template.New("layouts/base").
122 Funcs(funcs).
123 ParseFS(p.embedFS, append(fragmentPaths, p.nameToPath("layouts/base"))...)
124 if err != nil {
125 return nil, err
126 }
127
128 err = fs.WalkDir(extraFS, ".", func(path string, d fs.DirEntry, err error) error {
129 if err != nil {
130 return err
131 }
132 if d.IsDir() || !strings.HasSuffix(path, ".html") {
133 return nil
134 }
135 if path != extraPath && !strings.Contains(path, "fragments/") {
136 return nil
137 }
138 data, err := fs.ReadFile(extraFS, path)
139 if err != nil {
140 return err
141 }
142 if _, err = tpl.New(path).Parse(string(data)); err != nil {
143 return err
144 }
145 return nil
146 })
147 if err != nil {
148 return nil, err
149 }
150
151 return tpl, nil
152}
153
154func (p *Pages) fragmentPaths() ([]string, error) {
155 var fragmentPaths []string
156 err := fs.WalkDir(p.embedFS, "templates", func(path string, d fs.DirEntry, err error) error {
157 if err != nil {
158 return err
159 }
160 if d.IsDir() {
161 return nil
162 }
163 if !strings.HasSuffix(path, ".html") {
164 return nil
165 }
166 if !strings.Contains(path, "fragments/") {
167 return nil
168 }
169 fragmentPaths = append(fragmentPaths, path)
170 return nil
171 })
172 if err != nil {
173 return nil, err
174 }
175
176 return fragmentPaths, nil
177}
178
179// parse without memoization
180func (p *Pages) rawParse(stack ...string) (*template.Template, error) {
181 paths, err := p.fragmentPaths()
182 if err != nil {
183 return nil, err
184 }
185 for _, s := range stack {
186 paths = append(paths, p.nameToPath(s))
187 }
188
189 funcs := p.funcMap()
190 top := stack[len(stack)-1]
191 parsed, err := template.New(top).
192 Funcs(funcs).
193 ParseFS(p.embedFS, paths...)
194 if err != nil {
195 return nil, err
196 }
197
198 return parsed, nil
199}
200
201func (p *Pages) parse(stack ...string) (*template.Template, error) {
202 key := strings.Join(stack, "|")
203
204 // never cache in dev mode
205 if cached, exists := p.cache.Get(key); !p.dev && exists {
206 return cached, nil
207 }
208
209 result, err := p.rawParse(stack...)
210 if err != nil {
211 return nil, err
212 }
213
214 p.cache.Set(key, result)
215 return result, nil
216}
217
218func (p *Pages) parseBase(top string) (*template.Template, error) {
219 stack := []string{
220 "layouts/base",
221 top,
222 }
223 return p.parse(stack...)
224}
225
226func (p *Pages) parseRepoBase(top string) (*template.Template, error) {
227 stack := []string{
228 "layouts/base",
229 "layouts/repobase",
230 top,
231 }
232 return p.parse(stack...)
233}
234
235func (p *Pages) parseProfileBase(top string) (*template.Template, error) {
236 stack := []string{
237 "layouts/base",
238 "layouts/profilebase",
239 top,
240 }
241 return p.parse(stack...)
242}
243
244func (p *Pages) parseLoginBase(top string) (*template.Template, error) {
245 stack := []string{
246 "layouts/base",
247 "layouts/loginbase",
248 top,
249 }
250 return p.parse(stack...)
251}
252
253func (p *Pages) executePlain(name string, w io.Writer, params any) error {
254 tpl, err := p.parse(name)
255 if err != nil {
256 return err
257 }
258
259 return tpl.Execute(w, params)
260}
261
262func (p *Pages) executeLogin(name string, w io.Writer, params any) error {
263 tpl, err := p.parseLoginBase(name)
264 if err != nil {
265 return err
266 }
267
268 return tpl.ExecuteTemplate(w, "layouts/base", params)
269}
270
271func (p *Pages) execute(name string, w io.Writer, params any) error {
272 tpl, err := p.parseBase(name)
273 if err != nil {
274 return err
275 }
276
277 return tpl.ExecuteTemplate(w, "layouts/base", params)
278}
279
280func (p *Pages) executeRepo(name string, w io.Writer, params any) error {
281 tpl, err := p.parseRepoBase(name)
282 if err != nil {
283 return err
284 }
285
286 return tpl.ExecuteTemplate(w, "layouts/base", params)
287}
288
289func (p *Pages) executeProfile(name string, w io.Writer, params any) error {
290 tpl, err := p.parseProfileBase(name)
291 if err != nil {
292 return err
293 }
294
295 return tpl.ExecuteTemplate(w, "layouts/base", params)
296}
297
298type DollyParams struct {
299 Classes string
300 FillColor string
301 // Favicon embeds a prefers-color-scheme style block so the SVG
302 // adapts to dark mode when used as a standalone favicon document.
303 Favicon bool
304}
305
306func (p *Pages) Dolly(w io.Writer, params DollyParams) error {
307 return p.executePlain("fragments/dolly/logo", w, params)
308}
309
310func (p *Pages) Favicon(w io.Writer) error {
311 return p.Dolly(w, DollyParams{
312 Favicon: true,
313 })
314}
315
316type LoginParams struct {
317 ReturnUrl string
318 ErrorCode string
319 AddAccount bool
320 Accounts []oauth.AccountInfo
321}
322
323func (p *Pages) Login(w io.Writer, params LoginParams) error {
324 return p.executeLogin("user/login", w, params)
325}
326
327type SignupParams struct {
328 CloudflareSiteKey string
329 EmailId string
330}
331
332func (p *Pages) Signup(w io.Writer, params SignupParams) error {
333 return p.executeLogin("user/signup", w, params)
334}
335
336func (p *Pages) CompleteSignup(w io.Writer) error {
337 return p.executeLogin("user/completeSignup", w, nil)
338}
339
340type TermsOfServiceParams struct {
341 LoggedInUser *oauth.MultiAccountUser
342 Content template.HTML
343}
344
345func (p *Pages) TermsOfService(w io.Writer, params TermsOfServiceParams) error {
346 filename := "terms.md"
347 filePath := filepath.Join("legal", filename)
348
349 file, err := p.embedFS.Open(filePath)
350 if err != nil {
351 return fmt.Errorf("failed to read %s: %w", filename, err)
352 }
353 defer file.Close()
354
355 markdownBytes, err := io.ReadAll(file)
356 if err != nil {
357 return fmt.Errorf("failed to read %s: %w", filename, err)
358 }
359
360 rctx := p.rctx.Clone()
361 rctx.RendererType = markup.RendererTypeDefault
362 htmlString := rctx.RenderMarkdown(string(markdownBytes))
363 sanitized := rctx.SanitizeDefault(htmlString)
364 params.Content = template.HTML(sanitized)
365
366 return p.execute("legal/terms", w, params)
367}
368
369type PrivacyPolicyParams struct {
370 LoggedInUser *oauth.MultiAccountUser
371 Content template.HTML
372}
373
374func (p *Pages) PrivacyPolicy(w io.Writer, params PrivacyPolicyParams) error {
375 filename := "privacy.md"
376 filePath := filepath.Join("legal", filename)
377
378 file, err := p.embedFS.Open(filePath)
379 if err != nil {
380 return fmt.Errorf("failed to read %s: %w", filename, err)
381 }
382 defer file.Close()
383
384 markdownBytes, err := io.ReadAll(file)
385 if err != nil {
386 return fmt.Errorf("failed to read %s: %w", filename, err)
387 }
388
389 rctx := p.rctx.Clone()
390 rctx.RendererType = markup.RendererTypeDefault
391 htmlString := rctx.RenderMarkdown(string(markdownBytes))
392 sanitized := rctx.SanitizeDefault(htmlString)
393 params.Content = template.HTML(sanitized)
394
395 return p.execute("legal/privacy", w, params)
396}
397
398type BrandParams struct {
399 LoggedInUser *oauth.MultiAccountUser
400}
401
402func (p *Pages) Brand(w io.Writer, params BrandParams) error {
403 return p.execute("brand/brand", w, params)
404}
405
406type RecentItem struct {
407 Link *models.RecentLink
408 Repo *models.Repo
409 Issue *models.Issue
410 Pull *models.Pull
411}
412
413type BlogPost struct {
414 Slug string
415 Title string
416 Subtitle string
417 Date time.Time
418}
419
420type TimelineParams struct {
421 LoggedInUser *oauth.MultiAccountUser
422 Timeline []models.TimelineGroup
423 Repos []models.Repo
424 GfiLabel *models.LabelDefinition
425 BlueskyPosts []models.BskyPost
426 VouchSuggestions []models.VouchSuggestion
427 Notifications []*models.NotificationWithEntity
428 Recents []RecentItem
429 FollowingOnly bool
430 RecentBlogPosts []BlogPost
431 // ShowNewsletter controls whether the newsletter widget/CTA is rendered.
432 // For logged-in users it reflects their newsletter_preferences row; for
433 // anonymous visitors it is always true (dismissal falls back to
434 // localStorage on the client).
435 ShowNewsletter bool
436}
437
438func (p *Pages) Timeline(w io.Writer, params TimelineParams) error {
439 return p.execute("timeline/timeline", w, params)
440}
441
442type GoodFirstIssuesParams struct {
443 LoggedInUser *oauth.MultiAccountUser
444 Issues []models.Issue
445 RepoGroups []*models.RepoGroup
446 LabelDefs map[string]*models.LabelDefinition
447 GfiLabel *models.LabelDefinition
448 Page pagination.Page
449}
450
451func (p *Pages) GoodFirstIssues(w io.Writer, params GoodFirstIssuesParams) error {
452 return p.execute("goodfirstissues/index", w, params)
453}
454
455type UserProfileSettingsParams struct {
456 LoggedInUser *oauth.MultiAccountUser
457 Tab string
458 PunchcardPreference models.PunchcardPreference
459 IsTnglSh bool
460 IsDeactivated bool
461 HandleOpen bool
462}
463
464func (p *Pages) UserProfileSettings(w io.Writer, params UserProfileSettingsParams) error {
465 params.Tab = "profile"
466 return p.execute("user/settings/profile", w, params)
467}
468
469type GroupedNotifications struct {
470 Today []*models.NotificationWithEntity
471 ThisWeek []*models.NotificationWithEntity
472 Older []*models.NotificationWithEntity
473}
474
475func GroupNotificationsByDate(notifs []*models.NotificationWithEntity) GroupedNotifications {
476 now := time.Now()
477 todayStart := time.Date(now.Year(), now.Month(), now.Day(), 0, 0, 0, 0, now.Location())
478 weekStart := todayStart.AddDate(0, 0, -6)
479
480 var g GroupedNotifications
481 for _, n := range notifs {
482 switch {
483 case !n.Created.Before(todayStart):
484 g.Today = append(g.Today, n)
485 case !n.Created.Before(weekStart):
486 g.ThisWeek = append(g.ThisWeek, n)
487 default:
488 g.Older = append(g.Older, n)
489 }
490 }
491 return g
492}
493
494type NotificationsParams struct {
495 LoggedInUser *oauth.MultiAccountUser
496 WorkGroups GroupedNotifications
497 SocialGroups GroupedNotifications
498 MobileGroups GroupedNotifications
499 WorkUnreadCount int64
500 SocialUnreadCount int64
501 Page pagination.Page
502 Total int
503 ReadFilter string // "inbox" or "unread"
504 CategoryFilter string // "all", "work", "social"
505}
506
507func (p *Pages) Notifications(w io.Writer, params NotificationsParams) error {
508 return p.execute("notifications/list", w, params)
509}
510
511func (p *Pages) NotificationItem(w io.Writer, notif *models.NotificationWithEntity) error {
512 return p.executePlain("notifications/fragments/item", w, notif)
513}
514
515type NotificationCountParams struct {
516 Count int64
517}
518
519func (p *Pages) NotificationCount(w io.Writer, params NotificationCountParams) error {
520 return p.executePlain("notifications/fragments/count", w, params)
521}
522
523type NotificationPreviewParams struct {
524 LoggedInUser *oauth.MultiAccountUser
525 Notifications []*models.NotificationWithEntity
526 ReadFilter string
527 CategoryFilter string
528}
529
530func (p *Pages) NotificationPreview(w io.Writer, params NotificationPreviewParams) error {
531 return p.executePlain("notifications/fragments/preview", w, params)
532}
533
534type UserKeysSettingsParams struct {
535 LoggedInUser *oauth.MultiAccountUser
536 PubKeys []models.PublicKey
537 Tab string
538}
539
540func (p *Pages) UserKeysSettings(w io.Writer, params UserKeysSettingsParams) error {
541 params.Tab = "keys"
542 return p.execute("user/settings/keys", w, params)
543}
544
545type UserEmailsSettingsParams struct {
546 LoggedInUser *oauth.MultiAccountUser
547 Emails []models.Email
548 Tab string
549}
550
551func (p *Pages) UserEmailsSettings(w io.Writer, params UserEmailsSettingsParams) error {
552 params.Tab = "emails"
553 return p.execute("user/settings/emails", w, params)
554}
555
556type UserNotificationSettingsParams struct {
557 LoggedInUser *oauth.MultiAccountUser
558 Preferences *models.NotificationPreferences
559 Tab string
560}
561
562func (p *Pages) UserNotificationSettings(w io.Writer, params UserNotificationSettingsParams) error {
563 params.Tab = "notifications"
564 return p.execute("user/settings/notifications", w, params)
565}
566
567type UserSiteSettingsParams struct {
568 LoggedInUser *oauth.MultiAccountUser
569 Claim *models.DomainClaim
570 SitesDomain string
571 IsTnglHandle bool
572 Tab string
573}
574
575func (p *Pages) UserSiteSettings(w io.Writer, params UserSiteSettingsParams) error {
576 params.Tab = "sites"
577 return p.execute("user/settings/sites", w, params)
578}
579
580type UpgradeBannerParams struct {
581 Registrations []models.Registration
582 Spindles []models.Spindle
583}
584
585func (p *Pages) UpgradeBanner(w io.Writer, params UpgradeBannerParams) error {
586 return p.executePlain("banner", w, params)
587}
588
589type NewsletterResponseParams struct {
590 // Id identifies the calling form instance; the response span's id will
591 // be "newsletter-msg-<Id>" so it round-trips with the form's hx-target.
592 Id string
593 // Error, when non-empty, switches the template to the error variant.
594 Error string
595}
596
597func (p *Pages) NewsletterResponse(w io.Writer, params NewsletterResponseParams) error {
598 return p.executePlain("timeline/fragments/newsletterResponse", w, params)
599}
600
601type KnotsParams struct {
602 LoggedInUser *oauth.MultiAccountUser
603 Knots []KnotListingParams
604 Tab string
605}
606
607func (p *Pages) Knots(w io.Writer, params KnotsParams) error {
608 params.Tab = "knots"
609 return p.execute("knots/index", w, params)
610}
611
612type KnotParams struct {
613 LoggedInUser *oauth.MultiAccountUser
614 Registration *models.Registration
615 Members []string
616 Repos map[string][]models.Repo
617 IsOwner bool
618 RepoCount int
619 Tab string
620}
621
622func (p *Pages) Knot(w io.Writer, params KnotParams) error {
623 return p.execute("knots/dashboard", w, params)
624}
625
626type KnotListingParams struct {
627 *models.Registration
628 RepoCount int
629}
630
631func (p *Pages) KnotListing(w io.Writer, params KnotListingParams) error {
632 return p.executePlain("knots/fragments/knotListing", w, params)
633}
634
635type SpindlesParams struct {
636 LoggedInUser *oauth.MultiAccountUser
637 Spindles []models.Spindle
638 Tab string
639}
640
641func (p *Pages) Spindles(w io.Writer, params SpindlesParams) error {
642 params.Tab = "spindles"
643 return p.execute("spindles/index", w, params)
644}
645
646type SpindleListingParams struct {
647 models.Spindle
648 Tab string
649}
650
651func (p *Pages) SpindleListing(w io.Writer, params SpindleListingParams) error {
652 return p.executePlain("spindles/fragments/spindleListing", w, params)
653}
654
655type SpindleDashboardParams struct {
656 LoggedInUser *oauth.MultiAccountUser
657 Spindle models.Spindle
658 Members []string
659 Repos map[string][]models.Repo
660 Tab string
661}
662
663func (p *Pages) SpindleDashboard(w io.Writer, params SpindleDashboardParams) error {
664 return p.execute("spindles/dashboard", w, params)
665}
666
667type NewRepoParams struct {
668 LoggedInUser *oauth.MultiAccountUser
669 Knots []string
670}
671
672func (p *Pages) NewRepo(w io.Writer, params NewRepoParams) error {
673 return p.execute("repo/new", w, params)
674}
675
676type ForkRepoParams struct {
677 LoggedInUser *oauth.MultiAccountUser
678 Knots []string
679 RepoInfo repoinfo.RepoInfo
680}
681
682func (p *Pages) ForkRepo(w io.Writer, params ForkRepoParams) error {
683 return p.execute("repo/fork", w, params)
684}
685
686type ProfileCard struct {
687 UserDid string
688 HasProfile bool
689 FollowStatus models.FollowStatus
690 VouchRelationship *models.VouchRelationship
691 Punchcard *models.Punchcard
692 Profile *models.Profile
693 Stats ProfileStats
694 Active string
695}
696
697type ProfileStats struct {
698 RepoCount int64
699 StarredCount int64
700 StringCount int64
701 FollowersCount int64
702 FollowingCount int64
703}
704
705func (p *ProfileCard) GetTabs() [][]any {
706 tabs := [][]any{
707 {"overview", "overview", "square-chart-gantt", nil},
708 {"repos", "repos", "book-marked", p.Stats.RepoCount},
709 {"starred", "starred", "star", p.Stats.StarredCount},
710 {"strings", "strings", "line-squiggle", p.Stats.StringCount},
711 {"vouches", "vouches", "shield", nil},
712 }
713
714 return tabs
715}
716
717type ProfileOverviewParams struct {
718 LoggedInUser *oauth.MultiAccountUser
719 Repos []models.Repo
720 CollaboratingRepos []models.Repo
721 ProfileTimeline *models.ProfileTimeline
722 Card *ProfileCard
723 Active string
724 ShowPunchcard bool
725}
726
727func (p *Pages) ProfileOverview(w io.Writer, params ProfileOverviewParams) error {
728 params.Active = "overview"
729 return p.executeProfile("user/overview", w, params)
730}
731
732type ProfileReposParams struct {
733 LoggedInUser *oauth.MultiAccountUser
734 Repos []models.Repo
735 StarStatuses map[string]bool
736 Card *ProfileCard
737 Active string
738 Page pagination.Page
739 RepoCount int
740 FilterQuery string
741}
742
743func (p *Pages) ProfileRepos(w io.Writer, params ProfileReposParams) error {
744 params.Active = "repos"
745 return p.executeProfile("user/repos", w, params)
746}
747
748type ProfileStarredParams struct {
749 LoggedInUser *oauth.MultiAccountUser
750 Repos []models.Repo
751 Card *ProfileCard
752 Page pagination.Page
753 Total int
754 Active string
755}
756
757func (p *Pages) ProfileStarred(w io.Writer, params ProfileStarredParams) error {
758 params.Active = "starred"
759 return p.executeProfile("user/starred", w, params)
760}
761
762type ProfileStringsParams struct {
763 LoggedInUser *oauth.MultiAccountUser
764 Strings []models.String
765 Card *ProfileCard
766 Active string
767}
768
769func (p *Pages) ProfileStrings(w io.Writer, params ProfileStringsParams) error {
770 params.Active = "strings"
771 return p.executeProfile("user/strings", w, params)
772}
773
774type ProfileVouchesParams struct {
775 LoggedInUser *oauth.MultiAccountUser
776 Vouches []models.Vouch
777 Suggestions []models.VouchSuggestion
778 Card *ProfileCard
779 Page pagination.Page
780 VouchCount int
781 Active string
782 EvidencePulls map[syntax.ATURI]*models.Pull
783 EvidenceIssues map[syntax.ATURI]*models.Issue
784}
785
786func (p *Pages) ProfileVouches(w io.Writer, params ProfileVouchesParams) error {
787 params.Active = "vouches"
788 return p.executeProfile("user/vouches", w, params)
789}
790
791type FollowCard struct {
792 UserDid string
793 LoggedInUser *oauth.MultiAccountUser
794 FollowStatus models.FollowStatus
795 FollowersCount int64
796 FollowingCount int64
797 Profile *models.Profile
798}
799
800type ProfileFollowersParams struct {
801 LoggedInUser *oauth.MultiAccountUser
802 Followers []FollowCard
803 Card *ProfileCard
804 Active string
805}
806
807func (p *Pages) ProfileFollowers(w io.Writer, params ProfileFollowersParams) error {
808 params.Active = "overview"
809 return p.executeProfile("user/followers", w, params)
810}
811
812type ProfileFollowingParams struct {
813 LoggedInUser *oauth.MultiAccountUser
814 Following []FollowCard
815 Card *ProfileCard
816 Active string
817}
818
819func (p *Pages) ProfileFollowing(w io.Writer, params ProfileFollowingParams) error {
820 params.Active = "overview"
821 return p.executeProfile("user/following", w, params)
822}
823
824type FollowFragmentParams struct {
825 UserDid string
826 FollowStatus models.FollowStatus
827 FollowersCount int64
828}
829
830func (p *Pages) FollowFragment(w io.Writer, params FollowFragmentParams) error {
831 return p.executePlain("user/fragments/follow-oob", w, params)
832}
833
834type ProfilePopoverParams struct {
835 LoggedInUser *oauth.MultiAccountUser
836 UserDid string
837 Profile *models.Profile
838 FollowStatus models.FollowStatus
839 VouchRelationship *models.VouchRelationship
840 Stats ProfilePopoverStats
841}
842
843type ProfilePopoverStats struct {
844 FollowersCount int64
845 FollowingCount int64
846}
847
848func (p *Pages) ProfilePopoverFragment(w io.Writer, params ProfilePopoverParams) error {
849 return p.executePlain("user/fragments/profilePopover", w, params)
850}
851
852type EditBioParams struct {
853 LoggedInUser *oauth.MultiAccountUser
854 Profile *models.Profile
855 AlsoKnownAs []string
856}
857
858func (p *Pages) EditBioFragment(w io.Writer, params EditBioParams) error {
859 return p.executePlain("user/fragments/editBio", w, params)
860}
861
862type EditPinsParams struct {
863 LoggedInUser *oauth.MultiAccountUser
864 Profile *models.Profile
865 AllRepos []PinnedRepo
866}
867
868type PinnedRepo struct {
869 IsPinned bool
870 models.Repo
871}
872
873func (p *Pages) EditPinsFragment(w io.Writer, params EditPinsParams) error {
874 return p.executePlain("user/fragments/editPins", w, params)
875}
876
877type StarBtnFragmentParams struct {
878 IsStarred bool
879 SubjectAt syntax.ATURI
880 StarCount int
881 RepoName string
882 HxSwapOob bool
883}
884
885func (p *Pages) StarBtnFragment(w io.Writer, params StarBtnFragmentParams) error {
886 params.HxSwapOob = true
887 return p.executePlain("fragments/starBtn", w, params)
888}
889
890type RepoIndexParams struct {
891 LoggedInUser *oauth.MultiAccountUser
892 RepoInfo repoinfo.RepoInfo
893 Active string
894 TagMap map[string][]string
895 CommitsTrunc []types.Commit
896 TagsTrunc []*types.TagReference
897 BranchesTrunc []types.Branch
898 // ForkInfo *types.ForkInfo
899 HTMLReadme template.HTML
900 Raw bool
901 EmailToDid map[string]string
902 VerifiedCommits commitverify.VerifiedCommits
903 Languages []types.RepoLanguageDetails
904 Pipelines map[string]models.Pipeline
905 NeedsKnotUpgrade bool
906 KnotUnreachable bool
907 types.RepoIndexResponse
908}
909
910func (p *Pages) RepoIndexPage(w io.Writer, params RepoIndexParams) error {
911 params.Active = "overview"
912 if params.IsEmpty {
913 return p.executeRepo("repo/empty", w, params)
914 }
915
916 if params.NeedsKnotUpgrade {
917 return p.executeRepo("repo/needsUpgrade", w, params)
918 }
919
920 if params.KnotUnreachable {
921 return p.executeRepo("repo/knotUnreachable", w, params)
922 }
923
924 rctx := p.rctx.Clone()
925 rctx.RepoInfo = params.RepoInfo
926 rctx.RepoInfo.Ref = params.Ref
927 rctx.RendererType = markup.RendererTypeRepoMarkdown
928
929 if params.ReadmeFileName != "" {
930 switch markup.GetFormat(params.ReadmeFileName) {
931 case markup.FormatMarkdown:
932 params.Raw = false
933 htmlString := rctx.RenderMarkdown(params.Readme)
934 sanitized := rctx.SanitizeDefault(htmlString)
935 params.HTMLReadme = template.HTML(sanitized)
936 default:
937 params.Raw = true
938 }
939 }
940
941 return p.executeRepo("repo/index", w, params)
942}
943
944type RepoLogParams struct {
945 LoggedInUser *oauth.MultiAccountUser
946 RepoInfo repoinfo.RepoInfo
947 TagMap map[string][]string
948 Active string
949 EmailToDid map[string]string
950 VerifiedCommits commitverify.VerifiedCommits
951 Pipelines map[string]models.Pipeline
952
953 types.RepoLogResponse
954}
955
956func (p *Pages) RepoLog(w io.Writer, params RepoLogParams) error {
957 params.Active = "overview"
958 return p.executeRepo("repo/log", w, params)
959}
960
961type RepoCommitParams struct {
962 LoggedInUser *oauth.MultiAccountUser
963 RepoInfo repoinfo.RepoInfo
964 Active string
965 EmailToDid map[string]string
966 Pipeline *models.Pipeline
967 DiffOpts types.DiffOpts
968
969 // singular because it's always going to be just one
970 VerifiedCommit commitverify.VerifiedCommits
971
972 types.RepoCommitResponse
973}
974
975func (p *Pages) RepoCommit(w io.Writer, params RepoCommitParams) error {
976 params.Active = "overview"
977 return p.executeRepo("repo/commit", w, params)
978}
979
980type RepoTreeParams struct {
981 LoggedInUser *oauth.MultiAccountUser
982 RepoInfo repoinfo.RepoInfo
983 Active string
984 BreadCrumbs [][]string
985 Path string
986 Raw bool
987 HTMLReadme template.HTML
988 EmailToDid map[string]string
989 LastCommitInfo *types.LastCommitInfo
990 types.RepoTreeResponse
991}
992
993type RepoTreeStats struct {
994 NumFolders uint64
995 NumFiles uint64
996}
997
998func (r RepoTreeParams) TreeStats() RepoTreeStats {
999 numFolders, numFiles := 0, 0
1000 for _, f := range r.Files {
1001 if !f.IsFile() {
1002 numFolders += 1
1003 } else if f.IsFile() {
1004 numFiles += 1
1005 }
1006 }
1007
1008 return RepoTreeStats{
1009 NumFolders: uint64(numFolders),
1010 NumFiles: uint64(numFiles),
1011 }
1012}
1013
1014func (p *Pages) RepoTree(w io.Writer, params RepoTreeParams) error {
1015 params.Active = "overview"
1016
1017 rctx := p.rctx.Clone()
1018 rctx.RepoInfo = params.RepoInfo
1019 rctx.RepoInfo.Ref = params.Ref
1020 rctx.RendererType = markup.RendererTypeRepoMarkdown
1021
1022 if params.ReadmeFileName != "" {
1023 switch markup.GetFormat(params.ReadmeFileName) {
1024 case markup.FormatMarkdown:
1025 params.Raw = false
1026 htmlString := rctx.RenderMarkdown(params.Readme)
1027 sanitized := rctx.SanitizeDefault(htmlString)
1028 params.HTMLReadme = template.HTML(sanitized)
1029 default:
1030 params.Raw = true
1031 }
1032 }
1033
1034 return p.executeRepo("repo/tree", w, params)
1035}
1036
1037type RepoBranchesParams struct {
1038 LoggedInUser *oauth.MultiAccountUser
1039 RepoInfo repoinfo.RepoInfo
1040 Active string
1041 types.RepoBranchesResponse
1042}
1043
1044func (p *Pages) RepoBranches(w io.Writer, params RepoBranchesParams) error {
1045 params.Active = "overview"
1046 return p.executeRepo("repo/branches", w, params)
1047}
1048
1049type RepoTagsParams struct {
1050 LoggedInUser *oauth.MultiAccountUser
1051 RepoInfo repoinfo.RepoInfo
1052 Active string
1053 types.RepoTagsResponse
1054 ArtifactMap map[plumbing.Hash][]models.Artifact
1055 DanglingArtifacts []models.Artifact
1056}
1057
1058func (p *Pages) RepoTags(w io.Writer, params RepoTagsParams) error {
1059 params.Active = "overview"
1060 return p.executeRepo("repo/tags", w, params)
1061}
1062
1063type RepoTagParams struct {
1064 LoggedInUser *oauth.MultiAccountUser
1065 RepoInfo repoinfo.RepoInfo
1066 Active string
1067 types.RepoTagResponse
1068 ArtifactMap map[plumbing.Hash][]models.Artifact
1069 DanglingArtifacts []models.Artifact
1070}
1071
1072func (p *Pages) RepoTag(w io.Writer, params RepoTagParams) error {
1073 params.Active = "overview"
1074 return p.executeRepo("repo/tag", w, params)
1075}
1076
1077type RepoArtifactParams struct {
1078 LoggedInUser *oauth.MultiAccountUser
1079 RepoInfo repoinfo.RepoInfo
1080 Artifact models.Artifact
1081}
1082
1083func (p *Pages) RepoArtifactFragment(w io.Writer, params RepoArtifactParams) error {
1084 return p.executePlain("repo/fragments/artifact", w, params)
1085}
1086
1087type RepoBlobParams struct {
1088 LoggedInUser *oauth.MultiAccountUser
1089 RepoInfo repoinfo.RepoInfo
1090 Active string
1091 BreadCrumbs [][]string
1092 BlobView models.BlobView
1093 EmailToDid map[string]string
1094 LastCommitInfo *types.LastCommitInfo
1095 *tangled.RepoBlob_Output
1096}
1097
1098func (p *Pages) RepoBlob(w io.Writer, params RepoBlobParams) error {
1099 params.Active = "overview"
1100 return p.executeRepo("repo/blob", w, params)
1101}
1102
1103type Collaborator struct {
1104 Did string
1105 Role string
1106}
1107
1108type RepoSettingsParams struct {
1109 LoggedInUser *oauth.MultiAccountUser
1110 RepoInfo repoinfo.RepoInfo
1111 Collaborators []Collaborator
1112 Active string
1113 Branches []types.Branch
1114 Spindles []string
1115 CurrentSpindle string
1116 Secrets []*tangled.RepoListSecrets_Secret
1117
1118 // TODO: use repoinfo.roles
1119 IsCollaboratorInviteAllowed bool
1120}
1121
1122func (p *Pages) RepoSettings(w io.Writer, params RepoSettingsParams) error {
1123 params.Active = "settings"
1124 return p.executeRepo("repo/settings", w, params)
1125}
1126
1127type RepoGeneralSettingsParams struct {
1128 LoggedInUser *oauth.MultiAccountUser
1129 RepoInfo repoinfo.RepoInfo
1130 Labels []models.LabelDefinition
1131 DefaultLabels []models.LabelDefinition
1132 SubscribedLabels map[string]struct{}
1133 ShouldSubscribeAll bool
1134 Active string
1135 Tab string
1136 Branches []types.Branch
1137}
1138
1139func (p *Pages) RepoGeneralSettings(w io.Writer, params RepoGeneralSettingsParams) error {
1140 params.Active = "settings"
1141 params.Tab = "general"
1142 return p.executeRepo("repo/settings/general", w, params)
1143}
1144
1145type RepoAccessSettingsParams struct {
1146 LoggedInUser *oauth.MultiAccountUser
1147 RepoInfo repoinfo.RepoInfo
1148 Active string
1149 Tab string
1150 Collaborators []Collaborator
1151 CanRemoveCollaborator bool
1152}
1153
1154func (p *Pages) RepoAccessSettings(w io.Writer, params RepoAccessSettingsParams) error {
1155 params.Active = "settings"
1156 params.Tab = "access"
1157 return p.executeRepo("repo/settings/access", w, params)
1158}
1159
1160type RepoPipelineSettingsParams struct {
1161 LoggedInUser *oauth.MultiAccountUser
1162 RepoInfo repoinfo.RepoInfo
1163 Active string
1164 Tab string
1165 Spindles []string
1166 CurrentSpindle string
1167 Secrets []map[string]any
1168}
1169
1170func (p *Pages) RepoPipelineSettings(w io.Writer, params RepoPipelineSettingsParams) error {
1171 params.Active = "settings"
1172 params.Tab = "pipelines"
1173 return p.executeRepo("repo/settings/pipelines", w, params)
1174}
1175
1176type RepoWebhooksSettingsParams struct {
1177 LoggedInUser *oauth.MultiAccountUser
1178 RepoInfo repoinfo.RepoInfo
1179 Active string
1180 Tab string
1181 Webhooks []models.Webhook
1182 WebhookDeliveries map[int64][]models.WebhookDelivery
1183}
1184
1185func (p *Pages) RepoWebhooksSettings(w io.Writer, params RepoWebhooksSettingsParams) error {
1186 params.Active = "settings"
1187 params.Tab = "hooks"
1188 return p.executeRepo("repo/settings/hooks", w, params)
1189}
1190
1191type WebhookDeliveriesListParams struct {
1192 LoggedInUser *oauth.MultiAccountUser
1193 RepoInfo repoinfo.RepoInfo
1194 Webhook *models.Webhook
1195 Deliveries []models.WebhookDelivery
1196}
1197
1198func (p *Pages) WebhookDeliveriesList(w io.Writer, params WebhookDeliveriesListParams) error {
1199 tpl, err := p.parse("repo/settings/fragments/webhookDeliveries")
1200 if err != nil {
1201 return err
1202 }
1203 return tpl.ExecuteTemplate(w, "repo/settings/fragments/webhookDeliveries", params)
1204}
1205
1206type RepoSiteSettingsParams struct {
1207 LoggedInUser *oauth.MultiAccountUser
1208 RepoInfo repoinfo.RepoInfo
1209 Active string
1210 Tab string
1211 Branches []types.Branch
1212 SiteConfig *models.RepoSite
1213 OwnerClaim *models.DomainClaim
1214 Deploys []models.SiteDeploy
1215 IndexSiteTakenBy string // repo_at of another repo that already holds is_index, or ""
1216}
1217
1218func (p *Pages) RepoSiteSettings(w io.Writer, params RepoSiteSettingsParams) error {
1219 params.Active = "settings"
1220 params.Tab = "sites"
1221 return p.executeRepo("repo/settings/sites", w, params)
1222}
1223
1224type RepoIssuesParams struct {
1225 LoggedInUser *oauth.MultiAccountUser
1226 RepoInfo repoinfo.RepoInfo
1227 Active string
1228 Issues []models.Issue
1229 IssueCount int
1230 LabelDefs map[string]*models.LabelDefinition
1231 Page pagination.Page
1232 FilterState string
1233 FilterQuery string
1234 BaseFilterQuery string
1235 VouchRelationships map[syntax.DID]*models.VouchRelationship
1236}
1237
1238func (p *Pages) RepoIssues(w io.Writer, params RepoIssuesParams) error {
1239 params.Active = "issues"
1240 return p.executeRepo("repo/issues/issues", w, params)
1241}
1242
1243type RepoSingleIssueParams struct {
1244 LoggedInUser *oauth.MultiAccountUser
1245 RepoInfo repoinfo.RepoInfo
1246 Active string
1247 Issue *models.Issue
1248 CommentList []models.CommentListItem
1249 Backlinks []models.RichReferenceLink
1250 LabelDefs map[string]*models.LabelDefinition
1251
1252 Reactions map[syntax.ATURI]map[models.ReactionKind]models.ReactionDisplayData
1253 UserReacted map[syntax.ATURI]map[models.ReactionKind]bool
1254 VouchRelationships map[syntax.DID]*models.VouchRelationship
1255}
1256
1257func (p *Pages) RepoSingleIssue(w io.Writer, params RepoSingleIssueParams) error {
1258 params.Active = "issues"
1259 return p.executeRepo("repo/issues/issue", w, params)
1260}
1261
1262type EditIssueParams struct {
1263 LoggedInUser *oauth.MultiAccountUser
1264 RepoInfo repoinfo.RepoInfo
1265 Issue *models.Issue
1266 Action string
1267}
1268
1269func (p *Pages) EditIssueFragment(w io.Writer, params EditIssueParams) error {
1270 params.Action = "edit"
1271 return p.executePlain("repo/issues/fragments/putIssue", w, params)
1272}
1273
1274type ThreadReactionFragmentParams struct {
1275 Kind models.ReactionKind
1276 Count int
1277 Users []string
1278 IsReacted bool
1279}
1280
1281func (p *Pages) ThreadReactionFragment(w io.Writer, params ThreadReactionFragmentParams) error {
1282 return p.executePlain("repo/fragments/reaction", w, params)
1283}
1284
1285type RepoNewIssueParams struct {
1286 LoggedInUser *oauth.MultiAccountUser
1287 RepoInfo repoinfo.RepoInfo
1288 Issue *models.Issue // existing issue if any -- passed when editing
1289 Active string
1290 Action string
1291}
1292
1293func (p *Pages) RepoNewIssue(w io.Writer, params RepoNewIssueParams) error {
1294 params.Active = "issues"
1295 params.Action = "create"
1296 return p.executeRepo("repo/issues/new", w, params)
1297}
1298
1299type StackedDiff struct {
1300 Diff *types.NiceDiff
1301 Opts types.DiffOpts
1302}
1303
1304type RepoNewPullParams struct {
1305 LoggedInUser *oauth.MultiAccountUser
1306 RepoInfo repoinfo.RepoInfo
1307 Branches []types.Branch
1308 SourceBranches []types.Branch
1309 ForkBranches []types.Branch
1310 Forks []models.Repo
1311 Source Source
1312 SourceBranch string
1313 TargetBranch string
1314 Fork string
1315 Patch string
1316 Title string
1317 Body string
1318 IsStacked bool
1319 Comparison *types.RepoFormatPatchResponse
1320 Diff *types.NiceDiff
1321 DiffOpts types.DiffOpts
1322 StackedDiffs []StackedDiff
1323 MergeCheck *types.MergeCheckResponse
1324 StackTitles map[string]string
1325 StackBodies map[string]string
1326 PrefillError string
1327 Active string
1328 LabelDefs map[string]*models.LabelDefinition
1329 LabelState models.LabelState
1330 StackLabelStates map[string]models.LabelState
1331}
1332
1333func (p *Pages) RepoNewPull(w io.Writer, params RepoNewPullParams) error {
1334 params.Active = "pulls"
1335 return p.executeRepo("repo/pulls/new", w, params)
1336}
1337
1338func (p *Pages) PullComposeHostFragment(w io.Writer, params RepoNewPullParams) error {
1339 return p.executePlain("repo/pulls/fragments/pullComposeHost", w, params)
1340}
1341
1342func (p *Pages) MarkdownPreviewFragment(w io.Writer, body string) error {
1343 return p.executePlain("fragments/markdownPreview", w, body)
1344}
1345
1346type RepoPullsParams struct {
1347 LoggedInUser *oauth.MultiAccountUser
1348 RepoInfo repoinfo.RepoInfo
1349 Pulls []*models.Pull
1350 Active string
1351 FilterState string
1352 FilterQuery string
1353 BaseFilterQuery string
1354 Stacks []models.Stack
1355 Pipelines map[string]models.Pipeline
1356 LabelDefs map[string]*models.LabelDefinition
1357 Page pagination.Page
1358 PullCount int
1359 VouchRelationships map[syntax.DID]*models.VouchRelationship
1360}
1361
1362func (p *Pages) RepoPulls(w io.Writer, params RepoPullsParams) error {
1363 params.Active = "pulls"
1364 return p.executeRepo("repo/pulls/pulls", w, params)
1365}
1366
1367type ResubmitResult uint64
1368
1369const (
1370 ShouldResubmit ResubmitResult = iota
1371 ShouldNotResubmit
1372 Unknown
1373)
1374
1375func (r ResubmitResult) Yes() bool {
1376 return r == ShouldResubmit
1377}
1378func (r ResubmitResult) No() bool {
1379 return r == ShouldNotResubmit
1380}
1381func (r ResubmitResult) Unknown() bool {
1382 return r == Unknown
1383}
1384
1385type RepoSinglePullParams struct {
1386 LoggedInUser *oauth.MultiAccountUser
1387 RepoInfo repoinfo.RepoInfo
1388 Active string
1389 Pull *models.Pull
1390 Stack models.Stack
1391 Backlinks []models.RichReferenceLink
1392 BranchDeleteStatus *models.BranchDeleteStatus
1393 MergeCheck types.MergeCheckResponse
1394 ResubmitCheck ResubmitResult
1395 Pipelines map[string]models.Pipeline
1396 Diff types.DiffRenderer
1397 DiffOpts types.DiffOpts
1398 ActiveRound int
1399 IsInterdiff bool
1400
1401 Reactions map[syntax.ATURI]map[models.ReactionKind]models.ReactionDisplayData
1402 UserReacted map[syntax.ATURI]map[models.ReactionKind]bool
1403
1404 LabelDefs map[string]*models.LabelDefinition
1405 VouchRelationships map[syntax.DID]*models.VouchRelationship
1406 VouchSkips map[syntax.DID]bool
1407}
1408
1409func (p *Pages) RepoSinglePull(w io.Writer, params RepoSinglePullParams) error {
1410 params.Active = "pulls"
1411 return p.executeRepo("repo/pulls/pull", w, params)
1412}
1413
1414type PullResubmitParams struct {
1415 LoggedInUser *oauth.MultiAccountUser
1416 RepoInfo repoinfo.RepoInfo
1417 Pull *models.Pull
1418 SubmissionId int
1419}
1420
1421func (p *Pages) PullResubmitFragment(w io.Writer, params PullResubmitParams) error {
1422 return p.executePlain("repo/pulls/fragments/pullResubmit", w, params)
1423}
1424
1425type PullActionsParams struct {
1426 LoggedInUser *oauth.MultiAccountUser
1427 RepoInfo repoinfo.RepoInfo
1428 Pull *models.Pull
1429 RoundNumber int
1430 MergeCheck types.MergeCheckResponse
1431 ResubmitCheck ResubmitResult
1432 BranchDeleteStatus *models.BranchDeleteStatus
1433 Stack models.Stack
1434}
1435
1436func (p *Pages) PullActionsFragment(w io.Writer, params PullActionsParams) error {
1437 return p.executePlain("repo/pulls/fragments/pullActions", w, params)
1438}
1439
1440type PullNewCommentParams struct {
1441 LoggedInUser *oauth.MultiAccountUser
1442 RepoInfo repoinfo.RepoInfo
1443 Pull *models.Pull
1444 RoundNumber int
1445}
1446
1447func (p *Pages) PullNewCommentFragment(w io.Writer, params PullNewCommentParams) error {
1448 return p.executePlain("repo/pulls/fragments/pullNewComment", w, params)
1449}
1450
1451type RepoCompareParams struct {
1452 LoggedInUser *oauth.MultiAccountUser
1453 RepoInfo repoinfo.RepoInfo
1454 Forks []models.Repo
1455 Branches []types.Branch
1456 Tags []*types.TagReference
1457 Base string
1458 Head string
1459 Diff *types.NiceDiff
1460 DiffOpts types.DiffOpts
1461
1462 Active string
1463}
1464
1465func (p *Pages) RepoCompare(w io.Writer, params RepoCompareParams) error {
1466 params.Active = "overview"
1467 return p.executeRepo("repo/compare/compare", w, params)
1468}
1469
1470type RepoCompareNewParams struct {
1471 LoggedInUser *oauth.MultiAccountUser
1472 RepoInfo repoinfo.RepoInfo
1473 Forks []models.Repo
1474 Branches []types.Branch
1475 Tags []*types.TagReference
1476 Base string
1477 Head string
1478
1479 Active string
1480}
1481
1482func (p *Pages) RepoCompareNew(w io.Writer, params RepoCompareNewParams) error {
1483 params.Active = "overview"
1484 return p.executeRepo("repo/compare/new", w, params)
1485}
1486
1487type RepoCompareAllowPullParams struct {
1488 LoggedInUser *oauth.MultiAccountUser
1489 RepoInfo repoinfo.RepoInfo
1490 Base string
1491 Head string
1492}
1493
1494func (p *Pages) RepoCompareAllowPullFragment(w io.Writer, params RepoCompareAllowPullParams) error {
1495 return p.executePlain("repo/fragments/compareAllowPull", w, params)
1496}
1497
1498type RepoCompareDiffFragmentParams struct {
1499 Diff types.NiceDiff
1500 DiffOpts types.DiffOpts
1501}
1502
1503func (p *Pages) RepoCompareDiffFragment(w io.Writer, params RepoCompareDiffFragmentParams) error {
1504 return p.executePlain("repo/fragments/diff", w, []any{¶ms.Diff, ¶ms.DiffOpts})
1505}
1506
1507type LabelPanelParams struct {
1508 LoggedInUser *oauth.MultiAccountUser
1509 RepoInfo repoinfo.RepoInfo
1510 Defs map[string]*models.LabelDefinition
1511 Subject string
1512 State models.LabelState
1513}
1514
1515func (p *Pages) LabelPanel(w io.Writer, params LabelPanelParams) error {
1516 return p.executePlain("repo/fragments/labelPanel", w, params)
1517}
1518
1519type EditLabelPanelParams struct {
1520 LoggedInUser *oauth.MultiAccountUser
1521 RepoInfo repoinfo.RepoInfo
1522 Defs map[string]*models.LabelDefinition
1523 Subject string
1524 State models.LabelState
1525 Prefix string
1526}
1527
1528func (p *Pages) EditLabelPanel(w io.Writer, params EditLabelPanelParams) error {
1529 return p.executePlain("repo/fragments/editLabelPanel", w, params)
1530}
1531
1532type RepoStarsParams struct {
1533 LoggedInUser *oauth.MultiAccountUser
1534 RepoInfo repoinfo.RepoInfo
1535 Active string
1536 Starrers []models.Star
1537 Page pagination.Page
1538 TotalCount int
1539}
1540
1541func (p *Pages) RepoStars(w io.Writer, params RepoStarsParams) error {
1542 params.Active = "overview"
1543 return p.executeRepo("repo/stars", w, params)
1544}
1545
1546type RepoForksParams struct {
1547 LoggedInUser *oauth.MultiAccountUser
1548 RepoInfo repoinfo.RepoInfo
1549 Active string
1550 Forks []models.Repo
1551 Page pagination.Page
1552 TotalCount int
1553}
1554
1555func (p *Pages) RepoForks(w io.Writer, params RepoForksParams) error {
1556 params.Active = "overview"
1557 return p.executeRepo("repo/forks", w, params)
1558}
1559
1560type PipelinesParams struct {
1561 LoggedInUser *oauth.MultiAccountUser
1562 RepoInfo repoinfo.RepoInfo
1563 Pipelines []models.Pipeline
1564 Active string
1565 FilterKind string
1566 Total int64
1567}
1568
1569func (p *Pages) Pipelines(w io.Writer, params PipelinesParams) error {
1570 params.Active = "pipelines"
1571 return p.executeRepo("repo/pipelines/pipelines", w, params)
1572}
1573
1574type LogBlockParams struct {
1575 Id int
1576 Name string
1577 Command string
1578 Collapsed bool
1579 StartTime time.Time
1580}
1581
1582func (p *Pages) LogBlock(w io.Writer, params LogBlockParams) error {
1583 return p.executePlain("repo/pipelines/fragments/logBlock", w, params)
1584}
1585
1586type LogBlockEndParams struct {
1587 Id int
1588 StartTime time.Time
1589 EndTime time.Time
1590}
1591
1592func (p *Pages) LogBlockEnd(w io.Writer, params LogBlockEndParams) error {
1593 return p.executePlain("repo/pipelines/fragments/logBlockEnd", w, params)
1594}
1595
1596type LogLineParams struct {
1597 Id int
1598 Content template.HTML
1599}
1600
1601func (p *Pages) LogLine(w io.Writer, params LogLineParams) error {
1602 return p.executePlain("repo/pipelines/fragments/logLine", w, params)
1603}
1604
1605type WorkflowSymbolOOBParams struct {
1606 Name string
1607 Statuses models.WorkflowStatus
1608}
1609
1610func (p *Pages) WorkflowSymbolOOB(w io.Writer, params WorkflowSymbolOOBParams) error {
1611 return p.executePlain("repo/pipelines/fragments/workflowSymbolOOB", w, params)
1612}
1613
1614type WorkflowParams struct {
1615 LoggedInUser *oauth.MultiAccountUser
1616 RepoInfo repoinfo.RepoInfo
1617 Pipeline models.Pipeline
1618 Workflow string
1619 LogUrl string
1620 Active string
1621}
1622
1623func (p *Pages) Workflow(w io.Writer, params WorkflowParams) error {
1624 params.Active = "pipelines"
1625 return p.executeRepo("repo/pipelines/workflow", w, params)
1626}
1627
1628type PutStringParams struct {
1629 LoggedInUser *oauth.MultiAccountUser
1630 Action string
1631
1632 // this is supplied in the case of editing an existing string
1633 String models.String
1634}
1635
1636func (p *Pages) PutString(w io.Writer, params PutStringParams) error {
1637 return p.execute("strings/put", w, params)
1638}
1639
1640type StringsDashboardParams struct {
1641 LoggedInUser *oauth.MultiAccountUser
1642 Card ProfileCard
1643 Strings []models.String
1644}
1645
1646func (p *Pages) StringsDashboard(w io.Writer, params StringsDashboardParams) error {
1647 return p.execute("strings/dashboard", w, params)
1648}
1649
1650type StringTimelineParams struct {
1651 LoggedInUser *oauth.MultiAccountUser
1652 Strings []models.String
1653}
1654
1655func (p *Pages) StringsTimeline(w io.Writer, params StringTimelineParams) error {
1656 return p.execute("strings/timeline", w, params)
1657}
1658
1659type SingleStringParams struct {
1660 LoggedInUser *oauth.MultiAccountUser
1661 ShowRendered bool
1662 RenderToggle bool
1663 RenderedContents template.HTML
1664 String *models.String
1665 Stats models.StringStats
1666 IsStarred bool
1667 StarCount int
1668 Owner identity.Identity
1669 CommentList []models.CommentListItem
1670
1671 Reactions map[syntax.ATURI]map[models.ReactionKind]models.ReactionDisplayData
1672 UserReacted map[syntax.ATURI]map[models.ReactionKind]bool
1673 VouchRelationships map[syntax.DID]*models.VouchRelationship
1674}
1675
1676func (p *Pages) SingleString(w io.Writer, params SingleStringParams) error {
1677 return p.execute("strings/string", w, params)
1678}
1679
1680type SearchReposParams struct {
1681 LoggedInUser *oauth.MultiAccountUser
1682 Repos []models.Repo
1683 Page pagination.Page
1684 ResultCount int
1685 FilterQuery string
1686 SortParam string
1687 TimeTaken time.Duration
1688 DocCount int64
1689}
1690
1691func (p *Pages) SearchRepos(w io.Writer, params SearchReposParams) error {
1692 return p.execute("search/search", w, params)
1693}
1694
1695type SearchQuickParams struct {
1696 Repos []models.Repo
1697 Query string
1698 Total int
1699}
1700
1701func (p *Pages) SearchQuick(w io.Writer, params SearchQuickParams) error {
1702 return p.executePlain("search/fragments/quick", w, params)
1703}
1704
1705func (p *Pages) SearchQuickMobile(w io.Writer, params SearchQuickParams) error {
1706 tpl, err := p.parse("search/fragments/quick")
1707 if err != nil {
1708 return err
1709 }
1710 return tpl.ExecuteTemplate(w, "search/fragments/quickMobile", params)
1711}
1712
1713func (p *Pages) Home(w io.Writer, params TimelineParams) error {
1714 return p.execute("timeline/home", w, params)
1715}
1716
1717type CommentBodyFragmentParams struct {
1718 Comment models.Comment
1719 Reactions map[models.ReactionKind]models.ReactionDisplayData
1720 UserReacted map[models.ReactionKind]bool
1721}
1722
1723func (p *Pages) CommentBodyFragment(w io.Writer, params CommentBodyFragmentParams) error {
1724 return p.executePlain("fragments/comment/commentBody", w, params)
1725}
1726
1727type EditCommentFragmentParams struct {
1728 Comment models.Comment
1729}
1730
1731func (p *Pages) EditCommentFragment(w io.Writer, params EditCommentFragmentParams) error {
1732 return p.executePlain("fragments/comment/edit", w, params)
1733}
1734
1735type ReplyCommentFragmentParams struct {
1736 LoggedInUser *oauth.MultiAccountUser
1737}
1738
1739func (p *Pages) ReplyCommentFragment(w io.Writer, params ReplyCommentFragmentParams) error {
1740 return p.executePlain("fragments/comment/reply", w, params)
1741}
1742
1743type ReplyPlaceholderFragmentParams struct {
1744 LoggedInUser *oauth.MultiAccountUser
1745}
1746
1747func (p *Pages) ReplyPlaceholderFragment(w io.Writer, params ReplyPlaceholderFragmentParams) error {
1748 return p.executePlain("fragments/comment/replyPlaceholder", w, params)
1749}
1750
1751func (p *Pages) Static() http.Handler {
1752 if p.dev {
1753 return http.StripPrefix("/static/", http.FileServer(http.Dir("appview/pages/static")))
1754 }
1755
1756 sub, err := fs.Sub(p.embedFS, "static")
1757 if err != nil {
1758 p.logger.Error("no static dir found? that's crazy", "err", err)
1759 panic(err)
1760 }
1761 // Custom handler to apply Cache-Control headers for font files
1762 return Cache(http.StripPrefix("/static/", http.FileServer(http.FS(sub))))
1763}
1764
1765func Cache(h http.Handler) http.Handler {
1766 return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
1767 path := strings.Split(r.URL.Path, "?")[0]
1768
1769 if strings.HasSuffix(path, ".css") {
1770 // on day for css files
1771 w.Header().Set("Cache-Control", "public, max-age=86400")
1772 } else {
1773 w.Header().Set("Cache-Control", "public, max-age=31536000, immutable")
1774 }
1775 h.ServeHTTP(w, r)
1776 })
1777}
1778
1779func (p *Pages) CssContentHash() string {
1780 cssFile, err := p.embedFS.Open("static/tw.css")
1781 if err != nil {
1782 slog.Debug("Error opening CSS file", "err", err)
1783 return ""
1784 }
1785 defer cssFile.Close()
1786
1787 hasher := sha256.New()
1788 if _, err := io.Copy(hasher, cssFile); err != nil {
1789 slog.Debug("Error hashing CSS file", "err", err)
1790 return ""
1791 }
1792
1793 return hex.EncodeToString(hasher.Sum(nil))[:8] // Use first 8 chars of hash
1794}
1795
1796func (p *Pages) DangerPasswordTokenStep(w io.Writer) error {
1797 return p.executePlain("user/settings/fragments/dangerPasswordToken", w, nil)
1798}
1799
1800func (p *Pages) DangerPasswordSuccess(w io.Writer) error {
1801 return p.executePlain("user/settings/fragments/dangerPasswordSuccess", w, nil)
1802}
1803
1804func (p *Pages) DangerDeleteTokenStep(w io.Writer) error {
1805 return p.executePlain("user/settings/fragments/dangerDeleteToken", w, nil)
1806}
1807
1808func (p *Pages) Error500(w io.Writer) error {
1809 return p.execute("errors/500", w, nil)
1810}
1811
1812func (p *Pages) Error404(w io.Writer) error {
1813 return p.execute("errors/404", w, nil)
1814}
1815
1816func (p *Pages) ErrorKnot404(w io.Writer) error {
1817 return p.execute("errors/knot404", w, nil)
1818}
1819
1820func (p *Pages) Error503(w io.Writer) error {
1821 return p.execute("errors/503", w, nil)
1822}