UNPKG

aws-logs-comptroller

Version:

Set Log Retention and prune orphaned LogGroups on a schedule using Step Functions service integrations and intrinsic functions.

234 lines 27.6 kB
"use strict"; Object.defineProperty(exports, "__esModule", { value: true }); exports.getRunner = void 0; const aws_iam_1 = require("aws-cdk-lib/aws-iam"); const aws_logs_1 = require("aws-cdk-lib/aws-logs"); const aws_stepfunctions_1 = require("aws-cdk-lib/aws-stepfunctions"); const aws_stepfunctions_tasks_1 = require("aws-cdk-lib/aws-stepfunctions-tasks"); exports.getRunner = (scope, retentionInDays = aws_logs_1.RetentionDays.ONE_WEEK) => { /** * Input includes `LogGroups`, `Stats`, and `Token`. * Fan out on `LogGroups` (should be a max of 50). */ const map = new aws_stepfunctions_1.Map(scope, 'Map', { inputPath: '$.LogGroups', maxConcurrency: 10, resultPath: '$.MapResult', }); /** * Split LogGroupName into parts. We're looking for `lambda` in the second place. */ const getLGParts = new aws_stepfunctions_1.Pass(scope, 'GetLGParts', { parameters: { 'LGParts.$': 'States.StringSplit($.LogGroupName, \'/\')', }, resultPath: aws_stepfunctions_1.JsonPath.stringAt('$.Function'), }); /** * If we don't have at least two segments, `getLogType` will fail. Get the length here. * Can't seem to nest another intrinsic in `States.ArrayLength`. */ const getArrayLen = new aws_stepfunctions_1.Pass(scope, 'GetArrayLen', { parameters: { Len: aws_stepfunctions_1.JsonPath.numberAt('States.ArrayLength($.Function.LGParts)'), }, resultPath: '$.Array', }); const twoOrMoreParts = new aws_stepfunctions_1.Choice(scope, 'TwoOrMore?'); /** * Get the second part. We're looking for `lambda` to see if this is a LogGroup for a Lambda Function. */ const getLogType = new aws_stepfunctions_1.Pass(scope, 'GetLogType', { parameters: { LogType: aws_stepfunctions_1.JsonPath.numberAt('States.ArrayGetItem($.Function.LGParts, 1)'), }, resultPath: '$.Log', }); const isLambdaLog = new aws_stepfunctions_1.Choice(scope, 'IsLambdaLog?'); /** * The third part of the LogGroup name should be the name of the Lambda Function. */ const getFnName = new aws_stepfunctions_1.Pass(scope, 'GetFnName', { parameters: { FunctionName: aws_stepfunctions_1.JsonPath.stringAt('States.ArrayGetItem($.Function.LGParts, 2)'), }, resultPath: '$.Function', }); const isFnPresent = new aws_stepfunctions_1.Choice(scope, 'FunctionPresent?'); /** * Make API call to get the Lambda Function. * If this call is successful, then we have a LogGroup for an active Lambda Function. * If we get a 404 back, then the Lambda Function no longer exists and the LogGroup isn't needed. */ const getFn = new aws_stepfunctions_tasks_1.CallAwsService(scope, 'GetFunction', { action: 'getFunction', iamResources: ['*'], parameters: { 'FunctionName.$': '$.Function.FunctionName', }, resultPath: aws_stepfunctions_1.JsonPath.DISCARD, service: 'lambda', }); /** * Make an API call to delete a LogGroup. * This will end a branch of the Map State. * The resultSelector indicates the LogGroup was deleted. */ const deleteLG = new aws_stepfunctions_tasks_1.CallAwsService(scope, 'DeleteLG', { action: 'deleteLogGroup', iamAction: 'logs:DeleteLogGroup', iamResources: ['*'], parameters: { LogGroupName: aws_stepfunctions_1.JsonPath.stringAt('$.LogGroupName'), }, resultSelector: { IsDeleted: 1, IsRetained: 0, }, service: 'cloudwatchlogs', }); /** * Make an API call to set retention on the LogGroup. * This will end a branch of the Map State. * The resultSelector indicates retention was added. */ const addRetention = new aws_stepfunctions_tasks_1.CallAwsService(scope, 'AddRetention', { action: 'putRetentionPolicy', iamAction: 'logs:PutRetentionPolicy', iamResources: ['*'], parameters: { LogGroupName: aws_stepfunctions_1.JsonPath.stringAt('$.LogGroupName'), RetentionInDays: retentionInDays, }, resultSelector: { IsDeleted: 0, IsRetained: 1, }, service: 'cloudwatchlogs', }); /** * For any log group that survived the pruning above, check to see if it already has retention. * If it doesn't have retention, then set it to the `retentionInDays` prop. */ const hasRetention = new aws_stepfunctions_1.Choice(scope, 'HasRetention?') .when(aws_stepfunctions_1.Condition.isNotPresent('$.RetentionInDays'), addRetention) .otherwise(new aws_stepfunctions_1.Pass(scope, 'lgtm', { result: aws_stepfunctions_1.Result.fromObject({ IsDeleted: 0, IsRetained: 0, }), })); /** * Initialize the loop for adding up the stats. */ const initStatsLoop = new aws_stepfunctions_1.Pass(scope, 'InitStatsLoop', { parameters: { Index: 0, ResultLen: aws_stepfunctions_1.JsonPath.numberAt('States.ArrayLength($.MapResult)'), }, resultPath: '$.Iterator', }); /** * Get the next item of the array by index. */ const getNextResult = new aws_stepfunctions_1.Pass(scope, 'GetNextResult', { parameters: { 'Result.$': 'States.ArrayGetItem($.MapResult, $.Iterator.Index)', }, resultPath: '$.R', }); /** * Increment stats based on the `IsDeleted` and `IsRetained` values of the map result. */ const incrementStats = new aws_stepfunctions_1.Pass(scope, 'IncrementStats', { parameters: { 'LGsDeleted.$': 'States.MathAdd($.Stats.LGsDeleted, $.R.Result.IsDeleted)', 'LGsRetained.$': 'States.MathAdd($.Stats.LGsRetained, $.R.Result.IsRetained)', LGsSeen: aws_stepfunctions_1.JsonPath.numberAt('$.Stats.LGsSeen'), }, resultPath: '$.Stats', }); /** * Increment the counter for the next loop. */ const incrementCounter = new aws_stepfunctions_1.Pass(scope, 'IncrementCounter', { parameters: { 'Index.$': 'States.MathAdd($.Iterator.Index, 1)', ResultLen: aws_stepfunctions_1.JsonPath.numberAt('$.Iterator.ResultLen'), }, resultPath: '$.Iterator', }); /** * Return the Task Token to the parent state machine and pass stats back. */ const sendSuccess = new aws_stepfunctions_tasks_1.CallAwsService(scope, 'SendSuccess', { action: 'sendTaskSuccess', iamAction: 'states:SendTaskSuccess', iamResources: ['*'], parameters: { 'Output.$': '$.Stats', 'TaskToken.$': '$.Token', }, service: 'sfn', }); const hasNextMapResult = new aws_stepfunctions_1.Choice(scope, 'HasNextMapResult?'); /** * Iterate over LogGroups, parsing the name to determine if it's a Lambda Log or not. */ map.iterator(getLGParts); getLGParts.next(getArrayLen); getArrayLen.next(twoOrMoreParts); twoOrMoreParts.when(aws_stepfunctions_1.Condition.numberGreaterThanEquals('$.Array.Len', 2), getLogType); twoOrMoreParts.otherwise(hasRetention); getLogType.next(isLambdaLog); /** * For each Lambda log, get the function name as the 3rd part of the LogGroup name. * Call the Lambda service to see if that function still exists. * If the function doesn't exist, then delete the log. */ isLambdaLog.when(aws_stepfunctions_1.Condition.stringEquals('$.Log.LogType', 'lambda'), getFnName); isLambdaLog.otherwise(hasRetention); getFnName.next(isFnPresent); isFnPresent.when(aws_stepfunctions_1.Condition.isNotNull('$.Function.FunctionName'), getFn); isFnPresent.otherwise(hasRetention); getFn.next(hasRetention); getFn.addCatch(deleteLG, { errors: [aws_stepfunctions_1.Errors.TASKS_FAILED], resultPath: aws_stepfunctions_1.JsonPath.DISCARD, }); /** * After the map completes, loop over the results in order to track how many LogGroups were deleted * and how many had retention set. */ map.next(initStatsLoop); initStatsLoop.next(hasNextMapResult); hasNextMapResult.when(aws_stepfunctions_1.Condition.numberLessThanJsonPath('$.Iterator.Index', '$.Iterator.ResultLen'), getNextResult); getNextResult.next(incrementStats); incrementStats.next(incrementCounter); incrementCounter.next(hasNextMapResult); /** * Finally send the stats back to the parent state machine. */ hasNextMapResult.otherwise(sendSuccess); /** * These calls may be throttled, so retry many times. * We probably don't need 10 retries, but the default wasn't enough. */ addRetention.addRetry({ maxAttempts: 10, }); deleteLG.addRetry({ maxAttempts: 10, }); const sm = new aws_stepfunctions_1.StateMachine(scope, 'LogsComptrollerRunner', { definition: map, stateMachineName: 'logs-comptroller-runner', tracingEnabled: true, }); /** * Workaround for CDK throwing a circular dependency error when attempting `grantTaskResponse`. */ sm.addToRolePolicy(new aws_iam_1.PolicyStatement({ actions: ['states:SendTaskSuccess'], resources: ['*'], })); return sm; }; //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"runner-state-machine.js","sourceRoot":"","sources":["../src/runner-state-machine.ts"],"names":[],"mappings":"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","sourcesContent":["import { PolicyStatement } from 'aws-cdk-lib/aws-iam';\nimport { RetentionDays } from 'aws-cdk-lib/aws-logs';\nimport { Choice, Condition, Errors, JsonPath, Map, Pass, Result, StateMachine } from 'aws-cdk-lib/aws-stepfunctions';\nimport { CallAwsService } from 'aws-cdk-lib/aws-stepfunctions-tasks';\nimport { Construct } from 'constructs';\n\nexport const getRunner = (\n  scope: Construct,\n  retentionInDays: RetentionDays = RetentionDays.ONE_WEEK,\n): StateMachine => {\n  /**\n   * Input includes `LogGroups`, `Stats`, and `Token`.\n   * Fan out on `LogGroups` (should be a max of 50).\n   */\n  const map = new Map(scope, 'Map', {\n    inputPath: '$.LogGroups',\n    maxConcurrency: 10,\n    resultPath: '$.MapResult',\n  });\n\n  /**\n   * Split LogGroupName into parts. We're looking for `lambda` in the second place.\n   */\n  const getLGParts = new Pass(scope, 'GetLGParts', {\n    parameters: {\n      'LGParts.$': 'States.StringSplit($.LogGroupName, \\'/\\')',\n    },\n    resultPath: JsonPath.stringAt('$.Function'),\n  });\n\n  /**\n   * If we don't have at least two segments, `getLogType` will fail. Get the length here.\n   * Can't seem to nest another intrinsic in `States.ArrayLength`.\n   */\n  const getArrayLen = new Pass(scope, 'GetArrayLen', {\n    parameters: {\n      Len: JsonPath.numberAt('States.ArrayLength($.Function.LGParts)'),\n    },\n    resultPath: '$.Array',\n  });\n\n  const twoOrMoreParts = new Choice(scope, 'TwoOrMore?');\n\n  /**\n   * Get the second part. We're looking for `lambda` to see if this is a LogGroup for a Lambda Function.\n   */\n  const getLogType = new Pass(scope, 'GetLogType', {\n    parameters: {\n      LogType: JsonPath.numberAt('States.ArrayGetItem($.Function.LGParts, 1)'),\n    },\n    resultPath: '$.Log',\n  });\n\n  const isLambdaLog = new Choice(scope, 'IsLambdaLog?');\n\n  /**\n   * The third part of the LogGroup name should be the name of the Lambda Function.\n   */\n  const getFnName = new Pass(scope, 'GetFnName', {\n    parameters: {\n      FunctionName: JsonPath.stringAt(\n        'States.ArrayGetItem($.Function.LGParts, 2)',\n      ),\n    },\n    resultPath: '$.Function',\n  });\n\n  const isFnPresent = new Choice(scope, 'FunctionPresent?');\n\n  /**\n   * Make API call to get the Lambda Function.\n   * If this call is successful, then we have a LogGroup for an active Lambda Function.\n   * If we get a 404 back, then the Lambda Function no longer exists and the LogGroup isn't needed.\n   */\n  const getFn = new CallAwsService(scope, 'GetFunction', {\n    action: 'getFunction',\n    iamResources: ['*'],\n    parameters: {\n      'FunctionName.$': '$.Function.FunctionName',\n    },\n    resultPath: JsonPath.DISCARD,\n    service: 'lambda',\n  });\n\n  /**\n   * Make an API call to delete a LogGroup.\n   * This will end a branch of the Map State.\n   * The resultSelector indicates the LogGroup was deleted.\n   */\n  const deleteLG = new CallAwsService(scope, 'DeleteLG', {\n    action: 'deleteLogGroup',\n    iamAction: 'logs:DeleteLogGroup',\n    iamResources: ['*'],\n    parameters: {\n      LogGroupName: JsonPath.stringAt('$.LogGroupName'),\n    },\n    resultSelector: {\n      IsDeleted: 1, IsRetained: 0,\n    },\n    service: 'cloudwatchlogs',\n  });\n\n  /**\n   * Make an API call to set retention on the LogGroup.\n   * This will end a branch of the Map State.\n   * The resultSelector indicates retention was added.\n   */\n  const addRetention = new CallAwsService(scope, 'AddRetention', {\n    action: 'putRetentionPolicy',\n    iamAction: 'logs:PutRetentionPolicy',\n    iamResources: ['*'],\n    parameters: {\n      LogGroupName: JsonPath.stringAt('$.LogGroupName'),\n      RetentionInDays: retentionInDays,\n    },\n    resultSelector: {\n      IsDeleted: 0, IsRetained: 1,\n    },\n    service: 'cloudwatchlogs',\n  });\n\n  /**\n   * For any log group that survived the pruning above, check to see if it already has retention.\n   * If it doesn't have retention, then set it to the `retentionInDays` prop.\n   */\n  const hasRetention = new Choice(scope, 'HasRetention?')\n    .when(Condition.isNotPresent('$.RetentionInDays'), addRetention)\n    .otherwise(\n      new Pass(scope, 'lgtm', {\n        result: Result.fromObject({\n          IsDeleted: 0, IsRetained: 0,\n        }),\n      }),\n    );\n\n  /**\n   * Initialize the loop for adding up the stats.\n   */\n  const initStatsLoop = new Pass(scope, 'InitStatsLoop', {\n    parameters: {\n      Index: 0,\n      ResultLen: JsonPath.numberAt('States.ArrayLength($.MapResult)'),\n    },\n    resultPath: '$.Iterator',\n  });\n\n  /**\n   * Get the next item of the array by index.\n   */\n  const getNextResult = new Pass(scope, 'GetNextResult', {\n    parameters: {\n      'Result.$': 'States.ArrayGetItem($.MapResult, $.Iterator.Index)',\n    },\n    resultPath: '$.R',\n  });\n\n  /**\n   * Increment stats based on the `IsDeleted` and `IsRetained` values of the map result.\n   */\n  const incrementStats = new Pass(scope, 'IncrementStats', {\n    parameters: {\n      'LGsDeleted.$':\n        'States.MathAdd($.Stats.LGsDeleted, $.R.Result.IsDeleted)',\n      'LGsRetained.$':\n        'States.MathAdd($.Stats.LGsRetained, $.R.Result.IsRetained)',\n      LGsSeen: JsonPath.numberAt('$.Stats.LGsSeen'),\n    },\n    resultPath: '$.Stats',\n  });\n\n  /**\n   * Increment the counter for the next loop.\n   */\n  const incrementCounter = new Pass(scope, 'IncrementCounter', {\n    parameters: {\n      'Index.$': 'States.MathAdd($.Iterator.Index, 1)',\n      ResultLen: JsonPath.numberAt('$.Iterator.ResultLen'),\n    },\n    resultPath: '$.Iterator',\n  });\n\n  /**\n   * Return the Task Token to the parent state machine and pass stats back.\n   */\n  const sendSuccess = new CallAwsService(scope, 'SendSuccess', {\n    action: 'sendTaskSuccess',\n    iamAction: 'states:SendTaskSuccess',\n    iamResources: ['*'],\n    parameters: {\n      'Output.$': '$.Stats',\n      'TaskToken.$': '$.Token',\n    },\n    service: 'sfn',\n  });\n\n  const hasNextMapResult = new Choice(scope, 'HasNextMapResult?');\n\n  /**\n   * Iterate over LogGroups, parsing the name to determine if it's a Lambda Log or not.\n   */\n  map.iterator(getLGParts);\n  getLGParts.next(getArrayLen);\n  getArrayLen.next(twoOrMoreParts);\n  twoOrMoreParts.when(\n    Condition.numberGreaterThanEquals('$.Array.Len', 2),\n    getLogType,\n  );\n  twoOrMoreParts.otherwise(hasRetention);\n  getLogType.next(isLambdaLog);\n\n  /**\n   * For each Lambda log, get the function name as the 3rd part of the LogGroup name.\n   * Call the Lambda service to see if that function still exists.\n   * If the function doesn't exist, then delete the log.\n   */\n  isLambdaLog.when(\n    Condition.stringEquals('$.Log.LogType', 'lambda'),\n    getFnName,\n  );\n  isLambdaLog.otherwise(hasRetention);\n  getFnName.next(isFnPresent);\n  isFnPresent.when(Condition.isNotNull('$.Function.FunctionName'), getFn);\n  isFnPresent.otherwise(hasRetention);\n  getFn.next(hasRetention);\n  getFn.addCatch(deleteLG, {\n    errors: [Errors.TASKS_FAILED],\n    resultPath: JsonPath.DISCARD,\n  });\n\n  /**\n   * After the map completes, loop over the results in order to track how many LogGroups were deleted\n   * and how many had retention set.\n   */\n  map.next(initStatsLoop);\n  initStatsLoop.next(hasNextMapResult);\n  hasNextMapResult.when(\n    Condition.numberLessThanJsonPath(\n      '$.Iterator.Index',\n      '$.Iterator.ResultLen',\n    ),\n    getNextResult,\n  );\n  getNextResult.next(incrementStats);\n\n  incrementStats.next(incrementCounter);\n  incrementCounter.next(hasNextMapResult);\n\n  /**\n   * Finally send the stats back to the parent state machine.\n   */\n  hasNextMapResult.otherwise(sendSuccess);\n\n  /**\n   * These calls may be throttled, so retry many times.\n   * We probably don't need 10 retries, but the default wasn't enough.\n   */\n  addRetention.addRetry({\n    maxAttempts: 10,\n  });\n  deleteLG.addRetry({\n    maxAttempts: 10,\n  });\n\n  const sm = new StateMachine(scope, 'LogsComptrollerRunner', {\n    definition: map,\n    stateMachineName: 'logs-comptroller-runner',\n    tracingEnabled: true,\n  });\n\n  /**\n   * Workaround for CDK throwing a circular dependency error when attempting `grantTaskResponse`.\n   */\n  sm.addToRolePolicy(\n    new PolicyStatement({\n      actions: ['states:SendTaskSuccess'],\n      resources: ['*'],\n    }),\n  );\n\n  return sm;\n};\n"]}