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
JavaScript
"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"]}