-
Notifications
You must be signed in to change notification settings - Fork 4
Expand file tree
/
Copy pathenv.ts
More file actions
94 lines (82 loc) · 3.26 KB
/
env.ts
File metadata and controls
94 lines (82 loc) · 3.26 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
import dotenv from "dotenv";
import { z } from "zod";
import { stringify } from "@grants-stack-indexer/shared";
dotenv.config();
const stringToJSONSchema = z.string().transform((str, ctx): z.infer<ReturnType<typeof Object>> => {
try {
return JSON.parse(str);
} catch (e) {
ctx.addIssue({ code: "custom", message: "Invalid JSON" });
return z.NEVER;
}
});
const chainSchema = z.object({
rpcUrls: z.array(z.string().url()).nonempty(),
id: z.coerce.number().int().positive(),
name: z.string(),
fetchLimit: z.coerce.number().int().positive().default(500),
fetchDelayMs: z.coerce.number().int().positive().default(1000),
});
const baseSchema = z.object({
CHAINS: stringToJSONSchema.pipe(z.array(chainSchema).nonempty()).refine((chains) => {
const ids = chains.map((chain) => chain.id);
const uniqueIds = new Set(ids);
return ids.length === uniqueIds.size;
}, "Chain IDs must be unique"),
DATABASE_URL: z.string(),
DATABASE_SCHEMA: z.string().default("public"),
INDEXER_GRAPHQL_URL: z.string().url(),
INDEXER_ADMIN_SECRET: z.string(),
PRICING_SOURCE: z.enum(["dummy", "coingecko"]).default("coingecko"),
IPFS_GATEWAYS_URL: stringToJSONSchema
.pipe(z.array(z.string().url()))
.default('["https://ipfs.io"]'),
RETRY_MAX_ATTEMPTS: z.coerce.number().int().min(1).default(3),
RETRY_BASE_DELAY_MS: z.coerce.number().int().min(1).default(3000), // 3 seconds
RETRY_FACTOR: z.coerce.number().int().min(1).default(2),
RETRY_MAX_DELAY_MS: z.coerce.number().int().min(1).optional(), // 5 minute
NOTIFIER_PROVIDER: z.enum(["slack", "null"]).default("null"),
SLACK_WEBHOOK_URL: z.string().url().optional(),
LOG_LEVEL: z.enum(["error", "warn", "info", "debug"]).default("info"),
});
const dummyPricingSchema = baseSchema.extend({
PRICING_SOURCE: z.literal("dummy"),
DUMMY_PRICE: z.coerce.number().optional().default(1),
});
const coingeckoPricingSchema = baseSchema.extend({
PRICING_SOURCE: z.literal("coingecko"),
COINGECKO_API_KEY: z.string().min(1),
COINGECKO_API_TYPE: z.enum(["demo", "pro"]).default("pro"),
});
const validationSchema = z
.discriminatedUnion("PRICING_SOURCE", [dummyPricingSchema, coingeckoPricingSchema])
.transform((val) => {
if (val.PRICING_SOURCE === "dummy") {
return { pricingSource: val.PRICING_SOURCE, dummyPrice: val.DUMMY_PRICE, ...val };
}
return {
pricingSource: val.PRICING_SOURCE,
apiKey: val.COINGECKO_API_KEY,
apiType: val.COINGECKO_API_TYPE,
...val,
};
})
.superRefine((data, ctx) => {
if (data.NOTIFIER_PROVIDER === "slack" && !data.SLACK_WEBHOOK_URL) {
ctx.addIssue({
code: "custom",
message: "SLACK_WEBHOOK_URL is required when NOTIFIER_PROVIDER is slack",
path: ["SLACK_WEBHOOK_URL"],
});
}
});
const env = validationSchema.safeParse(process.env);
if (!env.success) {
console.error(
"Invalid environment variables:",
env.error.issues.map((issue) => stringify(issue)).join("\n"),
);
process.exit(1);
}
export const environment = env.data;
export type Environment = z.infer<typeof validationSchema>;