forked from
tangled.org/core
Monorepo for Tangled
1package config
2
3import (
4 "context"
5 "fmt"
6 "net/url"
7 "strings"
8 "time"
9
10 "github.com/sethvargo/go-envconfig"
11)
12
13type CoreConfig struct {
14 CookieSecret string `env:"COOKIE_SECRET, default=00000000000000000000000000000000"`
15 DbPath string `env:"DB_PATH, default=appview.db"`
16 ListenAddr string `env:"LISTEN_ADDR, default=0.0.0.0:3000"`
17 AppviewHost string `env:"APPVIEW_HOST, default=tangled.org"`
18 AppviewName string `env:"APPVIEW_Name, default=Tangled"`
19 Dev bool `env:"DEV, default=false"`
20 DisallowedNicknamesFile string `env:"DISALLOWED_NICKNAMES_FILE"`
21
22 // temporarily, to add users to default knot and spindle
23 AppPassword string `env:"APP_PASSWORD"`
24
25 // uhhhh this is because knot1 is under icy's did
26 TmpAltAppPassword string `env:"ALT_APP_PASSWORD"`
27}
28
29func (c *CoreConfig) UseTLS() bool {
30 return !c.Dev
31}
32
33func (c *CoreConfig) BaseUrl() string {
34 if c.UseTLS() {
35 return "https://" + c.AppviewHost
36 }
37 return "http://" + c.AppviewHost
38}
39
40type OAuthConfig struct {
41 ClientSecret string `env:"CLIENT_SECRET"`
42 ClientKid string `env:"CLIENT_KID"`
43}
44
45type PlcConfig struct {
46 PLCURL string `env:"URL, default=https://plc.directory"`
47}
48
49type KnotMirrorConfig struct {
50 Url string `env:"URL, default=https://mirror.tangled.network"`
51}
52
53type JetstreamConfig struct {
54 Endpoint string `env:"ENDPOINT, default=wss://jetstream1.us-east.bsky.network/subscribe"`
55}
56
57type ConsumerConfig struct {
58 RetryInterval time.Duration `env:"RETRY_INTERVAL, default=60s"`
59 MaxRetryInterval time.Duration `env:"MAX_RETRY_INTERVAL, default=120m"`
60 ConnectionTimeout time.Duration `env:"CONNECTION_TIMEOUT, default=5s"`
61 WorkerCount int `env:"WORKER_COUNT, default=64"`
62 QueueSize int `env:"QUEUE_SIZE, default=100"`
63}
64
65type ResendConfig struct {
66 ApiKey string `env:"API_KEY"`
67 SentFrom string `env:"SENT_FROM, default=noreply@notifs.tangled.sh"`
68 NewsletterSegmentId string `env:"NEWSLETTER_SEGMENT_ID"`
69}
70
71type CamoConfig struct {
72 Host string `env:"HOST, default=https://camo.tangled.sh"`
73 SharedSecret string `env:"SHARED_SECRET"`
74}
75
76type AvatarConfig struct {
77 Host string `env:"HOST, default=https://avatar.tangled.sh"`
78 SharedSecret string `env:"SHARED_SECRET"`
79}
80
81type PosthogConfig struct {
82 ApiKey string `env:"API_KEY"`
83 Endpoint string `env:"ENDPOINT, default=https://eu.i.posthog.com"`
84}
85
86type RedisConfig struct {
87 Addr string `env:"ADDR, default=localhost:6379"`
88 Password string `env:"PASS"`
89 DB int `env:"DB, default=0"`
90}
91
92type PdsConfig struct {
93 Host string `env:"HOST, default=https://tngl.sh"`
94 UserDomain string `env:"USER_DOMAIN, default=.tngl.sh"`
95 AdminSecret string `env:"ADMIN_SECRET"`
96}
97
98func (p *PdsConfig) IsTnglShUser(pdsHost string) bool {
99 return strings.TrimRight(pdsHost, "/") == strings.TrimRight(p.Host, "/")
100}
101
102type R2Config struct {
103 AccessKeyID string `env:"ACCESS_KEY_ID"`
104 SecretAccessKey string `env:"SECRET_ACCESS_KEY"`
105 Bucket string `env:"BUCKET, default=tangled-sites"`
106}
107
108type TurnstileConfig struct {
109 SiteKey string `env:"SITE_KEY"`
110 SecretKey string `env:"SECRET_KEY"`
111}
112
113type KVConfig struct {
114 NamespaceId string `env:"NAMESPACE_ID"`
115 ApiToken string `env:"API_TOKEN"`
116}
117
118type Cloudflare struct {
119 // Legacy top-level API token. For services like Workers KV, we
120 // now use a scoped Account API token configured under the relevant
121 // sub-struct.
122 ApiToken string `env:"API_TOKEN"`
123 ZoneId string `env:"ZONE_ID"`
124 AccountId string `env:"ACCOUNT_ID"`
125
126 KV KVConfig `env:",prefix=KV_"`
127 Turnstile TurnstileConfig `env:",prefix=TURNSTILE_"`
128 R2 R2Config `env:",prefix=R2_"`
129}
130
131type SitesConfig struct {
132 Domain string `env:"DOMAIN, default=tngl.io"`
133}
134
135type LabelConfig struct {
136 DefaultLabelDefs []string `env:"DEFAULTS, default=at://did:plc:wshs7t2adsemcrrd4snkeqli/sh.tangled.label.definition/wontfix,at://did:plc:wshs7t2adsemcrrd4snkeqli/sh.tangled.label.definition/good-first-issue,at://did:plc:wshs7t2adsemcrrd4snkeqli/sh.tangled.label.definition/duplicate,at://did:plc:wshs7t2adsemcrrd4snkeqli/sh.tangled.label.definition/documentation,at://did:plc:wshs7t2adsemcrrd4snkeqli/sh.tangled.label.definition/assignee"` // delimiter=,
137 GoodFirstIssue string `env:"GFI, default=at://did:plc:wshs7t2adsemcrrd4snkeqli/sh.tangled.label.definition/good-first-issue"`
138}
139
140type BlueskyConfig struct {
141 UpdateInterval time.Duration `env:"UPDATE_INTERVAL, default=1h"`
142}
143
144type OgreConfig struct {
145 Host string `env:"HOST, default=https://ogre.tangled.network"`
146}
147
148func (cfg RedisConfig) ToURL() string {
149 u := &url.URL{
150 Scheme: "redis",
151 Host: cfg.Addr,
152 Path: fmt.Sprintf("/%d", cfg.DB),
153 }
154
155 if cfg.Password != "" {
156 u.User = url.UserPassword("", cfg.Password)
157 }
158
159 return u.String()
160}
161
162type Config struct {
163 Core CoreConfig `env:",prefix=TANGLED_"`
164 Jetstream JetstreamConfig `env:",prefix=TANGLED_JETSTREAM_"`
165 Knotstream ConsumerConfig `env:",prefix=TANGLED_KNOTSTREAM_"`
166 Spindlestream ConsumerConfig `env:",prefix=TANGLED_SPINDLESTREAM_"`
167 Resend ResendConfig `env:",prefix=TANGLED_RESEND_"`
168 Posthog PosthogConfig `env:",prefix=TANGLED_POSTHOG_"`
169 Camo CamoConfig `env:",prefix=TANGLED_CAMO_"`
170 Avatar AvatarConfig `env:",prefix=TANGLED_AVATAR_"`
171 OAuth OAuthConfig `env:",prefix=TANGLED_OAUTH_"`
172 Redis RedisConfig `env:",prefix=TANGLED_REDIS_"`
173 Plc PlcConfig `env:",prefix=TANGLED_PLC_"`
174 Pds PdsConfig `env:",prefix=TANGLED_PDS_"`
175 Cloudflare Cloudflare `env:",prefix=TANGLED_CLOUDFLARE_"`
176 Label LabelConfig `env:",prefix=TANGLED_LABEL_"`
177 Bluesky BlueskyConfig `env:",prefix=TANGLED_BLUESKY_"`
178 Sites SitesConfig `env:",prefix=TANGLED_SITES_"`
179 KnotMirror KnotMirrorConfig `env:",prefix=TANGLED_KNOTMIRROR_"`
180 Ogre OgreConfig `env:",prefix=TANGLED_OGRE_"`
181}
182
183func LoadConfig(ctx context.Context) (*Config, error) {
184 var cfg Config
185 err := envconfig.Process(ctx, &cfg)
186 if err != nil {
187 return nil, err
188 }
189
190 return &cfg, nil
191}