Claude-code-plugins-plus-skills appfolio-upgrade-migration
install
source · Clone the upstream repo
git clone https://github.com/jeremylongshore/claude-code-plugins-plus-skills
Claude Code · Install into ~/.claude/skills/
T=$(mktemp -d) && git clone --depth=1 https://github.com/jeremylongshore/claude-code-plugins-plus-skills "$T" && mkdir -p ~/.claude/skills && cp -r "$T/plugins/saas-packs/appfolio-pack/skills/appfolio-upgrade-migration" ~/.claude/skills/jeremylongshore-claude-code-plugins-plus-skills-appfolio-upgrade-migration && rm -rf "$T"
manifest:
plugins/saas-packs/appfolio-pack/skills/appfolio-upgrade-migration/SKILL.mdsource content
AppFolio Upgrade & Migration
Overview
AppFolio property management integrations depend on versioned REST API endpoints that evolve with the platform. Upgrades can rename fields on property and tenant objects, change pagination models, deprecate work-order endpoints, and alter the basic-auth flow. This skill detects your current API version, maps deprecated response shapes to replacements, and rolls back automatically if the new version fails.
Prerequisites
- Current API version prefix documented (e.g.,
)/api/v1/ - Access to the AppFolio API changelog and release notes
- Staging environment with test property and tenant data
- Client ID and secret stored in environment variables
- Existing integration test suite that covers core endpoints
Instructions
- Run version detection to compare your active API version against the latest.
- Review the AppFolio changelog for breaking changes between the two versions.
- Apply schema migration transforms to property and tenant response objects.
- Update endpoint URLs from the old version prefix to the new one.
- Switch pagination from offset to cursor-based if required by the new version.
- Run the smoke test suite against staging to verify all endpoints respond.
- Deploy to production with the rollback strategy enabled.
- Monitor error logs for 410/401 responses indicating missed migration steps.
Output
After a successful migration the skill produces:
- A
object confirming current, latest, and deprecated versionsVersionInfo - Transformed property and tenant objects matching the new schema
- Smoke test results for properties, tenants, work orders, and accounting endpoints
- Rollback log entries if any endpoint fell back to the previous version
Version Detection
interface VersionInfo { current: string; latest: string; deprecated: string[]; } async function detectApiVersion(baseUrl: string, headers: Record<string, string>): Promise<VersionInfo> { const res = await fetch(`${baseUrl}/api/status`, { headers }); const body = await res.json(); const current = res.headers.get("X-AppFolio-Api-Version") ?? body.api_version; const deprecated: string[] = body.deprecated_versions ?? []; if (deprecated.includes(current)) { console.warn(`Version ${current} is deprecated. Migrate to ${body.latest_version}.`); } return { current, latest: body.latest_version, deprecated }; }
Schema Migration
interface LegacyProperty { address_line1: string; unit_count: number; mgr_id: string; } interface CurrentProperty { street_address: string; total_units: number; manager_id: string; } function migrateProperty(old: LegacyProperty): CurrentProperty { return { street_address: old.address_line1, total_units: old.unit_count, manager_id: old.mgr_id }; } interface LegacyTenant { lease_end: string; balance_due: number; } interface CurrentTenant { lease_expiry_date: string; outstanding_balance: number; } function migrateTenant(old: LegacyTenant): CurrentTenant { return { lease_expiry_date: old.lease_end, outstanding_balance: old.balance_due }; }
Rollback Strategy
async function versionAwareRequest( baseUrl: string, path: string, headers: Record<string, string>, targetVersion: string, fallbackVersion: string ): Promise<any> { const res = await fetch(`${baseUrl}/api/${targetVersion}${path}`, { headers }); if (res.status === 410 || res.status === 404) { console.warn(`${targetVersion} rejected; falling back to ${fallbackVersion}`); const fallback = await fetch(`${baseUrl}/api/${fallbackVersion}${path}`, { headers }); if (!fallback.ok) throw new Error(`Fallback failed: ${fallback.status}`); return fallback.json(); } if (!res.ok) throw new Error(`Request failed: ${res.status}`); return res.json(); }
Examples
// Detect version and migrate if needed const info = await detectApiVersion("https://acme.appfolio.com", authHeaders); if (info.deprecated.includes(info.current)) { const oldProps = await fetchLegacyProperties(); const migrated = oldProps.map(migrateProperty); await smokeTestEndpoints("https://acme.appfolio.com", authHeaders); }
Error Handling
| Migration Issue | Symptom | Fix |
|---|---|---|
| Deprecated version prefix | on every request | Update base URL to latest version prefix |
| Renamed property fields | values in property sync | Apply transform |
| Removed pagination offset | Empty result sets after page one | Switch to cursor-based pagination |
| Auth header rejected | after upgrade | Regenerate client secret, update env vars |
| Webhook envelope change | Event handler parse errors | Update payload parser for new envelope |
Resources
- AppFolio Stack APIs
- AppFolio Engineering Blog
- See
for post-migration CI validationappfolio-ci-integration