Система тегирования ресурсов и распределение затрат
Без тегирования невозможно ответить на вопрос «сколько стоит этот продукт?» или «почему счёт вырос?». Resource tagging — основа FinOps: возможность разбить облачные расходы по продуктам, командам, окружениям и клиентам.
Стратегия тегирования
Стандартный набор обязательных тегов:
| Тег | Значения | Описание |
|---|---|---|
Environment |
production, staging, dev | Окружение |
Team |
backend, frontend, data, devops | Владелец ресурса |
Product |
api, dashboard, billing, data-pipeline | Продуктовый компонент |
CostCenter |
eng-001, data-002 | Центр затрат для бухгалтерии |
ManagedBy |
terraform, manual, helm | Способ управления |
Дополнительные теги для конкретных случаев: Customer (для SaaS с per-customer billing), Project (для проектной работы), ExpiresAt (для временных ресурсов).
Enforcement через AWS Config
AWS Config Rule для проверки наличия обязательных тегов:
resource "aws_config_config_rule" "required_tags" {
name = "required-tags"
source {
owner = "AWS"
source_identifier = "REQUIRED_TAGS"
}
input_parameters = jsonencode({
tag1Key = "Environment"
tag1Value = "production,staging,dev,test"
tag2Key = "Team"
tag3Key = "Product"
tag4Key = "ManagedBy"
})
scope {
compliance_resource_types = [
"AWS::EC2::Instance",
"AWS::RDS::DBInstance",
"AWS::ElasticLoadBalancingV2::LoadBalancer",
"AWS::S3::Bucket",
"AWS::Lambda::Function"
]
}
}
# Auto-remediation: Lambda добавляет дефолтные теги при обнаружении нарушения
resource "aws_config_remediation_configuration" "tag_remediation" {
config_rule_name = aws_config_config_rule.required_tags.name
target_type = "SSM_DOCUMENT"
target_id = "AWS-SetRequiredTags"
automatic = false # manual approval перед применением
parameter {
name = "RequiredTags"
static_value = "Environment=unknown,Team=unknown"
}
}
Terraform: тегирование на уровне провайдера
# provider.tf — default_tags применяются ко всем ресурсам
provider "aws" {
region = "eu-central-1"
default_tags {
tags = {
ManagedBy = "terraform"
Repository = "github.com/company/infrastructure"
Environment = var.environment
Team = var.team
}
}
}
# Специфичные теги добавляются на уровне ресурса
resource "aws_instance" "api_server" {
ami = data.aws_ami.ubuntu.id
instance_type = "t3.medium"
tags = {
Name = "api-server-${var.environment}"
Product = "api"
# Environment и Team наследуются из default_tags
}
}
Cost Allocation Tags и AWS Cost Categories
После создания тегов нужно включить их в Cost Explorer:
import boto3
ce = boto3.client('ce', region_name='us-east-1')
# Активация тегов для cost allocation (занимает до 24 часов)
ce.activate_tags(
Tags=['Environment', 'Team', 'Product', 'CostCenter']
)
# Создание Cost Category для группировки по командам
ce.create_cost_category_definition(
Name='Team-Costs',
RuleVersion='CostCategoryExpression.v1',
Rules=[
{
'Value': 'Backend Team',
'Rule': {
'Tags': {
'Key': 'Team',
'Values': ['backend', 'api']
}
}
},
{
'Value': 'Data Team',
'Rule': {
'Tags': {
'Key': 'Team',
'Values': ['data', 'analytics', 'ml']
}
}
}
],
DefaultValue='Unallocated'
)
Скрипт аудита нетегированных ресурсов
import boto3
from collections import defaultdict
REQUIRED_TAGS = {'Environment', 'Team', 'Product'}
def audit_untagged_resources(region='eu-central-1'):
session = boto3.Session(region_name=region)
untagged = defaultdict(list)
# EC2 Instances
ec2 = session.client('ec2')
instances = ec2.describe_instances(
Filters=[{'Name': 'instance-state-name', 'Values': ['running', 'stopped']}]
)
for reservation in instances['Reservations']:
for inst in reservation['Instances']:
tags = {t['Key']: t['Value'] for t in inst.get('Tags', [])}
missing = REQUIRED_TAGS - set(tags.keys())
if missing:
untagged['EC2'].append({
'id': inst['InstanceId'],
'missing_tags': list(missing),
'name': tags.get('Name', 'unnamed')
})
# RDS
rds = session.client('rds')
dbs = rds.describe_db_instances()
for db in dbs['DBInstances']:
arn = db['DBInstanceArn']
tags_resp = rds.list_tags_for_resource(ResourceName=arn)
tags = {t['Key']: t['Value'] for t in tags_resp['TagList']}
missing = REQUIRED_TAGS - set(tags.keys())
if missing:
untagged['RDS'].append({
'id': db['DBInstanceIdentifier'],
'missing_tags': list(missing)
})
return untagged
if __name__ == '__main__':
result = audit_untagged_resources()
total = sum(len(v) for v in result.values())
print(f"\nUntagged resources: {total}")
for service, resources in result.items():
print(f"\n{service}: {len(resources)} resources")
for r in resources[:5]: # показываем первые 5
print(f" {r['id']}: missing {r['missing_tags']}")
Showback и Chargeback отчёты
def generate_team_cost_report(month: str):
"""month: '2025-11'"""
ce = boto3.client('ce', region_name='us-east-1')
start = f"{month}-01"
# last day of month
year, mon = map(int, month.split('-'))
import calendar
last_day = calendar.monthrange(year, mon)[1]
end = f"{month}-{last_day:02d}"
response = ce.get_cost_and_usage(
TimePeriod={'Start': start, 'End': end},
Granularity='MONTHLY',
Metrics=['UnblendedCost'],
GroupBy=[{'Type': 'TAG', 'Key': 'Team'}]
)
report = {}
for group in response['ResultsByTime'][0]['Groups']:
team = group['Keys'][0].replace('Team$', '') or 'Untagged'
cost = float(group['Metrics']['UnblendedCost']['Amount'])
report[team] = round(cost, 2)
return dict(sorted(report.items(), key=lambda x: x[1], reverse=True))
Сроки внедрения
- Разработка tagging strategy + документация — 1 день
- Terraform default_tags для всех модулей — 2-3 дня
- AWS Config Rules для enforcement — 1 день
- Аудит + тегирование существующих ресурсов — 2-5 дней (зависит от масштаба)
- Cost Categories + отчёты — 1-2 дня







