forked from prisma/prisma
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Studio.ts
168 lines (136 loc) · 4.54 KB
/
Studio.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
import Debug from '@prisma/debug'
import { enginesVersion } from '@prisma/engines'
import {
arg,
Command,
format,
getConfig,
getDirectUrl,
HelpError,
isError,
loadEnvFile,
resolveUrl,
} from '@prisma/internals'
import { getSchemaPathAndPrint } from '@prisma/migrate'
import { StudioServer } from '@prisma/studio-server'
import fs from 'fs'
import getPort from 'get-port'
import { bold, dim, red } from 'kleur/colors'
import open from 'open'
import path from 'path'
// Note that we have a test relying on the namespace
// Any change to the namespace must be done in the test as well
// See packages/client/tests/e2e/issues/studio-1128-spawn-enoent/_steps.ts
const debug = Debug('prisma:cli:studio')
const packageJson = require('../package.json') // eslint-disable-line @typescript-eslint/no-var-requires
export class Studio implements Command {
public instance?: StudioServer
public static new(): Studio {
return new Studio()
}
private static help = format(`
Browse your data with Prisma Studio
${bold('Usage')}
${dim('$')} prisma studio [options]
${bold('Options')}
-h, --help Display this help message
-p, --port Port to start Studio on
-b, --browser Browser to open Studio in
-n, --hostname Hostname to bind the Express server to
--schema Custom path to your Prisma schema
${bold('Examples')}
Start Studio on the default port
${dim('$')} prisma studio
Start Studio on a custom port
${dim('$')} prisma studio --port 5555
Start Studio in a specific browser
${dim('$')} prisma studio --port 5555 --browser firefox
${dim('$')} BROWSER=firefox prisma studio --port 5555
Start Studio without opening in a browser
${dim('$')} prisma studio --port 5555 --browser none
${dim('$')} BROWSER=none prisma studio --port 5555
Specify a schema
${dim('$')} prisma studio --schema=./schema.prisma
`)
/**
* Parses arguments passed to this command, and starts Studio
*
* @param argv Array of all arguments
*/
public async parse(argv: string[]): Promise<string | Error> {
const args = arg(argv, {
'--help': Boolean,
'-h': '--help',
'--port': Number,
'-p': '--port',
'--browser': String,
'-b': '--browser',
'--hostname': String,
'-n': '--hostname',
'--schema': String,
'--telemetry-information': String,
})
if (isError(args)) {
return this.help(args.message)
}
if (args['--help']) {
return this.help()
}
loadEnvFile({ schemaPath: args['--schema'], printMessage: true })
const schemaPath = await getSchemaPathAndPrint(args['--schema'])
const hostname = args['--hostname']
const port = args['--port'] || (await getPort({ port: getPort.makeRange(5555, 5600) }))
const browser = args['--browser'] || process.env.BROWSER
const staticAssetDir = path.resolve(__dirname, '../build/public')
const schema = await fs.promises.readFile(schemaPath, 'utf-8')
const config = await getConfig({ datamodel: schema, ignoreEnvVarErrors: true })
process.env.PRISMA_DISABLE_WARNINGS = 'true' // disable client warnings
const studio = new StudioServer({
schemaPath,
hostname,
port,
staticAssetDir,
prismaClient: {
resolve: {
'@prisma/client': path.resolve(__dirname, '../prisma-client/index.js'),
},
directUrl: resolveUrl(getDirectUrl(config.datasources[0])),
},
versions: {
prisma: packageJson.version,
queryEngine: enginesVersion,
},
})
await studio.start()
const serverUrl = `http://localhost:${port}`
if (!browser || browser.toLowerCase() !== 'none') {
try {
const subprocess = await open(serverUrl, {
app: browser,
url: true,
})
subprocess.on('spawn', () => {
// We match on this string in debug logs in tests
debug(`requested to open the url ${serverUrl}`)
})
subprocess.on('error', (e) => {
debug(e)
// We match on this string in debug logs in tests
debug(`failed to open the url ${serverUrl} in browser`)
})
} catch (e) {
// Ignore any errors that occur when trying to open the browser, since they should not halt the process
debug(e)
}
}
this.instance = studio
return `Prisma Studio is up on ${serverUrl}`
}
// help message
public help(error?: string): string | HelpError {
if (error) {
return new HelpError(`\n${bold(red(`!`))} ${error}\n${Studio.help}`)
}
return Studio.help
}
}