Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

A lifecycle flag to never refresh resources #15472

Open
gtmtech opened this issue Jul 4, 2017 · 10 comments
Open

A lifecycle flag to never refresh resources #15472

gtmtech opened this issue Jul 4, 2017 · 10 comments

Comments

@gtmtech
Copy link

gtmtech commented Jul 4, 2017

Terraform 0.9.8

Background: Terraform likes to fully manage the resources it creates, and goes somewhat to trying to reconcile their latest state by doing a refresh/fact-find before issuing a plan or applying a plan. This is great behaviour. Its also great that terraform can easily spin up an environment in its entirety, and this is very useful for creating dev, prod environments very easily.

However certain development workflows involve subsequent changes to these resources which terraform is not necessarily good at handling itself, maybe because its current api support isnt good enough, or maybe because some things are complex to manage.

An example of this (but not limited to the AWS provider) is the aws_rds_cluster resource. We regularly restore a snapshot (which creates a different aws rds cluster, as you can never restore over the top of an existing cluster in aws), and then update dns and/or apps to point to the new cluster for a seamless db upgrade. Then we delete the old database, but not the dns records. Putting aside the fact that terraform's aws_rds_cluster resource for doing this has bugs preventing a successful restore from a snapshot anyway, assuming it would work, this is very unwieldy in terraform, and perhaps even impossible currently.

Suppose then the initial aws_route53_record sensibly looks like this:

resource "aws_route53_record" "dbreader" {
    ...
    records = ["${aws_rds_cluster.foo.reader_endpoint}"]
    ...
    lifecycle = {
        ignore_changes = ["*"]
    }
}

The dns records managed by terraform point originally to aws_rds_cluster.foo.reader_endpoint - but now that database has been removed one of the following happens:

  1. terraform needs to create the aws_rds_cluster again for it to be able to supply the aws_route53_record resource with the record field, despite ignore_changes being present.

  2. if count=0 is set on the aws_rds_cluster (now deleted), then terraform will complain that aws_rds_cluster is no longer found wrt to the records field in aws_route53_record.

  3. if count=0 is set on both aws_rds_cluster and aws_route53_record, terraform will attempt to delete the record resource, resulting in downtime within the system

In short, ignore_changes ignores changes in the apply phase, but not during the validate phase.

I believe this workflow would be solved if it is possible to add a lifecycle flag which would refuse to refresh resources. That way terraform would always rely on the statefile and not need to make any changes at all, regardless of what had happened to the actual resources. A kind of refresh=false flag, which affected a resource, and all its dependencies.

I would welcome any comments - perhaps you can already do what I am after..

@apparentlymart
Copy link
Contributor

Hi @gtmtech! Thanks for writing up this proposal.

I just want to make sure I understand fully what you're suggesting here: in the example scenario you described, is your intent that you would set this "don't refresh" flag on the aws_rds_cluster.foo resource, so that it would remain available in the state even if the underlying physical resources is deleted/updated?


Although it's a bit of a tangent from what you requested here and a more AWS-specific question, I'm also curious about the "replace RDS instance via snapshot" problem itself. I did this in the past a couple times in on a more ad-hoc basis to restore from backups, rather than as a routine procedure, and was able to get it done with the following (rather awkward) workflow:

  • Restore the snapshot into a new RDS instance
  • terraform state rm the reference to the old one from the state (in practice I expect I probably just did this with state surgery since I think at the time I did this the state rm command didn't exist yet)
  • terraform import aws_db_instance.foo bar to import the new RDS instance into the place of the old one
  • Manually destroy the old one after verifying everything was looking good

This is of course not a reasonable workflow for routine work, but I was wondering if this is something you'd tried and found it doesn't work anymore for some reason or if you'd just rejected it because of how awkward it is. If the above has become broken somehow, it'd be good to track that in an AWS provider bug. In the long run, it'd be nice to have a way for Terraform itself to manage this, but indeed it's rather awkward right now due to the mismatch of how Terraform sees it (replacing an existing instance) vs. how RDS sees it (making a new instance alongside the old).

@gtmtech
Copy link
Author

gtmtech commented Jul 11, 2017

Thanks for your writeup - I was able to follow the workflow above which is good, however as you say, restoring the db is quite a common workflow for us (every day) - so having to run 4 or so terraform commands each day per env is quite a hassle. But thanks for the guidance

@rafalcieslak
Copy link

Another use-case where such flag would be useful is discussed in this AWS provider issue. There are some resources that, in large amount, take a very long time to refresh. If a developer who prepares this resource knows that it shouldn't require to be ever refreshed (as it gets never updated by means other than terraform), they would be able to save a lot of time by marking such resource as "do not refresh". Any state inconsistencies introduced by the use of such flag would be at their own responsibility - though in practice it shouldn't be difficult to avoid them.

Note that in this case using -refresh=false CLI flag is insufficient, as all other resources may need to be regularly refreshed.

@dgcaron
Copy link

dgcaron commented Jul 18, 2021

this would also be helpfull for resources that er in the "dataplane" of azure. for instance, if you are creating secrets in a keyvault or folders in a storage account and afterwards you restrict network access to these accounts. terraform will stall on these folders and secrets as the SDK can't reach the folder (as it is being blocked). seeding a folder structure is something we do once at the initial deploy

@so-jelly
Copy link

so-jelly commented Feb 9, 2023

i am have @dgcaron 's exact problem. i have a read only account to perform terraform plan in CI and unless I give that account access to read secrets, I will get failure.

resource "google_secret_manager_secret_version" "secret" {
  lifecycle {
    ignore_changes = all
  }
  secret      = google_secret_manager_secret.secret.id
  secret_data = random_password.password.result
}

Plan as a read only user

Error: Error reading SecretVersion: googleapi: Error 403: Permission 'secretmanager.versions.access' denied for resource 'projects//secrets//versions/1' (or it may not exist).

@watsonjm
Copy link

This would also be helpful for the databricks_mount resource: https://registry.terraform.io/providers/databricks/databricks/latest/docs/resources/mount. I need to seed the mount but I don't care about it afterwards, but every Terraform plan brings the entire cluster up just so it can check the state, which takes a lot of time and unneeded compute.

@gregnuj
Copy link

gregnuj commented Jan 19, 2024

I have an issue where I use terraform to create and apply branch protection to github repos, this leads to regular failures due too many github API requests. The branch protection does not really need refresh after it is applied initially but accounts for the bulk of the API calls. In this case this setting could prevent the unnecessary API calls to github.

image

@AurimasNav
Copy link

AurimasNav commented Mar 18, 2024

This would also be helpful for the databricks_mount resource: https://registry.terraform.io/providers/databricks/databricks/latest/docs/resources/mount. I need to seed the mount but I don't care about it afterwards, but every Terraform plan brings the entire cluster up just so it can check the state, which takes a lot of time and unneeded compute.

databricks mounts are annoying when developing/debugging locally, every terraform plan waits for cluster to come up, I would love some cli flag to ignore specific resources during refresh stage in these cases

@jeffreybartosiewicz
Copy link

Bumping this thread with another example. I have a TF repo to deploy and configure our Bigip F5 load balancers in azure. The script runs a Declarative Onboard process when the load balancer is first built but after that i dont care if the config changes. TF sometimes complains that the DO has changed and it adds a lot of time to deployments. Being able to set a lifecycle flag to no longer refresh the resource would be very handy.

@jwolf20
Copy link

jwolf20 commented Jan 10, 2025

Just to add another example here. I have a case where I am deploying an AWS Redshift Cluster and using redshiftdata_statements to do a little bit of initialization (setup some basic tables, users, etc.).

Once I have finished my initial deployment and the redshiftdata_statements are present in the state file I can assume they have run and the database is setup. I added ignore_changes=all to these resources in an effort to make it so that they only run once. However I noticed that every day terraform would want to recreate all of my redshiftdata_statement resources. That is when I realized that the refresh method for this resources checks an API that removes entries after 24 hours, so even though I have ignore_chages=all set, it is the refresh call that is messing me up and informing terraform that it needs to recreate these resources.

Adding an option to avoid attempting to refresh these specific resources would be incredibly useful to helping keep my development process clean and efficient.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Projects
None yet
Development

No branches or pull requests

10 participants