cdk-athena-log
Version:
A CDK construct to create an Athena table for querying ALB logs.
85 lines • 15.9 kB
JavaScript
"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":"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","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"]}