Use Existing Remote State
Harness IaCM lets you reuse existing remote state backends, such as AWS S3, Google Cloud Storage (GCS), or Azure Blob Storage, without migrating to Harness-managed storage. Just point your backends.tf file to your existing backend, and Harness will use it directly with OpenTofu.
This approach is ideal if you're onboarding to IaCM, already use remote backends, or need compatibility with other systems or CI pipelines.
Prerequisites
- Your repository contains a valid backends.tffile.
- Your existing remote state file (e.g., .tfstate) is accessible and versioned.
- Harness has read/write access to the remote backend. Go to Set environment variables for more information.
- You use the Plan and Apply steps in your pipeline, not custom script steps.
- Backend authentication credentials (e.g. for GCS, S3, Azure) are configured via environment variables in the Harness Workspace.
Example backends
- AWS S3
- GCP Cloud Storage
- Azure Blob Storage
terraform {
  backend "s3" {
    bucket         = "my-existing-tfstate-bucket"
    key            = "envs/dev/terraform.tfstate"
    region         = "us-east-1"
    dynamodb_table = "terraform-locks"
    encrypt        = true
  }
}
Ensure your Harness AWS connector or environment variables grant permission to access the bucket and DynamoDB table. OpenTofu S3 Backend Docs
terraform {
  backend "gcs" {
    bucket = "my-tfstate-bucket"
    prefix = "envs/dev"
  }
}
Your GCP connector or Workspace-level environment variables must include the necessary IAM roles (e.g.
roles/storage.objectAdmin). OpenTofu GCS Backend Docs
terraform {
  backend "azurerm" {
    resource_group_name  = "tf-state"
    storage_account_name = "tfstateaccount"
    container_name       = "tfstate"
    key                  = "terraform.tfstate"
  }
}
Configure access to Azure backend using environment variables like
ARM_CLIENT_ID,ARM_CLIENT_SECRET, and others. OpenTofu AzureRM Backend Docs
Pipeline Configuration & Testing
To provision your workspace with an existing remote backend, configure your pipeline using OpenTofu steps as shown below:
steps:
  - step:
      name: Init
      type: IACM_TOFU_INIT
      identifier: Init
  - step:
      name: Plan
      type: IACM_TOFU_PLAN
      identifier: Plan
  - step:
      name: Apply
      type: IACM_TOFU_APPLY
      identifier: Apply
If your workspace requires OpenTofu/Terraform variables or environment variables, add them in your workspace settings.
If no variables are specified, Harness uses any defaults defined in the source code (e.g. variables.tf files in your repo). Go to Declaring variables to see how to define variables in your source code.
To test your setup:
Run your pipeline to execute the init, plan, and apply steps (and any approval plugins you've configured). See IaCM Setup pipeline for more details.
- Check the logs during the approval or apply step to verify successful initialization of the remote backend.
- Ensure your backends.tffile is present in the repository connected to your workspace.
State Locking Considerations
Each remote backend implements its own locking mechanism:
- S3 uses DynamoDB for locking.
- GCS relies on object metadata.
- Azure Blob uses blob leases.
OpenTofu handles lock acquisition and release during pipeline execution. There is no additional locking layer in IaCM—locks are managed entirely by OpenTofu based on the backend settings.
Troubleshooting & Best Practices
| Issue or Question | Recommendation | 
|---|---|
| Error acquiring the state lock. | Ensure no other process (for example, a local OpenTofu/Terraform CLI run) is holding a lock. Lock retries, if any, are handled automatically by OpenTofu/Terraform. | 
| Plan step fails with missing or unexpected state. | Confirm that the keyorprefixin yourbackends.tffile matches the correct path in your remote backend. If you recently migrated from Terraform Cloud, verify that nocloud {}block remains in your configuration. Harness uses direct backend integrations instead of cloud-based blocks. | 
| Local CLI commands behave differently than Terraform Cloud. | Harness IaCM runs plans and applies remotely through pipelines and the IaCM CLI, not via cloud {}blocks. Use the Harness CLI or pipeline executions to perform remote plan and apply operations. A dedicated CLI command for direct state inspection is under development. | 
| Pipeline fails after local testing with a temporary backend. | If you created a local backend.tffor inspection, add it to.gitignoreor remove it before running your pipeline. A committed local backend configuration overrides the workspace-managed backend during remote execution. | 
| Permission denied or backend authentication errors. | Check that your Harness connector or environment variables grant full read/write access to the backend bucket, table, or storage account. Review your connector credentials and environment variables in Workspace settings. | 
| Concurrent operations on the same state file. | Each backend handles locking independently (for example, DynamoDB for S3, object metadata for GCS, leases for Azure Blob). Wait for existing locks to clear before retrying; Harness does not add an additional locking layer. | 
Best Practices
- Use versioned buckets or lock tables for safe collaboration.
- Keep backends.tfin version control, but avoid hardcoding secrets.
- Define backend credentials via environment variables in the Harness UI or Workspace settings.
- If migrating from Terraform Cloud or OpenTofu Cloud, remove any cloud {}blocks and rely on backend definitions instead.
- Validate that only one process accesses a given state at a time to prevent corruption.
- Run speculative plans only from pipelines to ensure consistent remote state access.
- Use workspace variables or pipeline inputs for backend paths when multiple environments share the same backend configuration.
Harness Workspaces provide the same remote execution context as Terraform Cloud workspaces — just with full control over backend configuration and pipeline orchestration.