name: tuist-migration description: Integrates Tuist into an existing iOS project that uses SPM local packages. Use when migrating a project from a manually maintained .xcodeproj to a Tuist-generated project, adding Tuist as an orchestration layer on top of SPM. Covers 8 sequential phases — audit, base structure, helpers, generation, schemes, CI, cache, and validation. Includes automation scripts and AI prompt templates for each phase.
Tuist Migration
Integrate Tuist into an existing iOS project with SPM local packages, without modifying the modules' Package.swift files. Tuist acts as an orchestration layer that generates the .xcodeproj/.xcworkspace from declarative Swift manifests.
Fundamental Principle
Tuist does not touch modules, it only orchestrates the project that consumes them. The Package.swift files remain intact. If Tuist is removed later, the modules are still functional SPM packages.
Reference Project
Use the current project's Tuist configuration (Tuist/ProjectDescriptionHelpers/, Project.swift, Workspace.swift, Tuist.swift, Tuist/Package.swift) as the primary reference for API syntax and helper patterns. Never copy configurations, bundle IDs, modules, or dependencies — adapt patterns to the target project using audit data.
Migration Phases
8 sequential phases. Each has a script and/or prompt template in the bundled resources.
| # | Phase | Script | Prompt | Output |
|---|---|---|---|---|
| 1 | Audit | scripts/audit.sh |
references/prompts/01-audit.md |
AUDIT_REPORT.md |
| 2 | Base structure | — | references/prompts/02-base-structure.md |
Tuist.swift, Tuist/Package.swift, .gitignore |
| 3 | Helpers | — | references/prompts/03-helpers.md |
Tuist/ProjectDescriptionHelpers/*.swift |
| 4 | Generation | scripts/generate.sh |
references/prompts/04-generation.md |
Project.swift, Workspace.swift, generate.sh |
| 5 | Schemes | scripts/compare_schemes.sh |
references/prompts/05-schemes.md |
Verified AppScheme.swift |
| 6 | CI | — | references/prompts/06-ci.md |
Updated CI pipelines |
| 7 | Cache | scripts/extract_cache_data.sh, scripts/warm_cache.sh |
references/prompts/07-cache.md |
Optimized productTypes |
| 8 | Validation | scripts/compare_settings.sh, scripts/validate_migration.sh |
references/prompts/08-validation.md |
PASS/FAIL report |
Workflow
Read references/prompts/00-context.md for the full orchestration context. For each phase:
- Read the phase prompt — Load
references/prompts/0X-*.mdfor detailed instructions. - Run the script (if applicable) — Execute the phase script to collect data.
- Execute the phase — Generate/modify files following the prompt instructions.
- Present results — Summarize files created, decisions made, warnings found.
- Get feedback — Ask user before proceeding to next phase.
- Validate — From phase 4 onward, verify
tuist generatesucceeds and project builds.
Never skip phases or execute multiple at once without explicit user approval.
Scripts Reference
All scripts use set -euo pipefail and require mise for tool versioning. All scripts have placeholder values marked with ← Adjust comments — update them before running.
| Script | Phase | Purpose |
|---|---|---|
scripts/setup.sh |
Pre-migration | Install brew, mise, and tools from .mise.toml |
scripts/audit.sh |
1 | Collect all project data into /tmp/tuist-audit/ |
scripts/generate.sh |
4 | tuist install + tuist generate with optional --clean |
scripts/compare_schemes.sh |
5 | Compare schemes between original and Tuist projects |
scripts/extract_cache_data.sh |
7 | Extract dependency graph for cache optimization |
scripts/warm_cache.sh |
7 | Pre-compile external dependencies |
scripts/compare_settings.sh |
8 | Compare build settings between both projects |
scripts/validate_migration.sh |
8 | End-to-end validation with 7 checks and PASS/FAIL |
Tuist File Structure
Project root (new files)
├── .mise.toml ← Tool versions (tuist, swiftlint)
├── Project.swift ← Minimal: `let project = App.project`
├── Workspace.swift ← Root project + module paths + schemes
├── Tuist.swift ← Xcode/Swift version constraints
├── Tuist/Package.swift ← External deps + targetSettings per module
└── Tuist/ProjectDescriptionHelpers/
├── Config.swift ← App name, destinations, base settings
├── BuildConfiguration.swift ← Debug/Staging/Release configurations
├── Environment.swift ← Dev/Staging/Prod: API URLs, bundle IDs
├── Module.swift ← Struct wrapping SPM local package refs
├── Modules.swift ← Central registry of all modules
├── App.swift ← App target + UI tests target
├── AppScheme.swift ← Scheme factory per environment
└── BuildScripts.swift ← Run Script build phases (SwiftLint, etc.)
Key Rules
- All values from audit — Build settings, bundle IDs, signing, dependencies come from
AUDIT_REPORT.md, never from reference projects. -Tuistsuffix during migration — Name the projectAppName-Tuistso both projects coexist. Remove suffix after validation.targetSettingssynchronizes, not overrides — Ensures Xcode build settings match what eachPackage.swiftdeclares.- Modules with
nonisolateddefault — Need a separateSettingsDictionaryintargetSettings. productTypesfor cache — External dependencies declared as.frameworkget cached as pre-compiled binaries.- Incremental validation — From phase 4, run
tuist generateafter each change.
Audit to Helpers Mapping
| Tuist Helper | Built From (Audit Section) |
|---|---|
Config.swift |
Build settings common to all targets |
BuildConfiguration.swift |
Build configurations (Debug/Release/Staging) |
Environment.swift |
Schemes + environment-specific values |
Module.swift |
Module list + Package.swift files |
Modules.swift |
Complete module list from targets |
App.swift |
Info.plist + resources + build phases + signing |
AppScheme.swift |
Schemes (run config, test targets, coverage) |
BuildScripts.swift |
Run Script build phases |
Tuist/Package.swift |
swiftSettings per module + external deps |
Validation Criteria
Migration is valid when ALL pass:
tuist migration check-empty-settingsreports no orphaned settings- Build settings diff shows no functional differences
xcdiffshows no missing source files or resources- Dependency graph matches
- Full test suite passes (unit + snapshot + UI)
validate_migration.shexits with code 0
Common Failure Patterns
.intentdefinitionmust go in sources, not resources.xcstringscollides with.strings/.stringsdictglobs- ObjC categories in static frameworks need
-ObjCinOTHER_LDFLAGS - SPM resource bundles require
.process("Resources")andBundle.module - Types not found → source files accidentally excluded
- Undefined symbols → missing SDK frameworks or dependency products
- Launch crashes → incorrect bundle IDs, entitlements, or resources