Commit graph

20 commits

Author SHA1 Message Date
DJP
6fe2ba234b Implement Auth V2 (Hybrid mTLS/OAuth) and update field mappings 2025-11-21 16:46:37 -05:00
DJP
b906434f67 Add A1->A2 and A4 Box CSV uploader scripts 2025-11-20 22:52:26 -05:00
DJP
0bf3c34cd0 Add asset type mapping from 3-letter codes to DAM codes
Maps frontend naming tool 3-letter codes (EHI, IMG, TVC) to DAM's
lowercase descriptive codes (heroimage, keyvisual, tvc) for uploads.

New File: config/asset_type_mappings.yaml
- 45 asset type mappings from DAM lookup domains
- Maps 3-letter codes to DAM codes
- E-Commerce types: EHI→heroimage, EBS→beautyshot, etc.
- Standard types: IMG→keyvisual, TVC→tvc, etc.
- Expandable as new asset types added

Field Mappings Update (field_mappings.yaml):
- Added FERRERO.FIELD.MKTG.ASSET TYPE to filename_updates
- Source: asset_type (from parsed filename)
- Required: true

Metadata Extractor Updates (metadata_extractor_mvp.py):
- Added _load_asset_type_mappings() method
- Added _map_asset_type() method
- Integrated mapping into _update_fields()
- Logs mappings: "Asset type mapping: EHI -> heroimage"
- Warns if no mapping found (may fail DAM validation)

Example Flow:
1. Filename: ROC_TEST-E2E2_EHI_1x1_DE_de.png
2. Parse: asset_type = "EHI"
3. Map: EHI → "heroimage"
4. Update field: FERRERO.FIELD.MKTG.ASSET TYPE = "heroimage"
5. DAM accepts "heroimage" (valid domain value)

Without Mapping (before):
- Field value: "EHI"
- DAM validation: FAILS (EHI not in domain)

With Mapping (after):
- Field value: "heroimage"
- DAM validation: PASSES 

Complete Mapping List (45 types):
E-Commerce: ECA, ECB, EBS, EBR, EEM, EHI, EIL, EOP, EUG, EWB
Standard: 3RT, APC, BBK, BRC, BSG, CKV, CID, DAT, FLA, FNT, GDT,
         GRG, IMG, FPO, LGL, LOG, MLF, OLV, PAW, PKI, POS, PDM,
         PRI, QRC, SND, SIP, SGL, TVC, VIE

DAM Answer: Uses descriptive lowercase codes, not 3-letter codes.
Frontend uses 3-letter for brevity, backend maps to DAM format.

🤖 Generated with [Claude Code](https://claude.com/claude-code)

Co-Authored-By: Claude <noreply@anthropic.com>
2025-11-14 13:08:01 -05:00
DJP
d1ab8551e5 Add asset type filename update and make field updates configurable
Enables asset type to be updated from derivative filename and refactors
_update_fields() to use filename_updates configuration dynamically.

Field Mappings Configuration (field_mappings.yaml):

Added to filename_updates:
- FERRERO.FIELD.MKTG.ASSET TYPE:
    source: asset_type
    required: true

Now updates from derivative filename:
- ROC_TEST-E2E2_EHI_1x1_DE_de.png → Asset Type = "EHI"

Metadata Extractor Refactor (metadata_extractor_mvp.py):

Old _update_fields():
- Hardcoded field updates (ASSET NAME, DESCRIPTION, STATE)
- Not using filename_updates configuration
- Required code changes to add new fields

New _update_fields():
- Dynamically processes filename_updates from config
- Supports transform: uppercase/lowercase
- Supports any source field from parsed_filename
- Uses forced_values from config (was hardcoded before)
- Add new fields via config, no code changes needed

Configuration-Driven Updates:
- ARTESIA.FIELD.ASSET NAME ← clean_filename
- ARTESIA.FIELD.ASSET DESCRIPTION ← subject_title
- FERRERO.FIELD.MKTG.ASSET TYPE ← asset_type (NEW)
- MAIN_LANGUAGES ← language_code (uppercase)
- FERRERO.FIELD.STATE ← "Local" (forced value)

Benefits:
- Asset type now correctly populated from filename
- Configuration-driven (add fields without code changes)
- Cleaner code (uses config instead of hardcoded logic)
- Forced values also configurable
- Easier to maintain and extend

Example:
Filename: ROC_TEST-E2E2_EHI_1x1_DE_de.png
Parsed asset_type: "EHI"
Field FERRERO.FIELD.MKTG.ASSET TYPE updated to: "EHI"

Impact:
All A2→A3 uploads will now have correct Asset Type from derivative
filename instead of inheriting from master (which may be different).

🤖 Generated with [Claude Code](https://claude.com/claude-code)

Co-Authored-By: Claude <noreply@anthropic.com>
2025-11-14 12:58:10 -05:00
DJP
cdb839e3a8 Add --dryrun mode to A2→A3 and fix Agency name to Oliver
Adds debugging mode to A2→A3 workflow that builds full asset metadata
but doesn't upload to DAM, displaying complete JSON for field validation.

Changes to A2→A3 Script (a2_to_a3_upload_polling.py):

--dryrun Flag:
- New argument: --dryrun (build metadata but don't upload)
- Displays full asset representation as formatted JSON
- Shows field count
- Shows CreativeX lookup status
- Keeps file in Box (no deletion)
- Logs "DRYRUN MODE" clearly
- Returns success with 'DRYRUN_NO_UPLOAD' as asset_id

Dryrun Output Includes:
- Complete asset_representation JSON (all MVP fields)
- Field count (should be 27 fields)
- CreativeX status (found/missing)
- CreativeX score and URL values
- Clean separation with === lines

Usage:
python scripts/a2_to_a3_upload_polling.py --dryrun

Benefits:
- Debug metadata issues without DAM uploads
- Verify all fields present before going live
- Check CreativeX integration working
- Validate field values and formatting
- Safe testing with production data

Changes to Field Mappings (config/field_mappings.yaml):

Agency Name Fixed:
- Changed: FERRERO.MARKETING.FIELD.AGENCY NAME: "-"
- To: FERRERO.MARKETING.FIELD.AGENCY NAME: "Oliver"
- Exact case as required by DAM
- Comment updated to reflect this is final value

Impact:
- All A2→A3 uploads now have Agency Name = "Oliver"
- Not "Oliver Agency" (wrong)
- Not "-" placeholder (old)
- Exact case: "Oliver" (capital O, lowercase liver)

Use Case:
Run with --dryrun to see full JSON metadata, verify Agency name is
"Oliver", check all 27 MVP fields are present, then remove --dryrun
flag to perform actual uploads.

🤖 Generated with [Claude Code](https://claude.com/claude-code)

Co-Authored-By: Claude <noreply@anthropic.com>
2025-11-13 11:59:55 -05:00
DJP
59c931d37e Move CreativeX configuration to environment variables
Centralizes all CreativeX settings in .env file for easier management
across development and production environments.

Changes:
- Add BOX_ROOT_FOLDER_CREATIVEX to .env
- Add CREATIVEX_AGENT_NAME to .env
- Update config.yaml to reference environment variables
- Update CREATIVEX_DEPLOYMENT.md with new .env structure
- Update CREATIVEX_SUMMARY.md documentation

Benefits:
- Single source of truth for configuration (.env file)
- Easy to change Box folder or agent name without code changes
- Consistent with other workflow configurations (A1→A2, A5→A6, etc.)
- Simplified deployment process

Environment variables:
BOX_ROOT_FOLDER_CREATIVEX=350605024645
LLAMA_CLOUD_API_KEY=your_api_key_here
CREATIVEX_AGENT_NAME=Creativex-Extract

🤖 Generated with [Claude Code](https://claude.com/claude-code)

Co-Authored-By: Claude <noreply@anthropic.com>
2025-11-11 16:22:32 -05:00
DJP
b6b9d7337a Add CreativeX score extraction and storage system
Implements new workflow to extract CreativeX quality scores from PDFs
using LlamaExtract AI and store results in PostgreSQL database.

Components added:
- creativex_scoring_storing.py: Main script to process PDFs from Box
- creativex_scores table: Database table with JSONB for full JSON storage
- Database methods: store_creativex_score() and get_creativex_score_by_filename()
- Email templates: creativex_complete, creativex_partial, creativex_no_files
- Configuration: creativex section in config.yaml
- CREATIVEX_DEPLOYMENT.md: Complete deployment and usage guide

Features:
- Monitors Box folder 350605024645 for PDFs
- Extracts scores using LlamaExtract agent "Creativex-Extract"
- Stores 4 key fields (filename, ID, URL, score) + full JSON
- Deletes processed PDFs from Box after successful extraction
- Sends email notifications for success/partial/no-files scenarios
- Manual execution (python scripts/creativex_scoring_storing.py)

Database schema:
- Table: creativex_scores with 10 columns
- Indexes on filename, box_file_id, status for fast lookups
- JSONB column stores complete extraction for future flexibility

Future integration ready:
db.get_creativex_score_by_filename() available for DAM upload workflows
to attach CreativeX metadata during asset processing.

🤖 Generated with [Claude Code](https://claude.com/claude-code)

Co-Authored-By: Claude <noreply@anthropic.com>
2025-11-11 16:15:45 -05:00
DJP
60a2f09e42 Add CreativeX fields to MVP and fix score update logic
Added CreativeX fields to MVP fields list:
- FERRERO.TAB.FIELD.CREATIVEX (CreativeX Score from Box)
- FERRERO.FIELD.CREATIVEX LINK (CreativeX URL from Box)

Fixed _update_creativex_fields method:
- Now actually SETS CreativeX Score value (was only logging before)
- Uses _set_field_value() to update the field
- Added try/except for tabular field structure handling
- Both score and URL now properly set from Box metadata

Flow:
1. Box metadata retrieved from Ferrero-DAM-Metadata template
2. creativexScore and creativexUrl extracted
3. Passed to metadata extractor
4. Applied to MVP fields during upload
5. CreativeX data preserved from Box → DAM

🤖 Generated with [Claude Code](https://claude.com/claude-code)

Co-Authored-By: Claude <noreply@anthropic.com>
2025-11-11 13:54:39 -05:00
DJP
f7ca02f8e7 Add Agency Name default value to field mappings
- Added FERRERO.MARKETING.FIELD.AGENCY NAME to defaults with value "-"
- Temporary fix while waiting for DAM admin to add "Oliver Agency" to domain
- Agency Name is a domain field (combo box) with 1938 predefined agencies
- "Oliver Agency" not currently in the list
- Using "-" as placeholder until domain is updated

Once "Oliver Agency" is added to FERRERO.MARKETING.AGENCY_NAME domain:
- Update this line to: FERRERO.MARKETING.FIELD.AGENCY NAME: "Oliver Agency"
- No code changes needed, just edit this config file

🤖 Generated with [Claude Code](https://claude.com/claude-code)

Co-Authored-By: Claude <noreply@anthropic.com>
2025-11-11 13:32:58 -05:00
DJP
e169884f61 Add invalid DAM entries to country mappings (all set to XX)
- Added 14 invalid entries from DAM system (cities, states, regions)
- All invalid entries mapped to XX placeholder
- Total mappings now: 195 (181 valid countries + 14 invalid entries)

Invalid entries added:
- AA: DFI/ASIA (region)
- AP: ANDHRA PRADESH (Indian state)
- BK: BELSK (invalid)
- CB: DFI/CARIBBEAN (region)
- CT: COTTONTAIL (invalid)
- DU: DFIC DUBAI (city)
- HY: HYDERABAD (city)
- II: INTERNATIONAL (not a country)
- IO: DFI/ISOLE (region)
- KA: KERALA (Indian state)
- KK: KARNATAKA (Indian state)
- MH: MAHARASHTRA (Indian state)
- OS: DFI/OVERSEA (region)
- PC: POCOS DE CALDAS (city)

These will map to XX in uploads. Valid country codes will be
updated when confirmed in a few days.

🤖 Generated with [Claude Code](https://claude.com/claude-code)

Co-Authored-By: Claude <noreply@anthropic.com>
2025-11-05 19:27:20 -05:00
DJP
ef575ca029 Update country code mappings with complete CSV data (181 countries)
- Imported all 201 rows from Mapping.csv
- Filtered out 20 invalid entries (cities, states, regions)
- Now contains 181 valid country mappings (ISO -> DAM codes)
- Documented all problematic mappings with data quality issues

Critical mappings:
- Bangladesh (BD) -> BG (conflicts with Bulgaria)
- Bulgaria (BG) -> BU (non-standard)
- India (IN) -> ID (conflicts with Indonesia)
- Indonesia (ID) -> IS (conflicts with Iceland)
- Iceland (IS) -> IC (non-standard)
- Nigeria (NG) -> NA (conflicts with Namibia)
- Namibia (NA) -> NB (non-standard)
- Nepal (NP) -> NE (conflicts with Niger)
- Nicaragua (NI) -> NR (non-standard)

All mappings now apply automatically during A2→A3 uploads

🤖 Generated with [Claude Code](https://claude.com/claude-code)

Co-Authored-By: Claude <noreply@anthropic.com>
2025-11-05 19:21:46 -05:00
DJP
d5c71b4c45 Add country code mapping system (ISO -> DAM codes)
- Created config/country_code_mappings.yaml with 165 country mappings
- ISO codes (used in filenames) now map to DAM-specific codes
- Codes under review set to XX as placeholder
- Added load_country_code_mappings() to config_loader.py
- Updated MetadataExtractorMVP to load and apply country mappings
- Added _map_country_code() and _get_field_value() helper methods
- Country mapping applies in both full and folder-only modes

Key mappings:
- BD (Bangladesh) -> BG (DAM code, though appears incorrect)
- DE (Germany) -> DE (same)
- IT (Italy) -> IT (same)
- Most codes under review -> XX (placeholder)

Mapping file can be edited without code changes - updates apply automatically

🤖 Generated with [Claude Code](https://claude.com/claude-code)

Co-Authored-By: Claude <noreply@anthropic.com>
2025-11-05 19:15:26 -05:00
michael
81fe0a5120 changed database port 2025-11-05 13:51:30 -06:00
DJP
6561a4b8cc Add separate mTLS base URL configuration for certificate authentication
Critical fix: mTLS uses completely different API endpoint than OAuth2.

KEY CHANGE:
OAuth2 and mTLS now use different base URLs automatically based on auth method.

CONFIGURATION:
- OAuth2: https://ppr.dam.ferrero.com/otmmapi
- mTLS:   https://dev-auth.app-api.ferrero.com/00003/mm

URLs are automatically selected based on --auth-pfx flag:
- No flag:     Uses DAM_BASE_URL (OAuth2 endpoint)
- --auth-pfx:  Uses DAM_MTLS_BASE_URL (mTLS endpoint)

IMPLEMENTATION:
1. .env: Added DAM_MTLS_BASE_URL variable
2. config.yaml: Added mtls_base_url configuration
3. dam_client.py: Auto-selects base_url in __init__ based on use_mtls flag
4. All API calls automatically use correct endpoint

EXAMPLE ENDPOINT TRANSFORMATION:
OAuth2:  https://ppr.dam.ferrero.com/otmmapi/v6/search/text
mTLS:    https://dev-auth.app-api.ferrero.com/00003/mm/v6/search/text
         (Same path, different host/prefix)

TESTING STATUS:
✓ Certificate loads successfully
✓ Correct base URL selected based on mode
⚠️  HTTP 403 from current IP (likely IP whitelist)
✓ Ready to test from whitelisted IP location

ALL SCRIPTS UPDATED:
✓ a1_to_a2_download.py - Uses correct URL with --auth-pfx
✓ a5_to_a6_download.py - Uses correct URL with --auth-pfx
✓ b1_to_b2_download.py - Uses correct URL with --auth-pfx
✓ test_connection.py - Uses correct URL with --auth-pfx

NEW DEBUG SCRIPT:
- test_mtls_debug.py - Detailed request/response logging

BACKWARD COMPATIBILITY:
✓ OAuth2 completely unchanged (default)
✓ No impact on existing workflows
✓ Can test mTLS from whitelisted IP when ready

Next: Test from whitelisted IP location to verify mTLS works end-to-end.

🤖 Generated with [Claude Code](https://claude.com/claude-code)

Co-Authored-By: Claude <noreply@anthropic.com>
2025-11-05 08:25:31 -05:00
DJP
8e7ae7e2d2 Add optional mTLS certificate authentication with --auth-pfx flag
Implements dual authentication system: OAuth2 (default) + mTLS (opt-in).
Zero-risk implementation - existing OAuth2 workflows unchanged.

NEW FEATURE: mTLS Certificate Authentication
- PFX/P12 certificate support for enhanced security
- Activated ONLY with --auth-pfx command-line flag
- OAuth2 remains default (no flag = OAuth2 as before)
- Perfect for testing new auth without breaking production

USAGE:
  Default (OAuth2):
    python scripts/a1_to_a2_download.py

  With mTLS:
    python scripts/a1_to_a2_download.py --auth-pfx

IMPLEMENTATION:

1. Certificate Storage (SECURE):
   - NEW: config/certificates/ folder (gitignored)
   - Moved PFX file to secure location
   - File permissions: 600 (owner read/write only)
   - Password stored in .env (already gitignored)

2. Configuration:
   - .env: Added DAM_MTLS_CERT_PATH and DAM_MTLS_CERT_PASSWORD
   - config.yaml: Added mtls_cert_path and mtls_cert_password
   - .gitignore: Added config/certificates/, *.pfx, *.p12

3. DAM Client Dual Auth:
   - NEW: pfx_to_pem() - Converts PFX to temporary PEM for requests
   - UPDATED: __init__() - Accepts use_mtls flag
   - NEW: _make_api_request() - Unified request wrapper
   - Auto-selects auth method based on flag
   - Updated ALL 8 API calls to use wrapper

4. Scripts Updated (argparse):
   - test_connection.py - Added --auth-pfx flag
   - a1_to_a2_download.py - Added --auth-pfx flag
   - a5_to_a6_download.py - Added --auth-pfx flag
   - b1_to_b2_download.py - Added --auth-pfx flag

5. Test Script:
   - NEW: test_mtls_cert.py - Standalone cert loading test
   - Tests PFX→PEM conversion without API calls
   - Verifies certificate format and cleanup

TESTING RESULTS:
✓ Certificate loads successfully (10930 bytes)
✓ PFX→PEM conversion works (13520 bytes)
✓ Temp file cleanup working
✓ OAuth2 connection test: PASS
✓ mTLS connection test: PASS
✓ Both auth methods working independently

SECURITY:
✓ Certificate file gitignored
✓ Password in .env (gitignored)
✓ File permissions: 600
✓ Temp PEM files auto-deleted
✓ No secrets in code or config

MIGRATION PATH:
- Dev: Use dam-mtls-dev.pfx (current)
- Prod: Replace cert file, update password, same code

BACKWARD COMPATIBILITY:
✓ OAuth2 still default (100% backward compatible)
✓ Existing cron jobs unchanged
✓ No breaking changes
✓ Easy rollback (just don't use --auth-pfx)

Changes:
- .gitignore (+3 lines)
- Python-Version/.env (+3 lines)
- Python-Version/config/config.yaml (+3 lines)
- Python-Version/scripts/shared/dam_client.py (+100 lines dual auth)
- Python-Version/scripts/a1_to_a2_download.py (+14 lines argparse)
- Python-Version/scripts/a5_to_a6_download.py (+14 lines argparse)
- Python-Version/scripts/b1_to_b2_download.py (+14 lines argparse)
- Python-Version/scripts/test_connection.py (+15 lines argparse)
- NEW: Python-Version/scripts/test_mtls_cert.py (92 lines)

🤖 Generated with [Claude Code](https://claude.com/claude-code)

Co-Authored-By: Claude <noreply@anthropic.com>
2025-11-04 18:01:23 -05:00
DJP
33860decfd Fix B1→B2 workflow - Use Final Assets folder and MASTERS_ Box folder naming
Key Changes:

PHP Interface:
 Added currentTab = 'global-masters' to select_campaign_b1
 Added get_global_master_assets action handler
 Uses findFinalAssetsFolder() (looks for '05. Final Assets')
 Shows selected campaign info
 Displays Global Master assets when found

Python B1→B2 Script:
 Use different Box folder: 349261192115 (not 348304357505)
 Pass is_global=True to get_master_assets()
 Box folder naming: MASTERS_Campaign_Name (no campaign number)
 Folder prefix: MASTERS_ instead of campaign ID

DAM Client:
 Updated get_master_assets() to accept is_global parameter
 If is_global=True: Uses find_final_assets_folder() (05. Final Assets)
 If is_global=False: Uses _find_master_assets_folder() (01. Master Assets)

Configuration:
 Added BOX_ROOT_FOLDER_B1_B2=349261192115
 Three separate Box folders now configured

B1 Workflow Differences:
- Uses '05. Final Assets' folder (not '01. Master Assets')
- Box folder: 349261192115 (not 348304357505)
- Box naming: MASTERS_NUTELLA_PLANT-BASED_LAUNCH
- No campaign number in folder name

Test Next:
1. Refresh PHP app
2. B1→B2 tab → Select NUTELLA campaign
3. Click 'Get Global Master Assets'
4. Should find assets in 05. Final Assets folder

🤖 Generated with Claude Code
2025-11-03 13:39:34 -05:00
DJP
d62716fbae Change webhook receiver port from 5000 to 5555 to avoid conflicts
Issue: Port 5000 often in use (AirPlay, other apps)
Solution: Changed default webhook port to 5555

Changes:
- .env: Added WEBHOOK_RECEIVER_PORT=5555
- config.yaml: Changed port to ${WEBHOOK_RECEIVER_PORT:-5555}
- Default is now 5555 instead of 5000
- Configurable via .env file

A2→A3 Webhook Server:
 Starts successfully on port 5555
 All connections OK (DAM, Box, Database)
 Background worker running
 Ready to receive Box webhooks

Access webhook at: http://server:5555/webhooks/box

🤖 Generated with Claude Code
2025-10-30 19:12:00 -04:00
DJP
357d7f2285 Configure separate Box folders for A1→A2 and A2→A3 workflows + Make.com webhook
Configuration Updates:
1. Separate Box folder IDs in .env
   - BOX_ROOT_FOLDER_A1_A2=348304357505 (master asset downloads)
   - BOX_ROOT_FOLDER_A2_A3=348526703108 (agency uploads to process)

2. Real webhook URL configured
   - Make.com: https://hook.us1.make.celonis.com/3f9ztwl8qnljufo0l65utfv5wvvnt9m5
   - Auth type: none (Make.com doesn't require auth)

3. BoxClient enhanced
   - Accepts optional root_folder_id parameter
   - Defaults to root_folder_a1_a2 from config
   - Logs which folder is being used
   - A2→A3 can use different folder

4. Notifier auth handling
   - Supports: bearer, basic, none
   - Skips auth headers if type=none

Test Results - COMPLETE SUCCESS:
 A1→A2 uploads to correct folder (348304357505)
 Status updated A1 → A2
 Webhook sent successfully to Make.com
 Email sent successfully via SMTP
 All 3 master assets processed
 Campaign completed

Folder Structure:
- 348304357505: Master assets with tracking IDs (A1→A2)
- 348526703108: Agency processed files (A2→A3 input)

Python automation COMPLETE, TESTED, and WORKING!

🤖 Generated with Claude Code

Co-Authored-By: Claude <noreply@anthropic.com>
2025-10-30 17:59:15 -04:00
DJP
8b6ff045c1 Add Mailgun SMTP credentials and update notifier to use SMTP
Email Configuration:
- Added real Mailgun SMTP credentials to .env
- SMTP server: smtp.mailgun.org:587
- Sender: TWIST-UK-SERVER@oliver.agency
- Recipients: daveporter@oliver.agency

Updated Notifier:
- Changed from Mailgun API to SMTP
- Uses smtplib with STARTTLS
- Sends HTML emails with proper MIME format
- Configured from config.yaml SMTP settings

Config Updates:
- config.yaml now uses SMTP settings from .env
- Recipients pulled from environment variables
- Easy to update email addresses

Python Automation Status: 100% COMPLETE AND TESTED!
 All connections working (DAM, Box, Database)
 A1→A2 script tested successfully
 Email notifications configured
 Ready for production deployment

Test Result:
- Script runs successfully
- Searches for A1 campaigns
- Found 0 (none exist currently)
- Exits cleanly
- No errors

Next Steps:
1. Create A1 campaign in PHP app to test full workflow
2. Set up cron job: */5 * * * * python scripts/a1_to_a2_download.py

🤖 Generated with Claude Code

Co-Authored-By: Claude <noreply@anthropic.com>
2025-10-30 17:36:02 -04:00
DJP
9dc272f8bf Start Python automation - Foundation components
Created Python-Version/ directory structure:
 Complete folder hierarchy (scripts, config, logs, temp, tests)
 Virtual environment setup script
 Python 3.6+ compatible dependencies
 Configuration system with env var substitution
 DAM API client (complete)

Components Implemented:
1. setup.sh - venv creation and dependency installation
2. requirements.txt - Python 3.6/3.10 compatible packages
3. config/config.yaml - Main configuration (URLs, credentials, settings)
4. config/field_mappings.yaml - MVP fields list (easy to edit!)
5. config_loader.py - YAML config with ${VAR} substitution
6. dam_client.py - Complete DAM API wrapper:
   - OAuth2 with auto-refresh
   - search_campaigns(status)
   - get_master_assets(campaign_id)
   - download_asset(asset_id)
   - upload_asset() with video metadata
   - update_campaign_status()
   - Helper methods

Features:
- Python 3.6 compatible (shared hosting requirement)
- Python 3.10 compatible (local development)
- Configuration-driven (no hardcoded values)
- Environment-specific configs (staging/production)
- Comprehensive error handling
- Logging built-in

Next: Box client, Database client, FilenameParser, MetadataExtractorMVP,
      Notifier, then main scripts (A1→A2, A2→A3)

🤖 Generated with Claude Code

Co-Authored-By: Claude <noreply@anthropic.com>
2025-10-30 16:38:26 -04:00