E-1 完了サマリー
実施内容 # 変更内容 ファイル 1 OfficialChusankanField に 11 フィールド追加(17列化) models.py 2 中山間インポート: 17 列すべて読み込み対応 views.py 3 共済インポート: 面積カラム名不一致バグ修正 + a→m2 変換(×100) views.py 4 シリアライザに 11 フィールド追加 serializers.py 5 共済 PDF: A4 縦、表形式、@page 設定、ページ番号、中国語除去 kyosai_template.html 6 中山間 PDF: A4 横、表形式、@page 設定、ページ番号、中国語除去 chusankan_template.html 7 PDF 生成ロジック: フラットテーブル、null 安全、prefetch_related reports/views.py 8 既存データ再インポート(共済面積修正 + 中山間 17 列埋め) — 9 Playwright E2E テスト 11 件全 PASS verify-fixes.spec.ts 追加発見・修正したバグ 共済 ODS の 本地面積 (m2) カラム名にスペースが含まれ、インポート時に面積が全件 0 になっていた 面積の単位がアール(a)であることが判明。m2 への変換 (×100) を追加 PDF は http://localhost:3000/reports からダウンロードして確認できます。
This commit is contained in:
@@ -0,0 +1,78 @@
|
||||
# Generated by Django 5.0 on 2026-02-17 06:01
|
||||
|
||||
from django.db import migrations, models
|
||||
|
||||
|
||||
class Migration(migrations.Migration):
|
||||
|
||||
dependencies = [
|
||||
('fields', '0005_c2_kyosai_unique_together_c4_area_to_m2_integer'),
|
||||
]
|
||||
|
||||
operations = [
|
||||
migrations.AddField(
|
||||
model_name='officialchusankanfield',
|
||||
name='base_amount',
|
||||
field=models.IntegerField(blank=True, null=True, verbose_name='基本金額'),
|
||||
),
|
||||
migrations.AddField(
|
||||
model_name='officialchusankanfield',
|
||||
name='branch_num',
|
||||
field=models.CharField(blank=True, max_length=20, null=True, verbose_name='枝番'),
|
||||
),
|
||||
migrations.AddField(
|
||||
model_name='officialchusankanfield',
|
||||
name='chusankan_flag',
|
||||
field=models.CharField(blank=True, max_length=10, null=True, verbose_name='中山間フラグ'),
|
||||
),
|
||||
migrations.AddField(
|
||||
model_name='officialchusankanfield',
|
||||
name='land_type',
|
||||
field=models.CharField(blank=True, max_length=20, null=True, verbose_name='地目'),
|
||||
),
|
||||
migrations.AddField(
|
||||
model_name='officialchusankanfield',
|
||||
name='manager',
|
||||
field=models.CharField(blank=True, max_length=100, null=True, verbose_name='協定管理者'),
|
||||
),
|
||||
migrations.AddField(
|
||||
model_name='officialchusankanfield',
|
||||
name='original_crop',
|
||||
field=models.CharField(blank=True, max_length=100, null=True, verbose_name='作付け品目'),
|
||||
),
|
||||
migrations.AddField(
|
||||
model_name='officialchusankanfield',
|
||||
name='owner',
|
||||
field=models.CharField(blank=True, max_length=100, null=True, verbose_name='所有者'),
|
||||
),
|
||||
migrations.AddField(
|
||||
model_name='officialchusankanfield',
|
||||
name='planting_area',
|
||||
field=models.IntegerField(blank=True, null=True, verbose_name='植栽面積(m2)'),
|
||||
),
|
||||
migrations.AddField(
|
||||
model_name='officialchusankanfield',
|
||||
name='slope',
|
||||
field=models.CharField(blank=True, max_length=20, null=True, verbose_name='傾斜度'),
|
||||
),
|
||||
migrations.AddField(
|
||||
model_name='officialchusankanfield',
|
||||
name='smart_agri_addition',
|
||||
field=models.IntegerField(blank=True, null=True, verbose_name='スマート農業加算額'),
|
||||
),
|
||||
migrations.AddField(
|
||||
model_name='officialchusankanfield',
|
||||
name='steep_slope_addition',
|
||||
field=models.IntegerField(blank=True, null=True, verbose_name='超急傾斜加算額'),
|
||||
),
|
||||
migrations.AlterField(
|
||||
model_name='officialchusankanfield',
|
||||
name='area',
|
||||
field=models.IntegerField(default=0, verbose_name='農地面積(m2)'),
|
||||
),
|
||||
migrations.AlterField(
|
||||
model_name='officialchusankanfield',
|
||||
name='payment_amount',
|
||||
field=models.IntegerField(blank=True, null=True, verbose_name='交付金額'),
|
||||
),
|
||||
]
|
||||
@@ -20,11 +20,22 @@ class OfficialKyosaiField(models.Model):
|
||||
|
||||
class OfficialChusankanField(models.Model):
|
||||
c_id = models.CharField(max_length=20, unique=True, verbose_name="中山間ID")
|
||||
chusankan_flag = models.CharField(max_length=10, blank=True, null=True, verbose_name="中山間フラグ")
|
||||
oaza = models.CharField(max_length=100, verbose_name="大字")
|
||||
aza = models.CharField(max_length=100, verbose_name="字")
|
||||
chiban = models.CharField(max_length=50, verbose_name="地番")
|
||||
area = models.IntegerField(default=0, verbose_name="面積(m2)")
|
||||
payment_amount = models.IntegerField(blank=True, null=True, verbose_name="支払金額")
|
||||
branch_num = models.CharField(max_length=20, blank=True, null=True, verbose_name="枝番")
|
||||
land_type = models.CharField(max_length=20, blank=True, null=True, verbose_name="地目")
|
||||
area = models.IntegerField(default=0, verbose_name="農地面積(m2)")
|
||||
planting_area = models.IntegerField(blank=True, null=True, verbose_name="植栽面積(m2)")
|
||||
original_crop = models.CharField(max_length=100, blank=True, null=True, verbose_name="作付け品目")
|
||||
manager = models.CharField(max_length=100, blank=True, null=True, verbose_name="協定管理者")
|
||||
owner = models.CharField(max_length=100, blank=True, null=True, verbose_name="所有者")
|
||||
slope = models.CharField(max_length=20, blank=True, null=True, verbose_name="傾斜度")
|
||||
base_amount = models.IntegerField(blank=True, null=True, verbose_name="基本金額")
|
||||
steep_slope_addition = models.IntegerField(blank=True, null=True, verbose_name="超急傾斜加算額")
|
||||
smart_agri_addition = models.IntegerField(blank=True, null=True, verbose_name="スマート農業加算額")
|
||||
payment_amount = models.IntegerField(blank=True, null=True, verbose_name="交付金額")
|
||||
|
||||
class Meta:
|
||||
verbose_name = "中山間マスタ"
|
||||
|
||||
@@ -11,7 +11,10 @@ class OfficialKyosaiFieldSerializer(serializers.ModelSerializer):
|
||||
class OfficialChusankanFieldSerializer(serializers.ModelSerializer):
|
||||
class Meta:
|
||||
model = OfficialChusankanField
|
||||
fields = ['id', 'c_id', 'oaza', 'aza', 'chiban', 'area', 'payment_amount']
|
||||
fields = ['id', 'c_id', 'chusankan_flag', 'oaza', 'aza', 'chiban', 'branch_num',
|
||||
'land_type', 'area', 'planting_area', 'original_crop', 'manager', 'owner',
|
||||
'slope', 'base_amount', 'steep_slope_addition', 'smart_agri_addition',
|
||||
'payment_amount']
|
||||
|
||||
|
||||
class FieldSerializer(serializers.ModelSerializer):
|
||||
|
||||
@@ -32,21 +32,33 @@ def import_kyosai_master(request):
|
||||
try:
|
||||
df = pd.read_excel(ods_file, engine='odf')
|
||||
df.columns = df.columns.str.strip()
|
||||
|
||||
|
||||
# ODS カラム名 "本地面積 (m2)" のスペース有無に対応
|
||||
area_col = None
|
||||
for col in df.columns:
|
||||
if '本地面積' in col:
|
||||
area_col = col
|
||||
break
|
||||
|
||||
created_count = 0
|
||||
updated_count = 0
|
||||
|
||||
for _, row in df.iterrows():
|
||||
k_num = str(row.get('耕地番号', '')).strip() if pd.notna(row.get('耕地番号')) else ''
|
||||
s_num = str(row.get('分筆番号', '')).strip() if pd.notna(row.get('分筆番号')) else ''
|
||||
|
||||
|
||||
if not k_num:
|
||||
continue
|
||||
|
||||
area_val = 0
|
||||
if area_col and pd.notna(row.get(area_col)):
|
||||
# ODS の値はアール(a)単位。m2 に変換 (1a = 100m2)
|
||||
area_val = int(float(row.get(area_col)) * 100)
|
||||
|
||||
defaults = {
|
||||
'address': str(row.get('地名 地番', '')).strip() if pd.notna(row.get('地名 地番')) else '',
|
||||
'kanji_name': str(row.get('漢字地名', '')).strip() if pd.notna(row.get('漢字地名')) else '',
|
||||
'area': int(float(row.get('本地面積(m2)', 0))) if pd.notna(row.get('本地面積(m2)')) else 0,
|
||||
'area': area_val,
|
||||
}
|
||||
|
||||
obj, created = OfficialKyosaiField.objects.update_or_create(
|
||||
@@ -169,37 +181,50 @@ def import_chusankan_master(request):
|
||||
|
||||
ods_file = request.FILES['file']
|
||||
|
||||
def safe_str(val, default=''):
|
||||
return str(val).strip() if pd.notna(val) else default
|
||||
|
||||
def safe_int(val, default=None):
|
||||
if pd.isna(val):
|
||||
return default
|
||||
try:
|
||||
return int(float(val))
|
||||
except (ValueError, TypeError):
|
||||
return default
|
||||
|
||||
try:
|
||||
df = pd.read_excel(ods_file, engine='odf')
|
||||
df.columns = df.columns.str.strip()
|
||||
|
||||
|
||||
created_count = 0
|
||||
updated_count = 0
|
||||
|
||||
for _, row in df.iterrows():
|
||||
raw_id = row.get('ID')
|
||||
c_id = str(raw_id).strip() if pd.notna(raw_id) else ''
|
||||
|
||||
|
||||
if not c_id:
|
||||
continue
|
||||
if not any(char.isdigit() for char in c_id):
|
||||
continue
|
||||
|
||||
try:
|
||||
payment_amount_val = row.get('交付金額')
|
||||
if pd.notna(payment_amount_val):
|
||||
payment_amount = int(payment_amount_val)
|
||||
else:
|
||||
payment_amount = None
|
||||
except (ValueError, TypeError):
|
||||
payment_amount = None
|
||||
|
||||
defaults = {
|
||||
'oaza': str(row.get('大字', '')).strip() if pd.notna(row.get('大字')) else '',
|
||||
'aza': str(row.get('字', '')).strip() if pd.notna(row.get('字')) else '',
|
||||
'chiban': str(row.get('地番', '')).strip() if pd.notna(row.get('地番')) else '',
|
||||
'area': int(float(row.get('農地面積', 0))) if pd.notna(row.get('農地面積')) else 0,
|
||||
'payment_amount': payment_amount,
|
||||
'chusankan_flag': safe_str(row.get('中山間')) or None,
|
||||
'oaza': safe_str(row.get('大字')),
|
||||
'aza': safe_str(row.get('字')),
|
||||
'chiban': safe_str(row.get('地番')),
|
||||
'branch_num': safe_str(row.get('枝番')) or None,
|
||||
'land_type': safe_str(row.get('地目')) or None,
|
||||
'area': safe_int(row.get('農地面積'), 0),
|
||||
'planting_area': safe_int(row.get('植栽面積')),
|
||||
'original_crop': safe_str(row.get('作付け品目')) or None,
|
||||
'manager': safe_str(row.get('協定管理者')) or None,
|
||||
'owner': safe_str(row.get('所有者')) or None,
|
||||
'slope': safe_str(row.get('傾斜度')) or None,
|
||||
'base_amount': safe_int(row.get('基本金額')),
|
||||
'steep_slope_addition': safe_int(row.get('超急傾斜加算額')),
|
||||
'smart_agri_addition': safe_int(row.get('スマート農業加算額')),
|
||||
'payment_amount': safe_int(row.get('交付金額')),
|
||||
}
|
||||
|
||||
obj, created = OfficialChusankanField.objects.update_or_create(
|
||||
|
||||
Reference in New Issue
Block a user