UNPKG

@graphprotocol/graph-cli

Version:

CLI for building for and deploying to The Graph

135 lines (134 loc) 4.97 kB
import { print, prompt } from 'gluegun'; import open from 'open'; import { Args, Command, Flags, ux } from '@oclif/core'; // eslint-disable-next-line no-restricted-imports import { URL, URLSearchParams } from '@whatwg-node/fetch'; import { createCompiler } from '../command-helpers/compiler.js'; import * as DataSourcesExtractor from '../command-helpers/data-sources.js'; import { DEFAULT_IPFS_URL } from '../command-helpers/ipfs.js'; import Protocol from '../protocols/index.js'; export default class PublishCommand extends Command { static description = 'Publish to the Graph Network'; static args = { 'subgraph-manifest': Args.string({ default: 'subgraph.yaml', }), }; static flags = { help: Flags.help({ char: 'h', }), 'subgraph-id': Flags.string({ summary: 'Subgraph ID to publish to.', required: false, }), 'protocol-network': Flags.string({ summary: 'The network to use for the subgraph deployment.', options: ['arbitrum-one', 'arbitrum-sepolia'], default: 'arbitrum-one', required: false, }), ipfs: Flags.string({ summary: 'Upload build results to an IPFS node.', char: 'i', default: DEFAULT_IPFS_URL, }), 'ipfs-hash': Flags.string({ summary: 'IPFS hash of the subgraph manifest to deploy.', required: false, }), 'webapp-url': Flags.string({ summary: 'URL of the web UI you want to use to deploy.', required: false, default: 'https://cli.thegraph.com/publish', }), 'api-key': Flags.string({ summary: 'The API key to use for the Subgraph queries.', required: false, }), }; /** * Prompt the user to open up the browser to continue publishing the subgraph */ async publishWithBrowser({ ipfsHash, webapp, subgraphId, protocolNetwork, apiKey, }) { const { openBrowser } = await prompt .ask([ { type: 'confirm', name: 'openBrowser', message: () => `Open up the browser to continue publishing ?`, initial: true, required: true, }, ]) .catch(() => this.exit(1)); if (!openBrowser) { this.exit(0); } const url = new URL(webapp); const searchParams = new URLSearchParams(url.search); searchParams.set('id', ipfsHash); if (subgraphId) { searchParams.set('subgraphId', subgraphId); } if (protocolNetwork) { searchParams.set('network', protocolNetwork); } if (apiKey) { searchParams.set('apiKey', apiKey); } url.search = searchParams.toString(); const openUrl = url.toString(); print.success(`Finalize the publish of the subgraph from the Graph CLI publish page. Opening up the browser to continue publishing at ${openUrl}`); await open(openUrl); return; } async run() { const { args: { 'subgraph-manifest': manifest }, flags: { 'ipfs-hash': ipfsHash, 'webapp-url': webUiUrl, ipfs, 'subgraph-id': subgraphId, 'protocol-network': protocolNetwork, 'api-key': apiKey, }, } = await this.parse(PublishCommand); if (subgraphId && !apiKey) { ux.error('API key is required to publish to an existing subgraph (`--api-key`).\nSee https://thegraph.com/docs/en/deploying/subgraph-studio-faqs/#2-how-do-i-create-an-api-key', { exit: 1 }); } if (ipfsHash) { await this.publishWithBrowser({ ipfsHash, webapp: webUiUrl, subgraphId, protocolNetwork, apiKey, }); return; } let protocol; try { const dataSourcesAndTemplates = await DataSourcesExtractor.fromFilePath(manifest); protocol = Protocol.fromDataSources(dataSourcesAndTemplates); } catch (e) { this.error(e, { exit: 1 }); } const compiler = createCompiler(manifest, { ipfs, outputDir: 'build/', outputFormat: 'wasm', skipMigrations: false, protocol, }); // Exit with an error code if the compiler couldn't be created if (!compiler) { this.exit(1); } const result = await compiler.compile({ validate: true }); if (result === undefined || result === false) { // Compilation failed, not deploying. this.exit(1); } await this.publishWithBrowser({ ipfsHash: result, webapp: webUiUrl, subgraphId, protocolNetwork, apiKey, }); return; } }