-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathmod.ts
194 lines (170 loc) · 4.79 KB
/
mod.ts
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
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
import {
ArkiveConsoleLogHandler,
ArkiveManifest,
Arkiver,
buildSchemaFromEntities,
defaultArkiveData,
} from '../../mod.ts'
import { $, createYoga, delay, join, log, logLevel, serve } from '../deps.ts'
import { ArkiverMetadata } from '../../src/arkiver/arkive-metadata.ts'
import { createManifestHandlers } from './logger.ts'
import { colors, mongoose, SchemaComposer } from '../../src/deps.ts'
import { logger } from '../../src/logger.ts'
import { collectRpcUrls } from '../utils.ts'
export const action = async (
options: {
manifest?: string
rpcUrl?: string[]
mongoConnection?: string
db: boolean
gql: boolean
logLevel: string
gqlOnly?: true
},
directory: string,
) => {
Deno.env.set('DENO_ENV', 'PROD')
try {
logLevel.getLevelByName(options.logLevel.toUpperCase() as log.LevelName)
} catch (e) {
console.error(e)
Deno.exit(1)
}
if (!options.mongoConnection && options.db) {
try {
const cleanup = async () => {
console.log(`\nCleaning up...`)
const stopRes = await $`docker stop ${
containerId.stdout.substring(0, 12)
}`
.stdout('piped')
Deno.exit(stopRes.code)
}
const addSignalToCleanup = (signal: Deno.Signal) => {
try {
Deno.addSignalListener(signal, cleanup)
// deno-lint-ignore no-unused-vars no-empty
} catch (e) {}
}
addSignalToCleanup('SIGINT')
addSignalToCleanup('SIGHUP')
addSignalToCleanup('SIGTERM')
addSignalToCleanup('SIGQUIT')
addSignalToCleanup('SIGTSTP')
addSignalToCleanup('SIGABRT')
const containerId =
await $`docker run --name arkiver_mongodb -d -p 27017:27017 --env MONGO_INITDB_ROOT_USERNAME=admin --env MONGO_INITDB_ROOT_PASSWORD=password --rm mongo`
.stdout('piped')
await delay(3000) // wait for db to start
} catch (e) {
console.error(
`Failed to start mongodb container: ${e.message}, ${e.stack}`,
)
Deno.exit(1)
}
}
const { manifest: manifestPath } = options
const dir = `file://${
join(Deno.cwd(), directory, manifestPath ?? 'manifest.ts')
}`
const manifestImport = await import(dir)
const manifest: ArkiveManifest | undefined = manifestImport.default ??
manifestImport.manifest
if (!manifest) {
throw new Error(
`Manifest file must export a default or manifest object.`,
)
}
const { handlers, loggers } = createManifestHandlers(
manifest,
options.logLevel.toUpperCase() as log.LevelName,
)
log.setup({
handlers: {
arkiver: new ArkiveConsoleLogHandler(
options.logLevel.toUpperCase() as log.LevelName,
{
arkive: {
name: manifest.name ?? 'my-arkive',
id: 0,
majorVersion: 1,
minorVersion: 0,
},
},
),
...handlers,
},
loggers: {
arkiver: {
level: options.logLevel.toUpperCase() as log.LevelName,
handlers: ['arkiver'],
},
...loggers,
},
})
// An RPC for our Arkive is going to be assigned at some point.
// The order of assignment is as follows:
// 1. CLI command line option -r, --rpc-url
// 2. Env variables such as {CHAIN}_RPC_URL
// 3. RPC url defined in manifest
// 4. Default RPC of Viem
const rpcUrls = options.rpcUrl?.reduce((acc, rpc) => {
const [name, url] = rpc.split('=')
acc[name] = url
return acc
}, {} as Record<string, string>) ?? collectRpcUrls() ?? {}
logger('arkiver').debug(`Connecting to database...`)
const connectionString = options.mongoConnection ??
'mongodb://admin:password@localhost:27017'
await mongoose.connect(connectionString, {
dbName: '0-0',
})
logger('arkiver').debug(`Connected to database`)
if (!options.gqlOnly) {
const arkiver = new Arkiver({
manifest,
noDb: !options.db,
rpcUrls,
arkiveData: {
...defaultArkiveData,
name: manifest.name ?? 'my-arkive',
},
})
await arkiver.run()
}
if (!options.gql || !options.db) {
return
}
const schemaComposer = new SchemaComposer()
buildSchemaFromEntities(
schemaComposer,
[...manifest.entities, { model: ArkiverMetadata, list: true }],
)
if (manifest.schemaComposerCustomizer) {
manifest.schemaComposerCustomizer(schemaComposer)
}
const schema = schemaComposer.buildSchema()
const yoga = createYoga({
schema,
fetchAPI: {
Response,
},
graphiql: {
title: 'Arkiver Playground',
},
})
await serve(yoga, {
port: 4000,
onListen: ({ hostname, port }) => {
console.log(
colors.magenta(
colors.bold(
`🚀 Arkiver playground ready at ${
colors.underline(`http://${hostname}:${port}/graphql`)
}`,
),
),
)
},
})
}