projen
Version:
CDK for software projects
135 lines • 21.3 kB
JavaScript
;
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 type { DockerComposeBuild } from \"./docker-compose\";\nimport type {\n  DockerComposeNetworkConfig,\n  IDockerComposeNetworkConfig,\n} from \"./docker-compose-network\";\nimport type { DockerComposeServicePort } from \"./docker-compose-port\";\nimport type { DockerComposeService } from \"./docker-compose-service\";\nimport type {\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"]}