datadog-ecs-cdk
Advanced tools
Comparing version 1.1.1 to 1.2.0
@@ -1,54 +0,2 @@ | ||
import { Construct } from 'constructs'; | ||
import * as ecs from 'aws-cdk-lib/aws-ecs'; | ||
import type * as secretsmanager from 'aws-cdk-lib/aws-secretsmanager'; | ||
export interface EcsDatadogDaemonServiceProps { | ||
readonly ecsCluster: ecs.Cluster; | ||
readonly datadogApiKeySecret: secretsmanager.ISecret; | ||
readonly datadogSite?: string; | ||
readonly logsDisabled?: boolean; | ||
} | ||
export declare class EcsDatadogDaemonService extends Construct { | ||
constructor(scope: Construct, id: string, props: EcsDatadogDaemonServiceProps); | ||
} | ||
export interface AddDatadogToFargateTaskProps { | ||
datadogApiKeySecret: ecs.Secret; | ||
datadogSite?: string; | ||
agent?: { | ||
enabled?: boolean; | ||
image?: ecs.ContainerImage; | ||
imageTag?: string; | ||
memoryLimitMiB?: number; | ||
cpu?: number; | ||
logToCloudWatch?: boolean; | ||
apm?: { | ||
enabled?: boolean; | ||
port?: number; | ||
applicationEnvVars?: { | ||
doNotSet?: boolean; | ||
apmHostEnvVarName?: string; | ||
apmPortEnvVarName?: string; | ||
apmTraceEnabledEnvVarName?: string; | ||
}; | ||
}; | ||
statsd?: { | ||
enabled?: boolean; | ||
port?: number; | ||
applicationEnvVars?: { | ||
doNotSet?: boolean; | ||
statsdHostEnvVarName?: string; | ||
statsdPortEnvVarName?: string; | ||
}; | ||
}; | ||
}; | ||
fireLensLogging?: { | ||
enabled?: boolean; | ||
service?: string; | ||
source?: string; | ||
tags?: Record<string, string>; | ||
memoryLimitMiB?: number; | ||
cpu?: number; | ||
image?: ecs.ContainerImage; | ||
imageTag?: string; | ||
}; | ||
} | ||
export declare const addDatadogToFargateTask: (task: ecs.TaskDefinition, props: AddDatadogToFargateTaskProps) => void; | ||
export * from './ec2'; | ||
export * from './fargate'; |
251
index.js
"use strict"; | ||
Object.defineProperty(exports, "__esModule", { value: true }); | ||
exports.addDatadogToFargateTask = exports.EcsDatadogDaemonService = void 0; | ||
const constructs_1 = require("constructs"); | ||
const cdk = require("aws-cdk-lib"); | ||
const ecs = require("aws-cdk-lib/aws-ecs"); | ||
class EcsDatadogDaemonService extends constructs_1.Construct { | ||
constructor(scope, id, props) { | ||
var _a; | ||
super(scope, id); | ||
// To match as documented at | ||
// https://docs.datadoghq.com/containers/amazon_ecs/?tab=awscli#setup | ||
// and | ||
// https://docs.datadoghq.com/containers/amazon_ecs/logs/?tab=linux | ||
const taskDefinition = new ecs.Ec2TaskDefinition(this, 'TaskDefinition', { | ||
family: 'datadog-agent-task', | ||
}); | ||
const container = taskDefinition.addContainer('datadog-agent', { | ||
image: ecs.ContainerImage.fromRegistry('public.ecr.aws/datadog/agent:latest'), | ||
cpu: 100, | ||
memoryLimitMiB: 512, | ||
essential: true, | ||
environment: { | ||
DD_SITE: (_a = props.datadogSite) !== null && _a !== void 0 ? _a : 'datadoghq.com', | ||
...(props.logsDisabled | ||
? {} | ||
: { | ||
DD_LOGS_ENABLED: 'true', | ||
DD_LOGS_CONFIG_CONTAINER_COLLECT_ALL: 'true', | ||
}), | ||
}, | ||
secrets: { | ||
DD_API_KEY: ecs.Secret.fromSecretsManager(props.datadogApiKeySecret), | ||
}, | ||
healthCheck: { | ||
command: ['CMD-SHELL', 'agent health'], | ||
retries: 3, | ||
timeout: cdk.Duration.seconds(5), | ||
interval: cdk.Duration.seconds(10), | ||
startPeriod: cdk.Duration.seconds(15), | ||
}, | ||
logging: ecs.LogDrivers.awsLogs({ | ||
streamPrefix: 'datadog-agent', | ||
}), | ||
}); | ||
container.addMountPoints({ | ||
containerPath: '/var/run/docker.sock', | ||
sourceVolume: 'docker_sock', | ||
readOnly: true, | ||
}, { | ||
containerPath: '/host/sys/fs/cgroup', | ||
sourceVolume: 'cgroup', | ||
readOnly: true, | ||
}, { | ||
containerPath: '/host/proc', | ||
sourceVolume: 'proc', | ||
readOnly: true, | ||
}); | ||
if (!props.logsDisabled) { | ||
container.addMountPoints({ | ||
containerPath: '/opt/datadog-agent/run', | ||
sourceVolume: 'pointdir', | ||
readOnly: false, | ||
}, { | ||
containerPath: '/var/lib/docker/containers', | ||
sourceVolume: 'containers_root', | ||
readOnly: true, | ||
}); | ||
} | ||
taskDefinition.addVolume({ | ||
name: 'docker_sock', | ||
host: { | ||
sourcePath: '/var/run/docker.sock', | ||
}, | ||
}); | ||
taskDefinition.addVolume({ | ||
name: 'proc', | ||
host: { | ||
sourcePath: '/proc/', | ||
}, | ||
}); | ||
taskDefinition.addVolume({ | ||
name: 'cgroup', | ||
host: { | ||
sourcePath: '/sys/fs/cgroup/', | ||
}, | ||
}); | ||
if (!props.logsDisabled) { | ||
taskDefinition.addVolume({ | ||
name: 'pointdir', | ||
host: { | ||
sourcePath: '/opt/datadog-agent/run', | ||
}, | ||
}); | ||
taskDefinition.addVolume({ | ||
name: 'containers_root', | ||
host: { | ||
sourcePath: '/var/lib/docker/containers', | ||
}, | ||
}); | ||
} | ||
new ecs.Ec2Service(this, 'Service', { | ||
cluster: props.ecsCluster, | ||
taskDefinition, | ||
daemon: true, | ||
}); | ||
var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) { | ||
if (k2 === undefined) k2 = k; | ||
var desc = Object.getOwnPropertyDescriptor(m, k); | ||
if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) { | ||
desc = { enumerable: true, get: function() { return m[k]; } }; | ||
} | ||
} | ||
exports.EcsDatadogDaemonService = EcsDatadogDaemonService; | ||
const formatTags = (tags) => { | ||
const formattedTags = []; | ||
for (const [key, value] of Object.entries(tags)) { | ||
formattedTags.push(`${key}:${value}`); | ||
} | ||
return formattedTags.join(','); | ||
Object.defineProperty(o, k2, desc); | ||
}) : (function(o, m, k, k2) { | ||
if (k2 === undefined) k2 = k; | ||
o[k2] = m[k]; | ||
})); | ||
var __exportStar = (this && this.__exportStar) || function(m, exports) { | ||
for (var p in m) if (p !== "default" && !Object.prototype.hasOwnProperty.call(exports, p)) __createBinding(exports, m, p); | ||
}; | ||
const addDatadogToFargateTask = (task, props) => { | ||
var _a, _b, _c, _d, _e, _f, _g, _h, _j, _k, _l, _m, _o, _p, _q, _r, _s, _t, _u, _v, _w, _x, _y, _z, _0, _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; | ||
const containerNames = []; | ||
// eslint-disable-next-line @typescript-eslint/ban-ts-comment | ||
// @ts-ignore | ||
const containers = task.containers; | ||
for (const container of containers) { | ||
containerNames.push(container.containerName); | ||
} | ||
if ((_a = props.agent) === null || _a === void 0 ? void 0 : _a.enabled) { | ||
task.addContainer('datadog-agent', { | ||
image: (_c = (_b = props.agent) === null || _b === void 0 ? void 0 : _b.image) !== null && _c !== void 0 ? _c : ecs.ContainerImage.fromRegistry(`public.ecr.aws/datadog/agent:${(_e = (_d = props.agent) === null || _d === void 0 ? void 0 : _d.imageTag) !== null && _e !== void 0 ? _e : 'latest'}`), | ||
memoryLimitMiB: (_g = (_f = props.agent) === null || _f === void 0 ? void 0 : _f.memoryLimitMiB) !== null && _g !== void 0 ? _g : 256, | ||
...(((_h = props.agent) === null || _h === void 0 ? void 0 : _h.cpu) | ||
? { | ||
cpu: (_j = props.fireLensLogging) === null || _j === void 0 ? void 0 : _j.cpu, | ||
} | ||
: {}), | ||
...(((_k = props.agent) === null || _k === void 0 ? void 0 : _k.logToCloudWatch) | ||
? { | ||
logging: ecs.LogDrivers.awsLogs({ | ||
streamPrefix: 'datadog-agent', | ||
}), | ||
} | ||
: {}), | ||
environment: { | ||
ECS_FARGATE: 'true', | ||
DD_SITE: (_l = props.datadogSite) !== null && _l !== void 0 ? _l : 'datadoghq.com', | ||
...(((_o = (_m = props.agent) === null || _m === void 0 ? void 0 : _m.apm) === null || _o === void 0 ? void 0 : _o.enabled) | ||
? { | ||
DD_APM_ENABLED: 'true', | ||
DD_APM_RECEIVER_PORT: String((_r = (_q = (_p = props.agent) === null || _p === void 0 ? void 0 : _p.apm) === null || _q === void 0 ? void 0 : _q.port) !== null && _r !== void 0 ? _r : '8126'), | ||
} | ||
: { | ||
DD_APM_ENABLED: 'false', | ||
}), | ||
...(((_t = (_s = props.agent) === null || _s === void 0 ? void 0 : _s.statsd) === null || _t === void 0 ? void 0 : _t.enabled) | ||
? { | ||
DD_USE_DOGSTATSD: 'true', | ||
DD_DOGSTATSD_PORT: String((_w = (_v = (_u = props.agent) === null || _u === void 0 ? void 0 : _u.statsd) === null || _v === void 0 ? void 0 : _v.port) !== null && _w !== void 0 ? _w : '8125'), | ||
} | ||
: { | ||
DD_USE_DOGSTATSD: 'false', | ||
}), | ||
}, | ||
secrets: { | ||
DD_API_KEY: props.datadogApiKeySecret, | ||
}, | ||
}); | ||
if (((_y = (_x = props.agent) === null || _x === void 0 ? void 0 : _x.apm) === null || _y === void 0 ? void 0 : _y.enabled) && ((_1 = (_0 = (_z = props.agent) === null || _z === void 0 ? void 0 : _z.apm) === null || _0 === void 0 ? void 0 : _0.applicationEnvVars) === null || _1 === void 0 ? void 0 : _1.doNotSet) !== true) { | ||
for (const containerName of containerNames) { | ||
const container = task.findContainer(containerName); | ||
if (container) { | ||
container.addEnvironment((_4 = (_3 = (_2 = props.agent) === null || _2 === void 0 ? void 0 : _2.apm.applicationEnvVars) === null || _3 === void 0 ? void 0 : _3.apmTraceEnabledEnvVarName) !== null && _4 !== void 0 ? _4 : 'DD_TRACE_ENABLED', 'true'); | ||
container.addEnvironment((_7 = (_6 = (_5 = props.agent) === null || _5 === void 0 ? void 0 : _5.apm.applicationEnvVars) === null || _6 === void 0 ? void 0 : _6.apmHostEnvVarName) !== null && _7 !== void 0 ? _7 : 'DD_AGENT_HOST', 'localhost'); | ||
container.addEnvironment((_10 = (_9 = (_8 = props.agent) === null || _8 === void 0 ? void 0 : _8.apm.applicationEnvVars) === null || _9 === void 0 ? void 0 : _9.apmPortEnvVarName) !== null && _10 !== void 0 ? _10 : 'DD_TRACE_AGENT_PORT', String((_13 = (_12 = (_11 = props.agent) === null || _11 === void 0 ? void 0 : _11.apm) === null || _12 === void 0 ? void 0 : _12.port) !== null && _13 !== void 0 ? _13 : '8126')); | ||
} | ||
} | ||
} | ||
if (((_15 = (_14 = props.agent) === null || _14 === void 0 ? void 0 : _14.statsd) === null || _15 === void 0 ? void 0 : _15.enabled) && ((_18 = (_17 = (_16 = props.agent) === null || _16 === void 0 ? void 0 : _16.statsd) === null || _17 === void 0 ? void 0 : _17.applicationEnvVars) === null || _18 === void 0 ? void 0 : _18.doNotSet) !== true) { | ||
for (const containerName of containerNames) { | ||
const container = task.findContainer(containerName); | ||
if (container) { | ||
container.addEnvironment((_22 = (_21 = (_20 = (_19 = props.agent) === null || _19 === void 0 ? void 0 : _19.statsd) === null || _20 === void 0 ? void 0 : _20.applicationEnvVars) === null || _21 === void 0 ? void 0 : _21.statsdHostEnvVarName) !== null && _22 !== void 0 ? _22 : 'STATSD_HOST', 'localhost'); | ||
container.addEnvironment((_26 = (_25 = (_24 = (_23 = props.agent) === null || _23 === void 0 ? void 0 : _23.statsd) === null || _24 === void 0 ? void 0 : _24.applicationEnvVars) === null || _25 === void 0 ? void 0 : _25.statsdPortEnvVarName) !== null && _26 !== void 0 ? _26 : 'STATSD_PORT', String((_29 = (_28 = (_27 = props.agent) === null || _27 === void 0 ? void 0 : _27.statsd) === null || _28 === void 0 ? void 0 : _28.port) !== null && _29 !== void 0 ? _29 : '8125')); | ||
} | ||
} | ||
} | ||
} | ||
if ((_30 = props.fireLensLogging) === null || _30 === void 0 ? void 0 : _30.enabled) { | ||
task.addFirelensLogRouter('log_router', { | ||
image: (_32 = (_31 = props.fireLensLogging) === null || _31 === void 0 ? void 0 : _31.image) !== null && _32 !== void 0 ? _32 : ecs.ContainerImage.fromRegistry(`public.ecr.aws/datadog/aws-for-fluent-bit:${(_34 = (_33 = props.fireLensLogging) === null || _33 === void 0 ? void 0 : _33.imageTag) !== null && _34 !== void 0 ? _34 : 'latest'}`), | ||
memoryLimitMiB: (_36 = (_35 = props.fireLensLogging) === null || _35 === void 0 ? void 0 : _35.memoryLimitMiB) !== null && _36 !== void 0 ? _36 : 256, | ||
...(((_37 = props.fireLensLogging) === null || _37 === void 0 ? void 0 : _37.cpu) | ||
? { | ||
cpu: (_38 = props.fireLensLogging) === null || _38 === void 0 ? void 0 : _38.cpu, | ||
} | ||
: {}), | ||
firelensConfig: { | ||
type: ecs.FirelensLogRouterType.FLUENTBIT, | ||
options: { | ||
enableECSLogMetadata: true, | ||
}, | ||
}, | ||
}); | ||
const firelensLogDriver = ecs.LogDrivers.firelens({ | ||
options: { | ||
Name: 'datadog', | ||
Host: `http-intake.logs.${(_39 = props.datadogSite) !== null && _39 !== void 0 ? _39 : 'datadoghq.com'}`, | ||
TLS: 'on', | ||
provider: 'ecs', | ||
dd_message_key: 'log', | ||
...(((_40 = props.fireLensLogging) === null || _40 === void 0 ? void 0 : _40.service) | ||
? { | ||
dd_service: (_41 = props.fireLensLogging) === null || _41 === void 0 ? void 0 : _41.service, | ||
} | ||
: {}), | ||
...(((_42 = props.fireLensLogging) === null || _42 === void 0 ? void 0 : _42.source) | ||
? { | ||
dd_source: (_43 = props.fireLensLogging) === null || _43 === void 0 ? void 0 : _43.source, | ||
} | ||
: {}), | ||
...(((_44 = props.fireLensLogging) === null || _44 === void 0 ? void 0 : _44.tags) | ||
? { | ||
dd_tags: formatTags((_45 = props.fireLensLogging) === null || _45 === void 0 ? void 0 : _45.tags), | ||
} | ||
: {}), | ||
}, | ||
secretOptions: { | ||
apikey: props.datadogApiKeySecret, | ||
}, | ||
}); | ||
for (const containerName of containerNames) { | ||
const container = task.findContainer(containerName); | ||
// eslint-disable-next-line @typescript-eslint/ban-ts-comment | ||
// @ts-ignore | ||
container.logDriverConfig = firelensLogDriver.bind(container, container); | ||
} | ||
} | ||
}; | ||
exports.addDatadogToFargateTask = addDatadogToFargateTask; | ||
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["../index.ts"],"names":[],"mappings":"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","sourcesContent":["import { Construct } from 'constructs';\n\nimport * as cdk from 'aws-cdk-lib';\nimport * as ecs from 'aws-cdk-lib/aws-ecs';\nimport type * as secretsmanager from 'aws-cdk-lib/aws-secretsmanager';\n\nexport interface EcsDatadogDaemonServiceProps {\n    readonly ecsCluster: ecs.Cluster;\n    readonly datadogApiKeySecret: secretsmanager.ISecret;\n    readonly datadogSite?: string;\n    readonly logsDisabled?: boolean;\n}\n\nexport class EcsDatadogDaemonService extends Construct {\n    constructor(scope: Construct, id: string, props: EcsDatadogDaemonServiceProps) {\n        super(scope, id);\n\n        // To match as documented at\n        // https://docs.datadoghq.com/containers/amazon_ecs/?tab=awscli#setup\n        // and\n        // https://docs.datadoghq.com/containers/amazon_ecs/logs/?tab=linux\n        const taskDefinition = new ecs.Ec2TaskDefinition(this, 'TaskDefinition', {\n            family: 'datadog-agent-task',\n        });\n\n        const container = taskDefinition.addContainer('datadog-agent', {\n            image: ecs.ContainerImage.fromRegistry('public.ecr.aws/datadog/agent:latest'),\n            cpu: 100,\n            memoryLimitMiB: 512,\n            essential: true,\n            environment: {\n                DD_SITE: props.datadogSite ?? 'datadoghq.com',\n                ...(props.logsDisabled\n                    ? {}\n                    : {\n                          DD_LOGS_ENABLED: 'true',\n                          DD_LOGS_CONFIG_CONTAINER_COLLECT_ALL: 'true',\n                      }),\n            },\n            secrets: {\n                DD_API_KEY: ecs.Secret.fromSecretsManager(props.datadogApiKeySecret),\n            },\n            healthCheck: {\n                command: ['CMD-SHELL', 'agent health'],\n                retries: 3,\n                timeout: cdk.Duration.seconds(5),\n                interval: cdk.Duration.seconds(10),\n                startPeriod: cdk.Duration.seconds(15),\n            },\n            logging: ecs.LogDrivers.awsLogs({\n                streamPrefix: 'datadog-agent',\n            }),\n        });\n\n        container.addMountPoints(\n            {\n                containerPath: '/var/run/docker.sock',\n                sourceVolume: 'docker_sock',\n                readOnly: true,\n            },\n            {\n                containerPath: '/host/sys/fs/cgroup',\n                sourceVolume: 'cgroup',\n                readOnly: true,\n            },\n            {\n                containerPath: '/host/proc',\n                sourceVolume: 'proc',\n                readOnly: true,\n            },\n        );\n\n        if (!props.logsDisabled) {\n            container.addMountPoints(\n                {\n                    containerPath: '/opt/datadog-agent/run',\n                    sourceVolume: 'pointdir',\n                    readOnly: false,\n                },\n                {\n                    containerPath: '/var/lib/docker/containers',\n                    sourceVolume: 'containers_root',\n                    readOnly: true,\n                },\n            );\n        }\n\n        taskDefinition.addVolume({\n            name: 'docker_sock',\n            host: {\n                sourcePath: '/var/run/docker.sock',\n            },\n        });\n\n        taskDefinition.addVolume({\n            name: 'proc',\n            host: {\n                sourcePath: '/proc/',\n            },\n        });\n\n        taskDefinition.addVolume({\n            name: 'cgroup',\n            host: {\n                sourcePath: '/sys/fs/cgroup/',\n            },\n        });\n\n        if (!props.logsDisabled) {\n            taskDefinition.addVolume({\n                name: 'pointdir',\n                host: {\n                    sourcePath: '/opt/datadog-agent/run',\n                },\n            });\n\n            taskDefinition.addVolume({\n                name: 'containers_root',\n                host: {\n                    sourcePath: '/var/lib/docker/containers',\n                },\n            });\n        }\n\n        new ecs.Ec2Service(this, 'Service', {\n            cluster: props.ecsCluster,\n            taskDefinition,\n            daemon: true,\n        });\n    }\n}\n\nexport interface AddDatadogToFargateTaskProps {\n    // Note: it is important that the value specified by this secret\n    // doesn't have a newline at the end. Otherwise, the firelens\n    // logging configuration will fail to send logs to Datadog. This\n    // is an easy mistake to introduce when the source for the secret\n    // is a secretsmanager Secret containing a single value, rather\n    // than key-value pairs.\n    datadogApiKeySecret: ecs.Secret;\n    // Defaults to datadoghq.com\n    datadogSite?: string;\n    agent?: {\n        // Defaults to false\n        enabled?: boolean;\n        // Defaults to public.ecr.aws/datadog/agent:latest\n        image?: ecs.ContainerImage;\n        // Defaults to latest\n        imageTag?: string;\n        // Defaults to 256\n        memoryLimitMiB?: number;\n        // Defaults to unset\n        cpu?: number;\n        // Defaults to false\n        logToCloudWatch?: boolean;\n        apm?: {\n            // Defaults to false\n            enabled?: boolean;\n            // Defaults to 8126\n            port?: number;\n            applicationEnvVars?: {\n                // Defaults to false\n                doNotSet?: boolean;\n                // Defaults to DD_AGENT_HOST\n                apmHostEnvVarName?: string;\n                // Defaults to DD_TRACE_AGENT_PORT\n                apmPortEnvVarName?: string;\n                // Defaults to DD_TRACE_ENABLED\n                apmTraceEnabledEnvVarName?: string;\n            };\n        };\n        statsd?: {\n            // Defaults to false\n            enabled?: boolean;\n            // Defaults to 8125\n            port?: number;\n            applicationEnvVars?: {\n                // Defaults to false\n                doNotSet?: boolean;\n                // Defaults to STATSD_HOST\n                statsdHostEnvVarName?: string;\n                // Defaults to STATSD_PORT\n                statsdPortEnvVarName?: string;\n            };\n        };\n    };\n    fireLensLogging?: {\n        // Defaults to false\n        enabled?: boolean;\n        // Defaults to unset\n        service?: string;\n        // Defaults to unset\n        source?: string;\n        // Defaults to unset\n        tags?: Record<string, string>;\n        // Defaults to 256\n        memoryLimitMiB?: number;\n        // Defaults to unset\n        cpu?: number;\n        // Defaults to public.ecr.aws/datadog/aws-for-fluent-bit:latest\n        image?: ecs.ContainerImage;\n        // Defaults to latest\n        imageTag?: string;\n    };\n}\n\nconst formatTags = (tags: Record<string, string>): string => {\n    const formattedTags = [];\n    for (const [key, value] of Object.entries(tags)) {\n        formattedTags.push(`${key}:${value}`);\n    }\n    return formattedTags.join(',');\n};\n\nexport const addDatadogToFargateTask = (task: ecs.TaskDefinition, props: AddDatadogToFargateTaskProps) => {\n    const containerNames = [];\n    // eslint-disable-next-line @typescript-eslint/ban-ts-comment\n    // @ts-ignore\n    const containers = task.containers;\n    for (const container of containers) {\n        containerNames.push(container.containerName);\n    }\n\n    if (props.agent?.enabled) {\n        task.addContainer('datadog-agent', {\n            image: props.agent?.image ?? ecs.ContainerImage.fromRegistry(`public.ecr.aws/datadog/agent:${props.agent?.imageTag ?? 'latest'}`),\n            memoryLimitMiB: props.agent?.memoryLimitMiB ?? 256,\n            ...(props.agent?.cpu\n                ? {\n                      cpu: props.fireLensLogging?.cpu,\n                  }\n                : {}),\n            ...(props.agent?.logToCloudWatch\n                ? {\n                      logging: ecs.LogDrivers.awsLogs({\n                          streamPrefix: 'datadog-agent',\n                      }),\n                  }\n                : {}),\n            environment: {\n                ECS_FARGATE: 'true',\n                DD_SITE: props.datadogSite ?? 'datadoghq.com',\n                ...(props.agent?.apm?.enabled\n                    ? {\n                          DD_APM_ENABLED: 'true',\n                          DD_APM_RECEIVER_PORT: String(props.agent?.apm?.port ?? '8126'),\n                      }\n                    : {\n                          DD_APM_ENABLED: 'false',\n                      }),\n                ...(props.agent?.statsd?.enabled\n                    ? {\n                          DD_USE_DOGSTATSD: 'true',\n                          DD_DOGSTATSD_PORT: String(props.agent?.statsd?.port ?? '8125'),\n                      }\n                    : {\n                          DD_USE_DOGSTATSD: 'false',\n                      }),\n            },\n            secrets: {\n                DD_API_KEY: props.datadogApiKeySecret,\n            },\n        });\n\n        if (props.agent?.apm?.enabled && props.agent?.apm?.applicationEnvVars?.doNotSet !== true) {\n            for (const containerName of containerNames) {\n                const container = task.findContainer(containerName);\n                if (container) {\n                    container.addEnvironment(props.agent?.apm.applicationEnvVars?.apmTraceEnabledEnvVarName ?? 'DD_TRACE_ENABLED', 'true');\n                    container.addEnvironment(props.agent?.apm.applicationEnvVars?.apmHostEnvVarName ?? 'DD_AGENT_HOST', 'localhost');\n                    container.addEnvironment(props.agent?.apm.applicationEnvVars?.apmPortEnvVarName ?? 'DD_TRACE_AGENT_PORT', String(props.agent?.apm?.port ?? '8126'));\n                }\n            }\n        }\n\n        if (props.agent?.statsd?.enabled && props.agent?.statsd?.applicationEnvVars?.doNotSet !== true) {\n            for (const containerName of containerNames) {\n                const container = task.findContainer(containerName);\n                if (container) {\n                    container.addEnvironment(props.agent?.statsd?.applicationEnvVars?.statsdHostEnvVarName ?? 'STATSD_HOST', 'localhost');\n                    container.addEnvironment(props.agent?.statsd?.applicationEnvVars?.statsdPortEnvVarName ?? 'STATSD_PORT', String(props.agent?.statsd?.port ?? '8125'));\n                }\n            }\n        }\n    }\n\n    if (props.fireLensLogging?.enabled) {\n        task.addFirelensLogRouter('log_router', {\n            image: props.fireLensLogging?.image ?? ecs.ContainerImage.fromRegistry(`public.ecr.aws/datadog/aws-for-fluent-bit:${props.fireLensLogging?.imageTag ?? 'latest'}`),\n            memoryLimitMiB: props.fireLensLogging?.memoryLimitMiB ?? 256,\n            ...(props.fireLensLogging?.cpu\n                ? {\n                      cpu: props.fireLensLogging?.cpu,\n                  }\n                : {}),\n            firelensConfig: {\n                type: ecs.FirelensLogRouterType.FLUENTBIT,\n                options: {\n                    enableECSLogMetadata: true,\n                },\n            },\n        });\n\n        const firelensLogDriver = ecs.LogDrivers.firelens({\n            options: {\n                Name: 'datadog',\n                Host: `http-intake.logs.${props.datadogSite ?? 'datadoghq.com'}`,\n                TLS: 'on',\n                provider: 'ecs',\n                dd_message_key: 'log',\n                ...(props.fireLensLogging?.service\n                    ? {\n                          dd_service: props.fireLensLogging?.service,\n                      }\n                    : {}),\n                ...(props.fireLensLogging?.source\n                    ? {\n                          dd_source: props.fireLensLogging?.source,\n                      }\n                    : {}),\n                ...(props.fireLensLogging?.tags\n                    ? {\n                          dd_tags: formatTags(props.fireLensLogging?.tags),\n                      }\n                    : {}),\n            },\n            secretOptions: {\n                apikey: props.datadogApiKeySecret,\n            },\n        });\n\n        for (const containerName of containerNames) {\n            const container = task.findContainer(containerName);\n            // eslint-disable-next-line @typescript-eslint/ban-ts-comment\n            // @ts-ignore\n            container.logDriverConfig = firelensLogDriver.bind(container, container);\n        }\n    }\n};\n"]} | ||
Object.defineProperty(exports, "__esModule", { value: true }); | ||
__exportStar(require("./ec2"), exports); | ||
__exportStar(require("./fargate"), exports); | ||
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi9pbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7Ozs7Ozs7Ozs7Ozs7O0FBQUEsd0NBQXNCO0FBQ3RCLDRDQUEwQiIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCAqIGZyb20gJy4vZWMyJztcbmV4cG9ydCAqIGZyb20gJy4vZmFyZ2F0ZSc7XG4iXX0= |
{ | ||
"name": "datadog-ecs-cdk", | ||
"version": "1.1.1", | ||
"version": "1.2.0", | ||
"description": "", | ||
@@ -28,6 +28,6 @@ "license": "MIT", | ||
"version": "auto-changelog --hide-credit -p && git add CHANGELOG.md", | ||
"_lint": "eslint 'index.ts' 'test/**/*.ts' jest.config.js", | ||
"_lint": "eslint '*.ts' 'test/**/*.ts' jest.config.js", | ||
"lint": "npm run -- _lint --fix", | ||
"lint-check": "npm run -- _lint --max-warnings 0", | ||
"docs": "typedoc index.ts --includeVersion" | ||
"docs": "typedoc ec2.ts fargate.ts --includeVersion" | ||
}, | ||
@@ -34,0 +34,0 @@ "peerDependencies": { |
# datadog-ecs-cdk | ||
Docs: https://isotoma.github.io/datadog-ecs-cdk/ | ||
NPM: https://www.npmjs.com/package/datadog-ecs-cdk | ||
Source: https://github.com/isotoma/datadog-ecs-cdk | ||
## EC2 example | ||
```typescript | ||
import { EcsDatadogDaemonService } from 'datadog-ecs-cdk'; | ||
// ... | ||
new EcsDatadogDaemonService(this, 'EcsDatadog', { | ||
ecsCluster: myCluster, | ||
datadogApiKeySecret: mySecret, | ||
}); | ||
``` | ||
## Fargate example | ||
```typescript | ||
import { addDatadogToFargateTask } from 'datadog-ecs-cdk'; | ||
// ... | ||
const myTaskDef = ... | ||
addDatadogToFargateTask(myTaskDef, { | ||
datadogApiKeySecret: ecs.Secret.fromSecretsManager(mySecret), | ||
agent: { | ||
enabled: true, | ||
statsd: { | ||
enabled: true, | ||
}, | ||
}, | ||
fireLensLogging: { | ||
enabled: true, | ||
service: 'myservice', | ||
source: 'myservice', | ||
}, | ||
}); | ||
``` |
Major refactor
Supply chain riskPackage has recently undergone a major refactor. It may be unstable or indicate significant internal changes. Use caution when updating to versions that include significant changes.
Found 1 instance in 1 package
60639
8
574
45
1