@backstage/plugin-scaffolder-backend-module-rails 0.3.0 → 0.3.3

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/CHANGELOG.md CHANGED
@@ -1,5 +1,37 @@
1
1
  # @backstage/plugin-scaffolder-backend-module-rails
2
2
 
3
+ ## 0.3.3
4
+
5
+ ### Patch Changes
6
+
7
+ - Updated dependencies
8
+ - @backstage/backend-common@0.12.0
9
+ - @backstage/plugin-scaffolder-backend@0.17.3
10
+ - @backstage/integration@0.8.0
11
+
12
+ ## 0.3.2
13
+
14
+ ### Patch Changes
15
+
16
+ - 0f37cdef19: Migrated over from the deprecated `spec.metadata` to `spec.templateInfo` for the `name` and the `baseUrl` of the template.
17
+ - Updated dependencies
18
+ - @backstage/backend-common@0.11.0
19
+ - @backstage/plugin-scaffolder-backend@0.17.0
20
+ - @backstage/integration@0.7.5
21
+
22
+ ## 0.3.1
23
+
24
+ ### Patch Changes
25
+
26
+ - Fix for the previous release with missing type declarations.
27
+ - Updated dependencies
28
+ - @backstage/backend-common@0.10.9
29
+ - @backstage/config@0.1.15
30
+ - @backstage/errors@0.2.2
31
+ - @backstage/integration@0.7.4
32
+ - @backstage/types@0.1.3
33
+ - @backstage/plugin-scaffolder-backend@0.16.1
34
+
3
35
  ## 0.3.0
4
36
 
5
37
  ### Minor Changes
package/dist/index.cjs.js CHANGED
@@ -101,7 +101,7 @@ class RailsNewRunner {
101
101
  const commandExistsToRun = await commandExists__default["default"](baseCommand);
102
102
  if (commandExistsToRun) {
103
103
  const arrayExtraArguments = railsArgumentResolver(workspacePath, railsArguments);
104
- await pluginScaffolderBackend.runCommand({
104
+ await pluginScaffolderBackend.executeShellCommand({
105
105
  command: baseCommand,
106
106
  args: [
107
107
  ...baseArguments,
@@ -236,14 +236,14 @@ function createFetchRailsAction(options) {
236
236
  }
237
237
  },
238
238
  async handler(ctx) {
239
- var _a;
239
+ var _a, _b;
240
240
  ctx.logger.info("Fetching and then templating using rails");
241
241
  const workDir = await ctx.createTemporaryDirectory();
242
242
  const resultDir = path.resolve(workDir, "result");
243
243
  await pluginScaffolderBackend.fetchContents({
244
244
  reader,
245
245
  integrations,
246
- baseUrl: ctx.baseUrl,
246
+ baseUrl: (_a = ctx.templateInfo) == null ? void 0 : _a.baseUrl,
247
247
  fetchUrl: ctx.input.url,
248
248
  outputPath: workDir
249
249
  });
@@ -257,7 +257,7 @@ function createFetchRailsAction(options) {
257
257
  logStream: ctx.logStream,
258
258
  values
259
259
  });
260
- const targetPath = (_a = ctx.input.targetPath) != null ? _a : "./";
260
+ const targetPath = (_b = ctx.input.targetPath) != null ? _b : "./";
261
261
  const outputPath = path.resolve(ctx.workspacePath, targetPath);
262
262
  if (!outputPath.startsWith(ctx.workspacePath)) {
263
263
  throw new errors.InputError(`Fetch action targetPath may not specify a path outside the working directory`);
@@ -1 +1 @@
1
- {"version":3,"file":"index.cjs.js","sources":["../src/actions/fetch/rails/railsArgumentResolver.ts","../src/actions/fetch/rails/railsNewRunner.ts","../src/actions/fetch/rails/index.ts"],"sourcesContent":["/*\n * Copyright 2021 The Backstage Authors\n *\n * Licensed under the Apache License, Version 2.0 (the \"License\");\n * you may not use this file except in compliance with the License.\n * You may obtain a copy of the License at\n *\n * http://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software\n * distributed under the License is distributed on an \"AS IS\" BASIS,\n * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n * See the License for the specific language governing permissions and\n * limitations under the License.\n */\n\nimport { sep as separatorPath } from 'path';\n\nenum Webpacker {\n react = 'react',\n vue = 'vue',\n angular = 'angular',\n elm = 'elm',\n stimulus = 'stimulus',\n}\n\nenum Database {\n mysql = 'mysql',\n postgresql = 'postgresql',\n sqlite3 = 'sqlite3',\n oracle = 'oracle',\n sqlserver = 'sqlserver',\n jdbcmysql = 'jdbcmysql',\n jdbcsqlite3 = 'jdbcsqlite3',\n jdbcpostgresql = 'jdbcpostgresql',\n jdbc = 'jdbc',\n}\n\nenum RailsVersion {\n dev = 'dev',\n edge = 'edge',\n master = 'master',\n fromImage = 'fromImage',\n}\n\nexport type RailsRunOptions = {\n minimal?: boolean;\n api?: boolean;\n template?: string;\n webpacker?: Webpacker;\n database?: Database;\n railsVersion?: RailsVersion;\n skipBundle?: boolean;\n skipWebpackInstall?: boolean;\n skipTest?: boolean;\n force?: boolean;\n};\n\nexport const railsArgumentResolver = (\n projectRoot: string,\n options: RailsRunOptions,\n executionOnContainer = false,\n): string[] => {\n const argumentsToRun: string[] = [];\n\n if (options?.minimal) {\n argumentsToRun.push('--minimal');\n }\n\n if (options?.api) {\n argumentsToRun.push('--api');\n }\n\n if (options?.skipBundle) {\n argumentsToRun.push('--skip-bundle');\n }\n\n if (options?.skipWebpackInstall) {\n argumentsToRun.push('--skip-webpack-install');\n }\n\n if (options?.skipTest) {\n argumentsToRun.push('--skip-test');\n }\n\n if (options?.force) {\n argumentsToRun.push('--force');\n }\n\n if (\n options?.webpacker &&\n Object.values(Webpacker).includes(options?.webpacker as Webpacker)\n ) {\n argumentsToRun.push('--webpack');\n argumentsToRun.push(options.webpacker);\n }\n\n if (\n options?.database &&\n Object.values(Database).includes(options?.database as Database)\n ) {\n argumentsToRun.push('--database');\n argumentsToRun.push(options.database);\n }\n\n if (\n options?.railsVersion !== RailsVersion.fromImage &&\n Object.values(RailsVersion).includes(options?.railsVersion as RailsVersion)\n ) {\n argumentsToRun.push(`--${options.railsVersion}`);\n }\n\n if (options?.template) {\n argumentsToRun.push('--template');\n argumentsToRun.push(\n options.template.replace(\n `.${separatorPath}`,\n `${projectRoot}${executionOnContainer ? '/' : separatorPath}`,\n ),\n );\n }\n\n return argumentsToRun;\n};\n","/*\n * Copyright 2021 The Backstage Authors\n *\n * Licensed under the Apache License, Version 2.0 (the \"License\");\n * you may not use this file except in compliance with the License.\n * You may obtain a copy of the License at\n *\n * http://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software\n * distributed under the License is distributed on an \"AS IS\" BASIS,\n * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n * See the License for the specific language governing permissions and\n * limitations under the License.\n */\n\nimport { ContainerRunner } from '@backstage/backend-common';\nimport fs from 'fs-extra';\nimport path from 'path';\nimport { runCommand } from '@backstage/plugin-scaffolder-backend';\nimport commandExists from 'command-exists';\nimport {\n railsArgumentResolver,\n RailsRunOptions,\n} from './railsArgumentResolver';\nimport { JsonObject } from '@backstage/types';\nimport { Writable } from 'stream';\n\nexport class RailsNewRunner {\n private readonly containerRunner: ContainerRunner;\n\n constructor({ containerRunner }: { containerRunner: ContainerRunner }) {\n this.containerRunner = containerRunner;\n }\n\n public async run({\n workspacePath,\n values,\n logStream,\n }: {\n workspacePath: string;\n values: JsonObject;\n logStream: Writable;\n }): Promise<void> {\n const intermediateDir = path.join(workspacePath, 'intermediate');\n await fs.ensureDir(intermediateDir);\n const resultDir = path.join(workspacePath, 'result');\n\n const { name, imageName, railsArguments } = values;\n\n // Directories to bind on container\n const mountDirs = {\n [workspacePath]: '/input',\n [intermediateDir]: '/output',\n };\n\n const baseCommand = 'rails';\n const baseArguments = ['new'];\n const commandExistsToRun = await commandExists(baseCommand);\n\n if (commandExistsToRun) {\n const arrayExtraArguments = railsArgumentResolver(\n workspacePath,\n railsArguments as RailsRunOptions,\n );\n\n await runCommand({\n command: baseCommand,\n args: [\n ...baseArguments,\n `${intermediateDir}${path.sep}${name}`,\n ...arrayExtraArguments,\n ],\n logStream,\n });\n } else {\n const arrayExtraArguments = railsArgumentResolver(\n '/input',\n railsArguments as RailsRunOptions,\n true,\n );\n await this.containerRunner.runContainer({\n imageName: imageName as string,\n command: baseCommand,\n args: [...baseArguments, `/output/${name}`, ...arrayExtraArguments],\n mountDirs,\n workingDir: '/input',\n // Set the home directory inside the container as something that applications can\n // write to, otherwise they will just fail trying to write to /\n envVars: { HOME: '/tmp' },\n logStream,\n });\n }\n\n // if command was successful, intermediateDir should contain\n // exactly one directory.\n const [generated] = await fs.readdir(intermediateDir);\n\n if (generated === undefined) {\n throw new Error(`No data generated by ${baseCommand}`);\n }\n\n await fs.move(path.join(intermediateDir, generated), resultDir);\n }\n}\n","/*\n * Copyright 2021 The Backstage Authors\n *\n * Licensed under the Apache License, Version 2.0 (the \"License\");\n * you may not use this file except in compliance with the License.\n * You may obtain a copy of the License at\n *\n * http://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software\n * distributed under the License is distributed on an \"AS IS\" BASIS,\n * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n * See the License for the specific language governing permissions and\n * limitations under the License.\n */\n\nimport { ContainerRunner, UrlReader } from '@backstage/backend-common';\nimport { JsonObject } from '@backstage/types';\nimport { InputError } from '@backstage/errors';\nimport { ScmIntegrations } from '@backstage/integration';\nimport fs from 'fs-extra';\nimport {\n createTemplateAction,\n fetchContents,\n} from '@backstage/plugin-scaffolder-backend';\n\nimport { resolve as resolvePath } from 'path';\nimport { RailsNewRunner } from './railsNewRunner';\n\n/**\n * Creates the `fetch:rails` Scaffolder action.\n *\n * @remarks\n *\n * See {@link https://guides.rubyonrails.org/rails_application_templates.html} and {@link https://backstage.io/docs/features/software-templates/writing-custom-actions}.\n *\n * @param options - Configuration of the templater.\n * @public\n */\nexport function createFetchRailsAction(options: {\n reader: UrlReader;\n integrations: ScmIntegrations;\n containerRunner: ContainerRunner;\n}) {\n const { reader, integrations, containerRunner } = options;\n\n return createTemplateAction<{\n url: string;\n targetPath?: string;\n values: JsonObject;\n imageName?: string;\n }>({\n id: 'fetch:rails',\n description:\n 'Downloads a template from the given URL into the workspace, and runs a rails new generator on it.',\n schema: {\n input: {\n type: 'object',\n required: ['url'],\n properties: {\n url: {\n title: 'Fetch URL',\n description:\n 'Relative path or absolute URL pointing to the directory tree to fetch',\n type: 'string',\n },\n targetPath: {\n title: 'Target Path',\n description:\n 'Target path within the working directory to download the contents to.',\n type: 'string',\n },\n values: {\n title: 'Template Values',\n description: 'Values to pass on to rails for templating',\n type: 'object',\n properties: {\n railsArguments: {\n title: 'Arguments to pass to new command',\n description:\n 'You can provide some arguments to create a custom app',\n type: 'object',\n properties: {\n minimal: {\n title: 'minimal',\n description: 'Preconfigure a minimal rails app',\n type: 'boolean',\n },\n skipBundle: {\n title: 'skipBundle',\n description: \"Don't run bundle install\",\n type: 'boolean',\n },\n skipWebpackInstall: {\n title: 'skipWebpackInstall',\n description: \"Don't run Webpack install\",\n type: 'boolean',\n },\n skipTest: {\n title: 'skipTest',\n description: 'Skip test files',\n type: 'boolean',\n },\n force: {\n title: 'force',\n description: 'Overwrite files that already exist',\n type: 'boolean',\n },\n api: {\n title: 'api',\n description: 'Preconfigure smaller stack for API only apps',\n type: 'boolean',\n },\n template: {\n title: 'template',\n description:\n 'Path to some application template (can be a filesystem path or URL)',\n type: 'string',\n },\n webpacker: {\n title: 'webpacker',\n description:\n 'Preconfigure Webpack with a particular framework (options: react, vue, angular, elm, stimulus)',\n type: 'string',\n enum: ['react', 'vue', 'angular', 'elm', 'stimulus'],\n },\n database: {\n title: 'database',\n description:\n 'Preconfigure for selected database (options: mysql/postgresql/sqlite3/oracle/sqlserver/jdbcmysql/jdbcsqlite3/jdbcpostgresql/jdbc)',\n type: 'string',\n enum: [\n 'mysql',\n 'postgresql',\n 'sqlite3',\n 'oracle',\n 'sqlserver',\n 'jdbcmysql',\n 'jdbcsqlite3',\n 'jdbcpostgresql',\n 'jdbc',\n ],\n },\n railsVersion: {\n title: 'Rails version in Gemfile',\n description:\n 'Set up the application with Gemfile pointing to a specific version (options: fromImage, dev, edge, master)',\n type: 'string',\n enum: ['dev', 'edge', 'master', 'fromImage'],\n },\n },\n },\n },\n },\n imageName: {\n title: 'Rails Docker image',\n description:\n 'Specify a Docker image to run rails new. Used only when a local rails is not found.',\n type: 'string',\n },\n },\n },\n },\n async handler(ctx) {\n ctx.logger.info('Fetching and then templating using rails');\n\n const workDir = await ctx.createTemporaryDirectory();\n const resultDir = resolvePath(workDir, 'result');\n\n await fetchContents({\n reader,\n integrations,\n baseUrl: ctx.baseUrl,\n fetchUrl: ctx.input.url,\n outputPath: workDir,\n });\n\n const templateRunner = new RailsNewRunner({ containerRunner });\n\n const values = {\n ...ctx.input.values,\n imageName: ctx.input.imageName,\n };\n\n // Will execute the template in ./template and put the result in ./result\n await templateRunner.run({\n workspacePath: workDir,\n logStream: ctx.logStream,\n values,\n });\n\n // Finally move the template result into the task workspace\n const targetPath = ctx.input.targetPath ?? './';\n const outputPath = resolvePath(ctx.workspacePath, targetPath);\n if (!outputPath.startsWith(ctx.workspacePath)) {\n throw new InputError(\n `Fetch action targetPath may not specify a path outside the working directory`,\n );\n }\n await fs.copy(resultDir, outputPath);\n },\n });\n}\n"],"names":["separatorPath","path","fs","commandExists","runCommand","createTemplateAction","resolvePath","fetchContents","InputError"],"mappings":"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"}
1
+ {"version":3,"file":"index.cjs.js","sources":["../src/actions/fetch/rails/railsArgumentResolver.ts","../src/actions/fetch/rails/railsNewRunner.ts","../src/actions/fetch/rails/index.ts"],"sourcesContent":["/*\n * Copyright 2021 The Backstage Authors\n *\n * Licensed under the Apache License, Version 2.0 (the \"License\");\n * you may not use this file except in compliance with the License.\n * You may obtain a copy of the License at\n *\n * http://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software\n * distributed under the License is distributed on an \"AS IS\" BASIS,\n * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n * See the License for the specific language governing permissions and\n * limitations under the License.\n */\n\nimport { sep as separatorPath } from 'path';\n\nenum Webpacker {\n react = 'react',\n vue = 'vue',\n angular = 'angular',\n elm = 'elm',\n stimulus = 'stimulus',\n}\n\nenum Database {\n mysql = 'mysql',\n postgresql = 'postgresql',\n sqlite3 = 'sqlite3',\n oracle = 'oracle',\n sqlserver = 'sqlserver',\n jdbcmysql = 'jdbcmysql',\n jdbcsqlite3 = 'jdbcsqlite3',\n jdbcpostgresql = 'jdbcpostgresql',\n jdbc = 'jdbc',\n}\n\nenum RailsVersion {\n dev = 'dev',\n edge = 'edge',\n master = 'master',\n fromImage = 'fromImage',\n}\n\nexport type RailsRunOptions = {\n minimal?: boolean;\n api?: boolean;\n template?: string;\n webpacker?: Webpacker;\n database?: Database;\n railsVersion?: RailsVersion;\n skipBundle?: boolean;\n skipWebpackInstall?: boolean;\n skipTest?: boolean;\n force?: boolean;\n};\n\nexport const railsArgumentResolver = (\n projectRoot: string,\n options: RailsRunOptions,\n executionOnContainer = false,\n): string[] => {\n const argumentsToRun: string[] = [];\n\n if (options?.minimal) {\n argumentsToRun.push('--minimal');\n }\n\n if (options?.api) {\n argumentsToRun.push('--api');\n }\n\n if (options?.skipBundle) {\n argumentsToRun.push('--skip-bundle');\n }\n\n if (options?.skipWebpackInstall) {\n argumentsToRun.push('--skip-webpack-install');\n }\n\n if (options?.skipTest) {\n argumentsToRun.push('--skip-test');\n }\n\n if (options?.force) {\n argumentsToRun.push('--force');\n }\n\n if (\n options?.webpacker &&\n Object.values(Webpacker).includes(options?.webpacker as Webpacker)\n ) {\n argumentsToRun.push('--webpack');\n argumentsToRun.push(options.webpacker);\n }\n\n if (\n options?.database &&\n Object.values(Database).includes(options?.database as Database)\n ) {\n argumentsToRun.push('--database');\n argumentsToRun.push(options.database);\n }\n\n if (\n options?.railsVersion !== RailsVersion.fromImage &&\n Object.values(RailsVersion).includes(options?.railsVersion as RailsVersion)\n ) {\n argumentsToRun.push(`--${options.railsVersion}`);\n }\n\n if (options?.template) {\n argumentsToRun.push('--template');\n argumentsToRun.push(\n options.template.replace(\n `.${separatorPath}`,\n `${projectRoot}${executionOnContainer ? '/' : separatorPath}`,\n ),\n );\n }\n\n return argumentsToRun;\n};\n","/*\n * Copyright 2021 The Backstage Authors\n *\n * Licensed under the Apache License, Version 2.0 (the \"License\");\n * you may not use this file except in compliance with the License.\n * You may obtain a copy of the License at\n *\n * http://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software\n * distributed under the License is distributed on an \"AS IS\" BASIS,\n * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n * See the License for the specific language governing permissions and\n * limitations under the License.\n */\n\nimport { ContainerRunner } from '@backstage/backend-common';\nimport fs from 'fs-extra';\nimport path from 'path';\nimport { executeShellCommand } from '@backstage/plugin-scaffolder-backend';\nimport commandExists from 'command-exists';\nimport {\n railsArgumentResolver,\n RailsRunOptions,\n} from './railsArgumentResolver';\nimport { JsonObject } from '@backstage/types';\nimport { Writable } from 'stream';\n\nexport class RailsNewRunner {\n private readonly containerRunner: ContainerRunner;\n\n constructor({ containerRunner }: { containerRunner: ContainerRunner }) {\n this.containerRunner = containerRunner;\n }\n\n public async run({\n workspacePath,\n values,\n logStream,\n }: {\n workspacePath: string;\n values: JsonObject;\n logStream: Writable;\n }): Promise<void> {\n const intermediateDir = path.join(workspacePath, 'intermediate');\n await fs.ensureDir(intermediateDir);\n const resultDir = path.join(workspacePath, 'result');\n\n const { name, imageName, railsArguments } = values;\n\n // Directories to bind on container\n const mountDirs = {\n [workspacePath]: '/input',\n [intermediateDir]: '/output',\n };\n\n const baseCommand = 'rails';\n const baseArguments = ['new'];\n const commandExistsToRun = await commandExists(baseCommand);\n\n if (commandExistsToRun) {\n const arrayExtraArguments = railsArgumentResolver(\n workspacePath,\n railsArguments as RailsRunOptions,\n );\n\n await executeShellCommand({\n command: baseCommand,\n args: [\n ...baseArguments,\n `${intermediateDir}${path.sep}${name}`,\n ...arrayExtraArguments,\n ],\n logStream,\n });\n } else {\n const arrayExtraArguments = railsArgumentResolver(\n '/input',\n railsArguments as RailsRunOptions,\n true,\n );\n await this.containerRunner.runContainer({\n imageName: imageName as string,\n command: baseCommand,\n args: [...baseArguments, `/output/${name}`, ...arrayExtraArguments],\n mountDirs,\n workingDir: '/input',\n // Set the home directory inside the container as something that applications can\n // write to, otherwise they will just fail trying to write to /\n envVars: { HOME: '/tmp' },\n logStream,\n });\n }\n\n // if command was successful, intermediateDir should contain\n // exactly one directory.\n const [generated] = await fs.readdir(intermediateDir);\n\n if (generated === undefined) {\n throw new Error(`No data generated by ${baseCommand}`);\n }\n\n await fs.move(path.join(intermediateDir, generated), resultDir);\n }\n}\n","/*\n * Copyright 2021 The Backstage Authors\n *\n * Licensed under the Apache License, Version 2.0 (the \"License\");\n * you may not use this file except in compliance with the License.\n * You may obtain a copy of the License at\n *\n * http://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software\n * distributed under the License is distributed on an \"AS IS\" BASIS,\n * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n * See the License for the specific language governing permissions and\n * limitations under the License.\n */\n\nimport { ContainerRunner, UrlReader } from '@backstage/backend-common';\nimport { JsonObject } from '@backstage/types';\nimport { InputError } from '@backstage/errors';\nimport { ScmIntegrations } from '@backstage/integration';\nimport fs from 'fs-extra';\nimport {\n createTemplateAction,\n fetchContents,\n} from '@backstage/plugin-scaffolder-backend';\n\nimport { resolve as resolvePath } from 'path';\nimport { RailsNewRunner } from './railsNewRunner';\n\n/**\n * Creates the `fetch:rails` Scaffolder action.\n *\n * @remarks\n *\n * See {@link https://guides.rubyonrails.org/rails_application_templates.html} and {@link https://backstage.io/docs/features/software-templates/writing-custom-actions}.\n *\n * @param options - Configuration of the templater.\n * @public\n */\nexport function createFetchRailsAction(options: {\n reader: UrlReader;\n integrations: ScmIntegrations;\n containerRunner: ContainerRunner;\n}) {\n const { reader, integrations, containerRunner } = options;\n\n return createTemplateAction<{\n url: string;\n targetPath?: string;\n values: JsonObject;\n imageName?: string;\n }>({\n id: 'fetch:rails',\n description:\n 'Downloads a template from the given URL into the workspace, and runs a rails new generator on it.',\n schema: {\n input: {\n type: 'object',\n required: ['url'],\n properties: {\n url: {\n title: 'Fetch URL',\n description:\n 'Relative path or absolute URL pointing to the directory tree to fetch',\n type: 'string',\n },\n targetPath: {\n title: 'Target Path',\n description:\n 'Target path within the working directory to download the contents to.',\n type: 'string',\n },\n values: {\n title: 'Template Values',\n description: 'Values to pass on to rails for templating',\n type: 'object',\n properties: {\n railsArguments: {\n title: 'Arguments to pass to new command',\n description:\n 'You can provide some arguments to create a custom app',\n type: 'object',\n properties: {\n minimal: {\n title: 'minimal',\n description: 'Preconfigure a minimal rails app',\n type: 'boolean',\n },\n skipBundle: {\n title: 'skipBundle',\n description: \"Don't run bundle install\",\n type: 'boolean',\n },\n skipWebpackInstall: {\n title: 'skipWebpackInstall',\n description: \"Don't run Webpack install\",\n type: 'boolean',\n },\n skipTest: {\n title: 'skipTest',\n description: 'Skip test files',\n type: 'boolean',\n },\n force: {\n title: 'force',\n description: 'Overwrite files that already exist',\n type: 'boolean',\n },\n api: {\n title: 'api',\n description: 'Preconfigure smaller stack for API only apps',\n type: 'boolean',\n },\n template: {\n title: 'template',\n description:\n 'Path to some application template (can be a filesystem path or URL)',\n type: 'string',\n },\n webpacker: {\n title: 'webpacker',\n description:\n 'Preconfigure Webpack with a particular framework (options: react, vue, angular, elm, stimulus)',\n type: 'string',\n enum: ['react', 'vue', 'angular', 'elm', 'stimulus'],\n },\n database: {\n title: 'database',\n description:\n 'Preconfigure for selected database (options: mysql/postgresql/sqlite3/oracle/sqlserver/jdbcmysql/jdbcsqlite3/jdbcpostgresql/jdbc)',\n type: 'string',\n enum: [\n 'mysql',\n 'postgresql',\n 'sqlite3',\n 'oracle',\n 'sqlserver',\n 'jdbcmysql',\n 'jdbcsqlite3',\n 'jdbcpostgresql',\n 'jdbc',\n ],\n },\n railsVersion: {\n title: 'Rails version in Gemfile',\n description:\n 'Set up the application with Gemfile pointing to a specific version (options: fromImage, dev, edge, master)',\n type: 'string',\n enum: ['dev', 'edge', 'master', 'fromImage'],\n },\n },\n },\n },\n },\n imageName: {\n title: 'Rails Docker image',\n description:\n 'Specify a Docker image to run rails new. Used only when a local rails is not found.',\n type: 'string',\n },\n },\n },\n },\n async handler(ctx) {\n ctx.logger.info('Fetching and then templating using rails');\n\n const workDir = await ctx.createTemporaryDirectory();\n const resultDir = resolvePath(workDir, 'result');\n\n await fetchContents({\n reader,\n integrations,\n baseUrl: ctx.templateInfo?.baseUrl,\n fetchUrl: ctx.input.url,\n outputPath: workDir,\n });\n\n const templateRunner = new RailsNewRunner({ containerRunner });\n\n const values = {\n ...ctx.input.values,\n imageName: ctx.input.imageName,\n };\n\n // Will execute the template in ./template and put the result in ./result\n await templateRunner.run({\n workspacePath: workDir,\n logStream: ctx.logStream,\n values,\n });\n\n // Finally move the template result into the task workspace\n const targetPath = ctx.input.targetPath ?? './';\n const outputPath = resolvePath(ctx.workspacePath, targetPath);\n if (!outputPath.startsWith(ctx.workspacePath)) {\n throw new InputError(\n `Fetch action targetPath may not specify a path outside the working directory`,\n );\n }\n await fs.copy(resultDir, outputPath);\n },\n });\n}\n"],"names":["separatorPath","path","fs","commandExists","executeShellCommand","createTemplateAction","resolvePath","fetchContents","InputError"],"mappings":"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"}
@@ -0,0 +1,27 @@
1
+ import * as _backstage_plugin_scaffolder_backend from '@backstage/plugin-scaffolder-backend';
2
+ import { UrlReader, ContainerRunner } from '@backstage/backend-common';
3
+ import { JsonObject } from '@backstage/types';
4
+ import { ScmIntegrations } from '@backstage/integration';
5
+
6
+ /**
7
+ * Creates the `fetch:rails` Scaffolder action.
8
+ *
9
+ * @remarks
10
+ *
11
+ * See {@link https://guides.rubyonrails.org/rails_application_templates.html} and {@link https://backstage.io/docs/features/software-templates/writing-custom-actions}.
12
+ *
13
+ * @param options - Configuration of the templater.
14
+ * @public
15
+ */
16
+ declare function createFetchRailsAction(options: {
17
+ reader: UrlReader;
18
+ integrations: ScmIntegrations;
19
+ containerRunner: ContainerRunner;
20
+ }): _backstage_plugin_scaffolder_backend.TemplateAction<{
21
+ url: string;
22
+ targetPath?: string | undefined;
23
+ values: JsonObject;
24
+ imageName?: string | undefined;
25
+ }>;
26
+
27
+ export { createFetchRailsAction };
package/package.json CHANGED
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "name": "@backstage/plugin-scaffolder-backend-module-rails",
3
3
  "description": "A module for the scaffolder backend that lets you template projects using Rails",
4
- "version": "0.3.0",
4
+ "version": "0.3.3",
5
5
  "main": "dist/index.cjs.js",
6
6
  "types": "dist/index.d.ts",
7
7
  "license": "Apache-2.0",
@@ -24,17 +24,17 @@
24
24
  "clean": "backstage-cli package clean"
25
25
  },
26
26
  "dependencies": {
27
- "@backstage/backend-common": "^0.10.8",
28
- "@backstage/config": "^0.1.14",
29
- "@backstage/errors": "^0.2.1",
30
- "@backstage/integration": "^0.7.3",
31
- "@backstage/plugin-scaffolder-backend": "^0.16.0",
32
- "@backstage/types": "^0.1.2",
27
+ "@backstage/backend-common": "^0.12.0",
28
+ "@backstage/config": "^0.1.15",
29
+ "@backstage/errors": "^0.2.2",
30
+ "@backstage/integration": "^0.8.0",
31
+ "@backstage/plugin-scaffolder-backend": "^0.17.3",
32
+ "@backstage/types": "^0.1.3",
33
33
  "command-exists": "^1.2.9",
34
34
  "fs-extra": "^9.0.0"
35
35
  },
36
36
  "devDependencies": {
37
- "@backstage/cli": "^0.14.0",
37
+ "@backstage/cli": "^0.15.0",
38
38
  "@types/command-exists": "^1.2.0",
39
39
  "@types/fs-extra": "^9.0.1",
40
40
  "@types/jest": "^26.0.7",
@@ -46,5 +46,5 @@
46
46
  "files": [
47
47
  "dist"
48
48
  ],
49
- "gitHead": "4805c3d13ce9bfc369e53c271b1b95e722b3b4dc"
49
+ "gitHead": "04bb0dd824b78f6b57dac62c3015e681f094045c"
50
50
  }