2016-03-31 11:29:39 -05:00
package command
import (
"fmt"
"strings"
2021-05-17 14:00:50 -05:00
"github.com/hashicorp/terraform/internal/addrs"
2021-05-17 14:07:38 -05:00
"github.com/hashicorp/terraform/internal/command/arguments"
"github.com/hashicorp/terraform/internal/command/clistate"
"github.com/hashicorp/terraform/internal/command/views"
2022-08-30 17:27:15 -05:00
"github.com/hashicorp/terraform/internal/terraform"
2021-05-17 12:11:06 -05:00
"github.com/hashicorp/terraform/internal/tfdiags"
2018-10-25 08:41:00 -05:00
"github.com/mitchellh/cli"
2016-03-31 11:29:39 -05:00
)
// StateRmCommand is a Command implementation that shows a single resource.
type StateRmCommand struct {
StateMeta
}
func ( c * StateRmCommand ) Run ( args [ ] string ) int {
2020-04-01 14:01:08 -05:00
args = c . Meta . process ( args )
2018-10-25 08:41:00 -05:00
var dryRun bool
backend: Validate remote backend Terraform version
When using the enhanced remote backend, a subset of all Terraform
operations are supported. Of these, only plan and apply can be executed
on the remote infrastructure (e.g. Terraform Cloud). Other operations
run locally and use the remote backend for state storage.
This causes problems when the local version of Terraform does not match
the configured version from the remote workspace. If the two versions
are incompatible, an `import` or `state mv` operation can cause the
remote workspace to be unusable until a manual fix is applied.
To prevent this from happening accidentally, this commit introduces a
check that the local Terraform version and the configured remote
workspace Terraform version are compatible. This check is skipped for
commands which do not write state, and can also be disabled by the use
of a new command-line flag, `-ignore-remote-version`.
Terraform version compatibility is defined as:
- For all releases before 0.14.0, local must exactly equal remote, as
two different versions cannot share state;
- 0.14.0 to 1.0.x are compatible, as we will not change the state
version number until at least Terraform 1.1.0;
- Versions after 1.1.0 must have the same major and minor versions, as
we will not change the state version number in a patch release.
If the two versions are incompatible, a diagnostic is displayed,
advising that the error can be suppressed with `-ignore-remote-version`.
When this flag is used, the diagnostic is still displayed, but as a
warning instead of an error.
Commands which will not write state can assert this fact by calling the
helper `meta.ignoreRemoteBackendVersionConflict`, which will disable the
checks. Those which can write state should instead call the helper
`meta.remoteBackendVersionCheck`, which will return diagnostics for
display.
In addition to these explicit paths for managing the version check, we
have an implicit check in the remote backend's state manager
initialization method. Both of the above helpers will disable this
check. This fallback is in place to ensure that future code paths which
access state cannot accidentally skip the remote version check.
2020-11-13 15:43:56 -06:00
cmdFlags := c . Meta . ignoreRemoteVersionFlagSet ( "state rm" )
2018-10-25 08:41:00 -05:00
cmdFlags . BoolVar ( & dryRun , "dry-run" , false , "dry run" )
2017-07-26 12:08:09 -05:00
cmdFlags . StringVar ( & c . backupPath , "backup" , "-" , "backup" )
2018-11-21 08:35:27 -06:00
cmdFlags . BoolVar ( & c . Meta . stateLock , "lock" , true , "lock state" )
cmdFlags . DurationVar ( & c . Meta . stateLockTimeout , "lock-timeout" , 0 , "lock timeout" )
2017-07-26 12:08:09 -05:00
cmdFlags . StringVar ( & c . statePath , "state" , "" , "path" )
2016-03-31 11:29:39 -05:00
if err := cmdFlags . Parse ( args ) ; err != nil {
2019-08-16 07:31:21 -05:00
c . Ui . Error ( fmt . Sprintf ( "Error parsing command-line flags: %s\n" , err . Error ( ) ) )
return 1
2016-03-31 11:29:39 -05:00
}
2018-11-21 08:35:27 -06:00
args = cmdFlags . Args ( )
2017-07-05 16:58:08 -05:00
if len ( args ) < 1 {
2018-10-25 08:41:00 -05:00
c . Ui . Error ( "At least one address is required.\n" )
return cli . RunResultHelp
2017-07-05 16:58:08 -05:00
}
2022-03-31 12:42:42 -05:00
if diags := c . Meta . checkRequiredVersion ( ) ; diags != nil {
c . showDiagnostics ( diags )
return 1
}
2018-10-22 08:52:53 -05:00
// Get the state
2018-10-15 10:38:06 -05:00
stateMgr , err := c . State ( )
2016-03-31 11:29:39 -05:00
if err != nil {
c . Ui . Error ( fmt . Sprintf ( errStateLoadingState , err ) )
2017-07-27 13:10:52 -05:00
return 1
2016-03-31 11:29:39 -05:00
}
2019-01-08 07:57:52 -06:00
if c . stateLock {
2021-02-16 06:19:22 -06:00
stateLocker := clistate . NewLocker ( c . stateLockTimeout , views . NewStateLocker ( arguments . ViewHuman , c . View ) )
if diags := stateLocker . Lock ( stateMgr , "state-rm" ) ; diags . HasErrors ( ) {
c . showDiagnostics ( diags )
2019-01-08 07:57:52 -06:00
return 1
}
2021-02-16 06:19:22 -06:00
defer func ( ) {
if diags := stateLocker . Unlock ( ) ; diags . HasErrors ( ) {
c . showDiagnostics ( diags )
}
} ( )
2019-01-08 07:57:52 -06:00
}
2018-10-15 10:38:06 -05:00
if err := stateMgr . RefreshState ( ) ; err != nil {
2018-10-22 08:52:53 -05:00
c . Ui . Error ( fmt . Sprintf ( "Failed to refresh state: %s" , err ) )
2017-02-21 22:35:43 -06:00
return 1
}
2016-03-31 11:29:39 -05:00
2018-10-15 10:38:06 -05:00
state := stateMgr . State ( )
if state == nil {
2020-12-01 11:34:50 -06:00
c . Ui . Error ( errStateNotFound )
2016-03-31 11:29:39 -05:00
return 1
}
2019-03-15 21:51:26 -05:00
// This command primarily works with resource instances, though it will
// also clean up any modules and resources left empty by actions it takes.
var addrs [ ] addrs . AbsResourceInstance
var diags tfdiags . Diagnostics
for _ , addrStr := range args {
moreAddrs , moreDiags := c . lookupResourceInstanceAddr ( state , true , addrStr )
addrs = append ( addrs , moreAddrs ... )
diags = diags . Append ( moreDiags )
2018-10-15 10:38:06 -05:00
}
2019-03-15 21:51:26 -05:00
if diags . HasErrors ( ) {
c . showDiagnostics ( diags )
return 1
2018-10-15 10:38:06 -05:00
}
2019-03-15 21:51:26 -05:00
prefix := "Removed "
2018-10-25 08:41:00 -05:00
if dryRun {
2019-03-15 21:51:26 -05:00
prefix = "Would remove "
2018-10-15 10:38:06 -05:00
}
2016-03-31 11:29:39 -05:00
2018-10-22 08:52:53 -05:00
var isCount int
2018-10-15 10:38:06 -05:00
ss := state . SyncWrapper ( )
2019-03-15 21:51:26 -05:00
for _ , addr := range addrs {
isCount ++
c . Ui . Output ( prefix + addr . String ( ) )
if ! dryRun {
ss . ForgetResourceInstanceAll ( addr )
ss . RemoveResourceIfEmpty ( addr . ContainingResource ( ) )
2018-10-22 08:52:53 -05:00
}
2018-10-15 10:38:06 -05:00
}
2018-10-25 08:41:00 -05:00
if dryRun {
2018-10-22 08:52:53 -05:00
if isCount == 0 {
c . Ui . Output ( "Would have removed nothing." )
}
return 0 // This is as far as we go in dry-run mode
2018-10-15 10:38:06 -05:00
}
2017-09-20 14:48:27 -05:00
2022-08-29 11:10:03 -05:00
b , backendDiags := c . Backend ( nil )
2022-08-26 15:32:18 -05:00
diags = diags . Append ( backendDiags )
if backendDiags . HasErrors ( ) {
c . showDiagnostics ( diags )
return 1
}
2022-08-25 14:57:40 -05:00
// Get schemas, if possible, before writing state
2022-08-29 11:10:03 -05:00
var schemas * terraform . Schemas
if isCloudMode ( b ) {
2022-08-30 17:52:51 -05:00
var schemaDiags tfdiags . Diagnostics
schemas , schemaDiags = c . MaybeGetSchemas ( state , nil )
2022-08-30 18:03:57 -05:00
diags = diags . Append ( schemaDiags )
2022-08-25 14:57:40 -05:00
}
2018-10-15 10:38:06 -05:00
if err := stateMgr . WriteState ( state ) ; err != nil {
2016-03-31 11:29:39 -05:00
c . Ui . Error ( fmt . Sprintf ( errStateRmPersist , err ) )
return 1
}
2022-08-25 14:57:40 -05:00
if err := stateMgr . PersistState ( schemas ) ; err != nil {
2016-03-31 11:29:39 -05:00
c . Ui . Error ( fmt . Sprintf ( errStateRmPersist , err ) )
return 1
}
2020-06-24 13:03:23 -05:00
if len ( diags ) > 0 && isCount != 0 {
2019-03-15 21:51:26 -05:00
c . showDiagnostics ( diags )
}
2018-10-22 08:52:53 -05:00
if isCount == 0 {
2020-06-24 13:03:23 -05:00
diags = diags . Append ( tfdiags . Sourceless (
tfdiags . Error ,
"Invalid target address" ,
"No matching objects found. To view the available instances, use \"terraform state list\". Please modify the address to reference a specific instance." ,
) )
c . showDiagnostics ( diags )
return 1
2018-10-22 08:52:53 -05:00
}
2020-06-24 13:03:23 -05:00
c . Ui . Output ( fmt . Sprintf ( "Successfully removed %d resource instance(s)." , isCount ) )
2016-03-31 11:29:39 -05:00
return 0
}
func ( c * StateRmCommand ) Help ( ) string {
helpText := `
2021-02-22 08:25:56 -06:00
Usage : terraform [ global options ] state rm [ options ] ADDRESS ...
2016-03-31 11:29:39 -05:00
2019-03-15 21:51:26 -05:00
Remove one or more items from the Terraform state , causing Terraform to
"forget" those items without first destroying them in the remote system .
2016-03-31 11:29:39 -05:00
2018-10-15 10:38:06 -05:00
This command removes one or more resource instances from the Terraform state
based on the addresses given . You can view and list the available instances
2016-03-31 11:29:39 -05:00
with "terraform state list" .
2019-03-15 21:51:26 -05:00
If you give the address of an entire module then all of the instances in
that module and any of its child modules will be removed from the state .
If you give the address of a resource that has "count" or "for_each" set ,
all of the instances of that resource will be removed from the state .
2016-03-31 11:29:39 -05:00
Options :
backend: Validate remote backend Terraform version
When using the enhanced remote backend, a subset of all Terraform
operations are supported. Of these, only plan and apply can be executed
on the remote infrastructure (e.g. Terraform Cloud). Other operations
run locally and use the remote backend for state storage.
This causes problems when the local version of Terraform does not match
the configured version from the remote workspace. If the two versions
are incompatible, an `import` or `state mv` operation can cause the
remote workspace to be unusable until a manual fix is applied.
To prevent this from happening accidentally, this commit introduces a
check that the local Terraform version and the configured remote
workspace Terraform version are compatible. This check is skipped for
commands which do not write state, and can also be disabled by the use
of a new command-line flag, `-ignore-remote-version`.
Terraform version compatibility is defined as:
- For all releases before 0.14.0, local must exactly equal remote, as
two different versions cannot share state;
- 0.14.0 to 1.0.x are compatible, as we will not change the state
version number until at least Terraform 1.1.0;
- Versions after 1.1.0 must have the same major and minor versions, as
we will not change the state version number in a patch release.
If the two versions are incompatible, a diagnostic is displayed,
advising that the error can be suppressed with `-ignore-remote-version`.
When this flag is used, the diagnostic is still displayed, but as a
warning instead of an error.
Commands which will not write state can assert this fact by calling the
helper `meta.ignoreRemoteBackendVersionConflict`, which will disable the
checks. Those which can write state should instead call the helper
`meta.remoteBackendVersionCheck`, which will return diagnostics for
display.
In addition to these explicit paths for managing the version check, we
have an implicit check in the remote backend's state manager
initialization method. Both of the above helpers will disable this
check. This fallback is in place to ensure that future code paths which
access state cannot accidentally skip the remote version check.
2020-11-13 15:43:56 -06:00
- dry - run If set , prints out what would ' ve been removed but
doesn ' t actually remove anything .
2018-10-15 10:38:06 -05:00
backend: Validate remote backend Terraform version
When using the enhanced remote backend, a subset of all Terraform
operations are supported. Of these, only plan and apply can be executed
on the remote infrastructure (e.g. Terraform Cloud). Other operations
run locally and use the remote backend for state storage.
This causes problems when the local version of Terraform does not match
the configured version from the remote workspace. If the two versions
are incompatible, an `import` or `state mv` operation can cause the
remote workspace to be unusable until a manual fix is applied.
To prevent this from happening accidentally, this commit introduces a
check that the local Terraform version and the configured remote
workspace Terraform version are compatible. This check is skipped for
commands which do not write state, and can also be disabled by the use
of a new command-line flag, `-ignore-remote-version`.
Terraform version compatibility is defined as:
- For all releases before 0.14.0, local must exactly equal remote, as
two different versions cannot share state;
- 0.14.0 to 1.0.x are compatible, as we will not change the state
version number until at least Terraform 1.1.0;
- Versions after 1.1.0 must have the same major and minor versions, as
we will not change the state version number in a patch release.
If the two versions are incompatible, a diagnostic is displayed,
advising that the error can be suppressed with `-ignore-remote-version`.
When this flag is used, the diagnostic is still displayed, but as a
warning instead of an error.
Commands which will not write state can assert this fact by calling the
helper `meta.ignoreRemoteBackendVersionConflict`, which will disable the
checks. Those which can write state should instead call the helper
`meta.remoteBackendVersionCheck`, which will return diagnostics for
display.
In addition to these explicit paths for managing the version check, we
have an implicit check in the remote backend's state manager
initialization method. Both of the above helpers will disable this
check. This fallback is in place to ensure that future code paths which
access state cannot accidentally skip the remote version check.
2020-11-13 15:43:56 -06:00
- backup = PATH Path where Terraform should write the backup
state .
2016-03-31 11:29:39 -05:00
2021-05-12 11:05:03 -05:00
- lock = false Don ' t hold a state lock during the operation . This is
dangerous if others might concurrently run commands
against the same workspace .
2018-11-21 08:35:27 -06:00
backend: Validate remote backend Terraform version
When using the enhanced remote backend, a subset of all Terraform
operations are supported. Of these, only plan and apply can be executed
on the remote infrastructure (e.g. Terraform Cloud). Other operations
run locally and use the remote backend for state storage.
This causes problems when the local version of Terraform does not match
the configured version from the remote workspace. If the two versions
are incompatible, an `import` or `state mv` operation can cause the
remote workspace to be unusable until a manual fix is applied.
To prevent this from happening accidentally, this commit introduces a
check that the local Terraform version and the configured remote
workspace Terraform version are compatible. This check is skipped for
commands which do not write state, and can also be disabled by the use
of a new command-line flag, `-ignore-remote-version`.
Terraform version compatibility is defined as:
- For all releases before 0.14.0, local must exactly equal remote, as
two different versions cannot share state;
- 0.14.0 to 1.0.x are compatible, as we will not change the state
version number until at least Terraform 1.1.0;
- Versions after 1.1.0 must have the same major and minor versions, as
we will not change the state version number in a patch release.
If the two versions are incompatible, a diagnostic is displayed,
advising that the error can be suppressed with `-ignore-remote-version`.
When this flag is used, the diagnostic is still displayed, but as a
warning instead of an error.
Commands which will not write state can assert this fact by calling the
helper `meta.ignoreRemoteBackendVersionConflict`, which will disable the
checks. Those which can write state should instead call the helper
`meta.remoteBackendVersionCheck`, which will return diagnostics for
display.
In addition to these explicit paths for managing the version check, we
have an implicit check in the remote backend's state manager
initialization method. Both of the above helpers will disable this
check. This fallback is in place to ensure that future code paths which
access state cannot accidentally skip the remote version check.
2020-11-13 15:43:56 -06:00
- lock - timeout = 0 s Duration to retry a state lock .
2018-11-21 08:35:27 -06:00
backend: Validate remote backend Terraform version
When using the enhanced remote backend, a subset of all Terraform
operations are supported. Of these, only plan and apply can be executed
on the remote infrastructure (e.g. Terraform Cloud). Other operations
run locally and use the remote backend for state storage.
This causes problems when the local version of Terraform does not match
the configured version from the remote workspace. If the two versions
are incompatible, an `import` or `state mv` operation can cause the
remote workspace to be unusable until a manual fix is applied.
To prevent this from happening accidentally, this commit introduces a
check that the local Terraform version and the configured remote
workspace Terraform version are compatible. This check is skipped for
commands which do not write state, and can also be disabled by the use
of a new command-line flag, `-ignore-remote-version`.
Terraform version compatibility is defined as:
- For all releases before 0.14.0, local must exactly equal remote, as
two different versions cannot share state;
- 0.14.0 to 1.0.x are compatible, as we will not change the state
version number until at least Terraform 1.1.0;
- Versions after 1.1.0 must have the same major and minor versions, as
we will not change the state version number in a patch release.
If the two versions are incompatible, a diagnostic is displayed,
advising that the error can be suppressed with `-ignore-remote-version`.
When this flag is used, the diagnostic is still displayed, but as a
warning instead of an error.
Commands which will not write state can assert this fact by calling the
helper `meta.ignoreRemoteBackendVersionConflict`, which will disable the
checks. Those which can write state should instead call the helper
`meta.remoteBackendVersionCheck`, which will return diagnostics for
display.
In addition to these explicit paths for managing the version check, we
have an implicit check in the remote backend's state manager
initialization method. Both of the above helpers will disable this
check. This fallback is in place to ensure that future code paths which
access state cannot accidentally skip the remote version check.
2020-11-13 15:43:56 -06:00
- state = PATH Path to the state file to update . Defaults to the
current workspace state .
- ignore - remote - version Continue even if remote and local Terraform versions
2021-01-27 12:58:41 -06:00
are incompatible . This may result in an unusable
workspace , and should be used with extreme caution .
2016-03-31 11:29:39 -05:00
`
return strings . TrimSpace ( helpText )
}
func ( c * StateRmCommand ) Synopsis ( ) string {
2018-10-15 10:38:06 -05:00
return "Remove instances from the state"
2016-03-31 11:29:39 -05:00
}
const errStateRmPersist = ` Error saving the state : % s
The state was not saved . No items were removed from the persisted
state . No backup was created since no modification occurred . Please
resolve the issue above and try again . `