UNPKG

cdk-athena-log

Version:

A CDK construct to create an Athena table for querying ALB logs.

85 lines 15.9 kB
"use strict"; var _a; Object.defineProperty(exports, "__esModule", { value: true }); exports.AthenaTableForWaf = void 0; const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti"); const aws_cdk_lib_1 = require("aws-cdk-lib"); const constructs_1 = require("constructs"); /** * S3に保存されたAWS WAFログをクエリするためのGlueテーブルを作成します。 * このコンストラクトは、公式ドキュメントのdate型パーティション射影に準拠します。 */ class AthenaTableForWaf extends constructs_1.Construct { constructor(scope, id, props) { super(scope, id); const { logBucketName, databaseName, tableName, wafScope, webAclName, projectionStartDate, // ★ プロパティを受け取る logPrefix, } = props; const stack = aws_cdk_lib_1.Stack.of(this); const account = stack.account; const logRegion = wafScope === 'CLOUDFRONT' ? 'cloudfront' : stack.region; const s3Prefix = logPrefix ? `${logPrefix}/` : ''; const s3BaseLocation = `s3://${logBucketName}/${s3Prefix}AWSLogs/${account}/WAFLogs/${logRegion}/${webAclName}/`; const s3LocationTemplate = `${s3BaseLocation}\${log_time}`; this.table = new aws_cdk_lib_1.aws_glue.CfnTable(this, 'Default', { catalogId: account, databaseName: databaseName, tableInput: { name: tableName, tableType: 'EXTERNAL_TABLE', parameters: { 'projection.enabled': 'true', 'projection.log_time.type': 'date', 'projection.log_time.format': 'yyyy/MM/dd/HH/mm', 'projection.log_time.range': `${projectionStartDate}/00/00,NOW`, 'projection.log_time.interval': '1', 'projection.log_time.interval.unit': 'MINUTES', 'storage.location.template': s3LocationTemplate, }, partitionKeys: [{ name: 'log_time', type: 'string' }], storageDescriptor: { columns: [ { name: 'timestamp', type: 'bigint' }, { name: 'formatversion', type: 'int' }, { name: 'webaclid', type: 'string' }, { name: 'terminatingruleid', type: 'string' }, { name: 'terminatingruletype', type: 'string' }, { name: 'action', type: 'string' }, { name: 'terminatingrulematchdetails', type: 'array<struct<conditiontype:string,sensitivitylevel:string,location:string,matcheddata:array<string>>>' }, { name: 'httpsourcename', type: 'string' }, { name: 'httpsourceid', type: 'string' }, { name: 'rulegrouplist', type: 'array<struct<rulegroupid:string,terminatingrule:struct<ruleid:string,action:string,rulematchdetails:string>,nonterminatingmatchingrules:array<struct<ruleid:string,action:string,rulematchdetails:string>>,excludedrules:string>>', }, { name: 'ratebasedrulelist', type: 'array<struct<ratebasedruleid:string,limitkey:string,maxrateallowed:int>>' }, { name: 'nonterminatingmatchingrules', type: 'array<struct<ruleid:string,action:string,rulematchdetails:string>>' }, { name: 'requestheadersinserted', type: 'array<struct<name:string,value:string>>' }, { name: 'responsecodesent', type: 'string' }, { name: 'httprequest', type: 'struct<clientip:string,country:string,headers:array<struct<name:string,value:string>>,uri:string,args:string,httpversion:string,httpmethod:string,requestid:string,fragment:string,scheme:string,host:string>', }, { name: 'labels', type: 'array<struct<name:string>>' }, { name: 'captcharesponse', type: 'struct<responsecode:string,solvetimestamp:string,failurereason:string>' }, { name: 'challengeresponse', type: 'struct<responsecode:string,solvetimestamp:string,failurereason:string>' }, { name: 'ja3fingerprint', type: 'string' }, { name: 'ja4fingerprint', type: 'string' }, { name: 'oversizefields', type: 'string' }, { name: 'requestbodysize', type: 'int' }, { name: 'requestbodysizeinspectedbywaf', type: 'int' }, ], location: s3BaseLocation, inputFormat: 'org.apache.hadoop.mapred.TextInputFormat', outputFormat: 'org.apache.hadoop.hive.ql.io.HiveIgnoreKeyTextOutputFormat', serdeInfo: { serializationLibrary: 'org.openx.data.jsonserde.JsonSerDe', }, }, }, }); } } exports.AthenaTableForWaf = AthenaTableForWaf; _a = JSII_RTTI_SYMBOL_1; AthenaTableForWaf[_a] = { fqn: "cdk-athena-log.AthenaTableForWaf", version: "0.0.14" }; //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"athena-table-for-waf.js","sourceRoot":"","sources":["../src/athena-table-for-waf.ts"],"names":[],"mappings":";;;;;AAAA,6CAAsD;AACtD,2CAAuC;AAyBvC;;;GAGG;AACH,MAAa,iBAAkB,SAAQ,sBAAS;IAG9C,YAAY,KAAgB,EAAE,EAAU,EAAE,KAA6B;QACrE,KAAK,CAAC,KAAK,EAAE,EAAE,CAAC,CAAC;QAEjB,MAAM,EACJ,aAAa,EACb,YAAY,EACZ,SAAS,EACT,QAAQ,EACR,UAAU,EACV,mBAAmB,EAAE,eAAe;QACpC,SAAS,GACV,GAAG,KAAK,CAAC;QAEV,MAAM,KAAK,GAAG,mBAAK,CAAC,EAAE,CAAC,IAAI,CAAC,CAAC;QAC7B,MAAM,OAAO,GAAG,KAAK,CAAC,OAAO,CAAC;QAE9B,MAAM,SAAS,GAAG,QAAQ,KAAK,YAAY,CAAC,CAAC,CAAC,YAAY,CAAC,CAAC,CAAC,KAAK,CAAC,MAAM,CAAC;QAE1E,MAAM,QAAQ,GAAG,SAAS,CAAC,CAAC,CAAC,GAAG,SAAS,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC;QAClD,MAAM,cAAc,GAAG,QAAQ,aAAa,IAAI,QAAQ,WAAW,OAAO,YAAY,SAAS,IAAI,UAAU,GAAG,CAAC;QACjH,MAAM,kBAAkB,GAAG,GAAG,cAAc,cAAc,CAAC;QAE3D,IAAI,CAAC,KAAK,GAAG,IAAI,sBAAI,CAAC,QAAQ,CAAC,IAAI,EAAE,SAAS,EAAE;YAC9C,SAAS,EAAE,OAAO;YAClB,YAAY,EAAE,YAAY;YAC1B,UAAU,EAAE;gBACV,IAAI,EAAE,SAAS;gBACf,SAAS,EAAE,gBAAgB;gBAC3B,UAAU,EAAE;oBACV,oBAAoB,EAAE,MAAM;oBAC5B,0BAA0B,EAAE,MAAM;oBAClC,4BAA4B,EAAE,kBAAkB;oBAChD,2BAA2B,EAAE,GAAG,mBAAmB,YAAY;oBAC/D,8BAA8B,EAAE,GAAG;oBACnC,mCAAmC,EAAE,SAAS;oBAC9C,2BAA2B,EAAE,kBAAkB;iBAChD;gBACD,aAAa,EAAE,CAAC,EAAE,IAAI,EAAE,UAAU,EAAE,IAAI,EAAE,QAAQ,EAAE,CAAC;gBACrD,iBAAiB,EAAE;oBACjB,OAAO,EAAE;wBACP,EAAE,IAAI,EAAE,WAAW,EAAE,IAAI,EAAE,QAAQ,EAAE;wBACrC,EAAE,IAAI,EAAE,eAAe,EAAE,IAAI,EAAE,KAAK,EAAE;wBACtC,EAAE,IAAI,EAAE,UAAU,EAAE,IAAI,EAAE,QAAQ,EAAE;wBACpC,EAAE,IAAI,EAAE,mBAAmB,EAAE,IAAI,EAAE,QAAQ,EAAE;wBAC7C,EAAE,IAAI,EAAE,qBAAqB,EAAE,IAAI,EAAE,QAAQ,EAAE;wBAC/C,EAAE,IAAI,EAAE,QAAQ,EAAE,IAAI,EAAE,QAAQ,EAAE;wBAClC,EAAE,IAAI,EAAE,6BAA6B,EAAE,IAAI,EAAE,uGAAuG,EAAE;wBACtJ,EAAE,IAAI,EAAE,gBAAgB,EAAE,IAAI,EAAE,QAAQ,EAAE;wBAC1C,EAAE,IAAI,EAAE,cAAc,EAAE,IAAI,EAAE,QAAQ,EAAE;wBACxC;4BACE,IAAI,EAAE,eAAe;4BACrB,IAAI,EAAE,mOAAmO;yBAC1O;wBACD,EAAE,IAAI,EAAE,mBAAmB,EAAE,IAAI,EAAE,0EAA0E,EAAE;wBAC/G,EAAE,IAAI,EAAE,6BAA6B,EAAE,IAAI,EAAE,oEAAoE,EAAE;wBACnH,EAAE,IAAI,EAAE,wBAAwB,EAAE,IAAI,EAAE,yCAAyC,EAAE;wBACnF,EAAE,IAAI,EAAE,kBAAkB,EAAE,IAAI,EAAE,QAAQ,EAAE;wBAC5C;4BACE,IAAI,EAAE,aAAa;4BACnB,IAAI,EAAE,+MAA+M;yBACtN;wBACD,EAAE,IAAI,EAAE,QAAQ,EAAE,IAAI,EAAE,4BAA4B,EAAE;wBACtD,EAAE,IAAI,EAAE,iBAAiB,EAAE,IAAI,EAAE,wEAAwE,EAAE;wBAC3G,EAAE,IAAI,EAAE,mBAAmB,EAAE,IAAI,EAAE,wEAAwE,EAAE;wBAC7G,EAAE,IAAI,EAAE,gBAAgB,EAAE,IAAI,EAAE,QAAQ,EAAE;wBAC1C,EAAE,IAAI,EAAE,gBAAgB,EAAE,IAAI,EAAE,QAAQ,EAAE;wBAC1C,EAAE,IAAI,EAAE,gBAAgB,EAAE,IAAI,EAAE,QAAQ,EAAE;wBAC1C,EAAE,IAAI,EAAE,iBAAiB,EAAE,IAAI,EAAE,KAAK,EAAE;wBACxC,EAAE,IAAI,EAAE,+BAA+B,EAAE,IAAI,EAAE,KAAK,EAAE;qBACvD;oBACD,QAAQ,EAAE,cAAc;oBACxB,WAAW,EAAE,0CAA0C;oBACvD,YAAY,EAAE,4DAA4D;oBAC1E,SAAS,EAAE;wBACT,oBAAoB,EAAE,oCAAoC;qBAC3D;iBACF;aACF;SACF,CAAC,CAAC;IACL,CAAC;;AAlFH,8CAmFC","sourcesContent":["import { aws_glue as glue, Stack } from 'aws-cdk-lib';\nimport { Construct } from 'constructs';\n\nexport interface AthenaTableForWafProps {\n  /** S3に保存されたWAFログのバケット名 */\n  readonly logBucketName: string;\n  /** Glueデータベース名 */\n  readonly databaseName: string;\n  /** 作成するGlueテーブル名 */\n  readonly tableName: string;\n  /**\n   * WAFのスコープを指定します。\n   * 'REGIONAL' または 'CLOUDFRONT'\n   */\n  readonly wafScope: 'REGIONAL' | 'CLOUDFRONT';\n  /** クエリ対象のWeb ACL名 */\n  readonly webAclName: string;\n  /**\n   * ★ パーティション射影の開始日 (yyyy/MM/dd形式)\n   * @example '2025/08/11'\n   */\n  readonly projectionStartDate: string;\n  /** バケット内のオプションのプレフィックス */\n  readonly logPrefix?: string;\n}\n\n/**\n * S3に保存されたAWS WAFログをクエリするためのGlueテーブルを作成します。\n * このコンストラクトは、公式ドキュメントのdate型パーティション射影に準拠します。\n */\nexport class AthenaTableForWaf extends Construct {\n  public readonly table: glue.CfnTable;\n\n  constructor(scope: Construct, id: string, props: AthenaTableForWafProps) {\n    super(scope, id);\n\n    const {\n      logBucketName,\n      databaseName,\n      tableName,\n      wafScope,\n      webAclName,\n      projectionStartDate, // ★ プロパティを受け取る\n      logPrefix,\n    } = props;\n\n    const stack = Stack.of(this);\n    const account = stack.account;\n\n    const logRegion = wafScope === 'CLOUDFRONT' ? 'cloudfront' : stack.region;\n\n    const s3Prefix = logPrefix ? `${logPrefix}/` : '';\n    const s3BaseLocation = `s3://${logBucketName}/${s3Prefix}AWSLogs/${account}/WAFLogs/${logRegion}/${webAclName}/`;\n    const s3LocationTemplate = `${s3BaseLocation}\\${log_time}`;\n\n    this.table = new glue.CfnTable(this, 'Default', {\n      catalogId: account,\n      databaseName: databaseName,\n      tableInput: {\n        name: tableName,\n        tableType: 'EXTERNAL_TABLE',\n        parameters: {\n          'projection.enabled': 'true',\n          'projection.log_time.type': 'date',\n          'projection.log_time.format': 'yyyy/MM/dd/HH/mm',\n          'projection.log_time.range': `${projectionStartDate}/00/00,NOW`,\n          'projection.log_time.interval': '1',\n          'projection.log_time.interval.unit': 'MINUTES',\n          'storage.location.template': s3LocationTemplate,\n        },\n        partitionKeys: [{ name: 'log_time', type: 'string' }],\n        storageDescriptor: {\n          columns: [\n            { name: 'timestamp', type: 'bigint' },\n            { name: 'formatversion', type: 'int' },\n            { name: 'webaclid', type: 'string' },\n            { name: 'terminatingruleid', type: 'string' },\n            { name: 'terminatingruletype', type: 'string' },\n            { name: 'action', type: 'string' },\n            { name: 'terminatingrulematchdetails', type: 'array<struct<conditiontype:string,sensitivitylevel:string,location:string,matcheddata:array<string>>>' },\n            { name: 'httpsourcename', type: 'string' },\n            { name: 'httpsourceid', type: 'string' },\n            {\n              name: 'rulegrouplist',\n              type: 'array<struct<rulegroupid:string,terminatingrule:struct<ruleid:string,action:string,rulematchdetails:string>,nonterminatingmatchingrules:array<struct<ruleid:string,action:string,rulematchdetails:string>>,excludedrules:string>>',\n            },\n            { name: 'ratebasedrulelist', type: 'array<struct<ratebasedruleid:string,limitkey:string,maxrateallowed:int>>' },\n            { name: 'nonterminatingmatchingrules', type: 'array<struct<ruleid:string,action:string,rulematchdetails:string>>' },\n            { name: 'requestheadersinserted', type: 'array<struct<name:string,value:string>>' },\n            { name: 'responsecodesent', type: 'string' },\n            {\n              name: 'httprequest',\n              type: 'struct<clientip:string,country:string,headers:array<struct<name:string,value:string>>,uri:string,args:string,httpversion:string,httpmethod:string,requestid:string,fragment:string,scheme:string,host:string>',\n            },\n            { name: 'labels', type: 'array<struct<name:string>>' },\n            { name: 'captcharesponse', type: 'struct<responsecode:string,solvetimestamp:string,failurereason:string>' },\n            { name: 'challengeresponse', type: 'struct<responsecode:string,solvetimestamp:string,failurereason:string>' },\n            { name: 'ja3fingerprint', type: 'string' },\n            { name: 'ja4fingerprint', type: 'string' },\n            { name: 'oversizefields', type: 'string' },\n            { name: 'requestbodysize', type: 'int' },\n            { name: 'requestbodysizeinspectedbywaf', type: 'int' },\n          ],\n          location: s3BaseLocation,\n          inputFormat: 'org.apache.hadoop.mapred.TextInputFormat',\n          outputFormat: 'org.apache.hadoop.hive.ql.io.HiveIgnoreKeyTextOutputFormat',\n          serdeInfo: {\n            serializationLibrary: 'org.openx.data.jsonserde.JsonSerDe',\n          },\n        },\n      },\n    });\n  }\n}\n"]}