UNPKG

projen

Version:

CDK for software projects

135 lines • 21.3 kB
"use strict"; Object.defineProperty(exports, "__esModule", { value: true }); exports.renderDockerComposeFile = renderDockerComposeFile; const util_1 = require("../util"); function renderDockerComposeFile(serviceDescriptions, version) { // Record volume configuration const volumeConfig = {}; const volumeInfo = { addVolumeConfiguration(volumeName, configuration) { if (!volumeConfig[volumeName]) { // First volume configuration takes precedence. volumeConfig[volumeName] = configuration; } }, }; // Record network configuration const networkConfig = {}; const networkInfo = { addNetworkConfiguration(networkName, configuration) { if (!networkConfig[networkName]) { // First network configuration takes precedence. networkConfig[networkName] = configuration; } }, }; // Render service configuration const services = {}; for (const [serviceName, serviceDescription] of Object.entries(serviceDescriptions ?? {})) { // Resolve the names of each dependency and check that they exist. // Note: They may not exist if the user made a mistake when referencing a // service by name via `DockerCompose.serviceName()`. // @see DockerCompose.serviceName const dependsOn = Array(); for (const dependsOnServiceName of serviceDescription.dependsOn ?? []) { const resolvedServiceName = dependsOnServiceName.serviceName; if (resolvedServiceName === serviceName) { throw new Error(`Service ${serviceName} cannot depend on itself`); } if (!serviceDescriptions[resolvedServiceName]) { throw new Error(`Unable to resolve service named ${resolvedServiceName} for ${serviceName}`); } dependsOn.push(resolvedServiceName); } // Give each volume binding a chance to bind any necessary volume // configuration and provide volume mount information for the service. const volumes = []; for (const volumeBinding of serviceDescription.volumes ?? []) { volumes.push(volumeBinding.bind(volumeInfo)); } // Give each network binding a chance to bind any necessary network // configuration and provide network mount information for the service. const networks = []; for (const networkBinding of serviceDescription.networks ?? []) { networks.push(networkBinding.bind(networkInfo)); } // Create and store the service configuration, taking care not to create // object members with undefined values. services[serviceName] = { ...getObjectWithKeyAndValueIfValueIsDefined("image", serviceDescription.image), ...getObjectWithKeyAndValueIfValueIsDefined("build", serviceDescription.imageBuild), ...getObjectWithKeyAndValueIfValueIsDefined("entrypoint", serviceDescription.entrypoint), ...getObjectWithKeyAndValueIfValueIsDefined("command", serviceDescription.command), ...getObjectWithKeyAndValueIfValueIsDefined("platform", serviceDescription.platform), ...getObjectWithKeyAndValueIfValueIsDefined("privileged", serviceDescription.privileged), ...(Object.keys(serviceDescription.environment).length > 0 ? { environment: serviceDescription.environment } : {}), ...(serviceDescription.ports.length > 0 ? { ports: serviceDescription.ports } : {}), ...(Object.keys(serviceDescription.labels).length > 0 ? { labels: serviceDescription.labels } : {}), ...(dependsOn.length > 0 ? { dependsOn } : {}), ...(volumes.length > 0 ? { volumes } : {}), ...(networks.length > 0 ? { networks } : {}), }; } // Explicit with the type here because the decamelize step after this wipes // out types. const input = { ...(version ? { version } : {}), services, ...(Object.keys(volumeConfig).length > 0 ? { volumes: volumeConfig } : {}), ...(Object.keys(networkConfig).length > 0 ? { networks: networkConfig } : {}), }; // Change most keys to snake case. return (0, util_1.decamelizeKeysRecursively)(input, { shouldDecamelize: shouldDecamelizeDockerComposeKey, separator: "_", }); } /** * Returns `{ [key]: value }` if `value` is defined, otherwise returns `{}` so * that object spreading can be used to generate a peculiar interface. * @param key * @param value */ function getObjectWithKeyAndValueIfValueIsDefined(key, value) { return value !== undefined ? { [key]: value } : {}; } /** * Determines whether the key at the given path should be decamelized. * Largely, all keys should be snake cased. But, there are some * exceptions for user-provided names for services, volumes, and * environment variables. * * @param path */ function shouldDecamelizeDockerComposeKey(path) { const poundPath = path.join("#"); // Does not decamelize user's names. // services.namehere: // volumes.namehere: // networks.namehere: if (/^(services|volumes|networks)#[^#]+$/.test(poundPath)) { return false; } // Does not decamelize environment variables and labels // services.namehere.environment.* // services.namehere.labels.* if (/^services#[^#]+#(environment|labels)#/.test(poundPath)) { return false; } // Does not decamelize build arguments // services.namehere.build.args.* if (/^services#[^#]+#build#args#/.test(poundPath)) { return false; } // Otherwise, let it all decamelize. return true; } //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"docker-compose-render.js","sourceRoot":"","sources":["../../src/docker-compose/docker-compose-render.ts"],"names":[],"mappings":"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","sourcesContent":["import { DockerComposeBuild } from \"./docker-compose\";\nimport {\n  DockerComposeNetworkConfig,\n  IDockerComposeNetworkConfig,\n} from \"./docker-compose-network\";\nimport { DockerComposeServicePort } from \"./docker-compose-port\";\nimport { DockerComposeService } from \"./docker-compose-service\";\nimport {\n  DockerComposeVolumeConfig,\n  DockerComposeVolumeMount,\n  IDockerComposeVolumeConfig,\n} from \"./docker-compose-volume\";\nimport { decamelizeKeysRecursively } from \"../util\";\n\n/**\n * Structure of a docker compose file's service before we decamelize.\n * @internal\n */\ninterface DockerComposeFileServiceSchema {\n  readonly dependsOn?: string[];\n  readonly build?: DockerComposeBuild;\n  readonly image?: string;\n  readonly command?: string[];\n  readonly volumes?: DockerComposeVolumeMount[];\n  readonly networks?: string[];\n  readonly ports?: DockerComposeServicePort[];\n  readonly environment?: Record<string, string>;\n  readonly labels?: Record<string, string>;\n  readonly entrypoint?: string[];\n  readonly privileged?: boolean;\n}\n\n/**\n * Structure of a docker compose file before we decamelize.\n * @internal\n */\ninterface DockerComposeFileSchema {\n  version?: string;\n  services: Record<string, DockerComposeFileServiceSchema>;\n  volumes?: Record<string, DockerComposeVolumeConfig>;\n}\n\nexport function renderDockerComposeFile(\n  serviceDescriptions: Record<string, DockerComposeService>,\n  version?: string\n): object {\n  // Record volume configuration\n  const volumeConfig: Record<string, DockerComposeVolumeConfig> = {};\n  const volumeInfo: IDockerComposeVolumeConfig = {\n    addVolumeConfiguration(\n      volumeName: string,\n      configuration: DockerComposeVolumeConfig\n    ) {\n      if (!volumeConfig[volumeName]) {\n        // First volume configuration takes precedence.\n        volumeConfig[volumeName] = configuration;\n      }\n    },\n  };\n  // Record network configuration\n  const networkConfig: Record<string, DockerComposeNetworkConfig> = {};\n  const networkInfo: IDockerComposeNetworkConfig = {\n    addNetworkConfiguration(\n      networkName: string,\n      configuration: DockerComposeNetworkConfig\n    ) {\n      if (!networkConfig[networkName]) {\n        // First network configuration takes precedence.\n        networkConfig[networkName] = configuration;\n      }\n    },\n  };\n\n  // Render service configuration\n  const services: Record<string, DockerComposeFileServiceSchema> = {};\n  for (const [serviceName, serviceDescription] of Object.entries(\n    serviceDescriptions ?? {}\n  )) {\n    // Resolve the names of each dependency and check that they exist.\n    // Note: They may not exist if the user made a mistake when referencing a\n    // service by name via `DockerCompose.serviceName()`.\n    // @see DockerCompose.serviceName\n    const dependsOn = Array<string>();\n    for (const dependsOnServiceName of serviceDescription.dependsOn ?? []) {\n      const resolvedServiceName = dependsOnServiceName.serviceName;\n      if (resolvedServiceName === serviceName) {\n        throw new Error(`Service ${serviceName} cannot depend on itself`);\n      }\n      if (!serviceDescriptions[resolvedServiceName]) {\n        throw new Error(\n          `Unable to resolve service named ${resolvedServiceName} for ${serviceName}`\n        );\n      }\n\n      dependsOn.push(resolvedServiceName);\n    }\n\n    // Give each volume binding a chance to bind any necessary volume\n    // configuration and provide volume mount information for the service.\n    const volumes: DockerComposeVolumeMount[] = [];\n    for (const volumeBinding of serviceDescription.volumes ?? []) {\n      volumes.push(volumeBinding.bind(volumeInfo));\n    }\n\n    // Give each network binding a chance to bind any necessary network\n    // configuration and provide network mount information for the service.\n    const networks: string[] = [];\n    for (const networkBinding of serviceDescription.networks ?? []) {\n      networks.push(networkBinding.bind(networkInfo));\n    }\n\n    // Create and store the service configuration, taking care not to create\n    // object members with undefined values.\n    services[serviceName] = {\n      ...getObjectWithKeyAndValueIfValueIsDefined(\n        \"image\",\n        serviceDescription.image\n      ),\n      ...getObjectWithKeyAndValueIfValueIsDefined(\n        \"build\",\n        serviceDescription.imageBuild\n      ),\n      ...getObjectWithKeyAndValueIfValueIsDefined(\n        \"entrypoint\",\n        serviceDescription.entrypoint\n      ),\n      ...getObjectWithKeyAndValueIfValueIsDefined(\n        \"command\",\n        serviceDescription.command\n      ),\n      ...getObjectWithKeyAndValueIfValueIsDefined(\n        \"platform\",\n        serviceDescription.platform\n      ),\n      ...getObjectWithKeyAndValueIfValueIsDefined(\n        \"privileged\",\n        serviceDescription.privileged\n      ),\n      ...(Object.keys(serviceDescription.environment).length > 0\n        ? { environment: serviceDescription.environment }\n        : {}),\n      ...(serviceDescription.ports.length > 0\n        ? { ports: serviceDescription.ports }\n        : {}),\n      ...(Object.keys(serviceDescription.labels).length > 0\n        ? { labels: serviceDescription.labels }\n        : {}),\n      ...(dependsOn.length > 0 ? { dependsOn } : {}),\n      ...(volumes.length > 0 ? { volumes } : {}),\n      ...(networks.length > 0 ? { networks } : {}),\n    };\n  }\n\n  // Explicit with the type here because the decamelize step after this wipes\n  // out types.\n  const input: DockerComposeFileSchema = {\n    ...(version ? { version } : {}),\n    services,\n    ...(Object.keys(volumeConfig).length > 0 ? { volumes: volumeConfig } : {}),\n    ...(Object.keys(networkConfig).length > 0\n      ? { networks: networkConfig }\n      : {}),\n  };\n\n  // Change most keys to snake case.\n  return decamelizeKeysRecursively(input, {\n    shouldDecamelize: shouldDecamelizeDockerComposeKey,\n    separator: \"_\",\n  });\n}\n\n/**\n * Returns `{ [key]: value }` if `value` is defined, otherwise returns `{}` so\n * that object spreading can be used to generate a peculiar interface.\n * @param key\n * @param value\n */\nfunction getObjectWithKeyAndValueIfValueIsDefined<K extends string, T>(\n  key: K,\n  value: T\n): { K: T } | {} {\n  return value !== undefined ? { [key]: value } : {};\n}\n\n/**\n * Determines whether the key at the given path should be decamelized.\n * Largely, all keys should be snake cased. But, there are some\n * exceptions for user-provided names for services, volumes, and\n * environment variables.\n *\n * @param path\n */\nfunction shouldDecamelizeDockerComposeKey(path: string[]) {\n  const poundPath = path.join(\"#\");\n\n  // Does not decamelize user's names.\n  // services.namehere:\n  // volumes.namehere:\n  // networks.namehere:\n  if (/^(services|volumes|networks)#[^#]+$/.test(poundPath)) {\n    return false;\n  }\n\n  // Does not decamelize environment variables and labels\n  // services.namehere.environment.*\n  // services.namehere.labels.*\n  if (/^services#[^#]+#(environment|labels)#/.test(poundPath)) {\n    return false;\n  }\n\n  // Does not decamelize build arguments\n  // services.namehere.build.args.*\n  if (/^services#[^#]+#build#args#/.test(poundPath)) {\n    return false;\n  }\n\n  // Otherwise, let it all decamelize.\n  return true;\n}\n"]}