Skip to content

Commit c8620fe

Browse files
RC-Repositoriesaggarg
authored andcommitted
tools: Extend AWS entity cleanup command
In `createIoTThings.py` there is a command that takes a .json config and deletes all AWS entities described in there. This commit extends that command to search the credentials directory, and identify AWS Thing certificates before deleting these Things and their possibly related entities (which are generated using the .json config as well). This behaviour only occurs if the `extended` flag is set on the cleanup command. This commit also documents this command in `aws_tool.md`. Signed-off-by: Reuben Cartwright <[email protected]>
1 parent 61904f8 commit c8620fe

File tree

3 files changed

+150
-22
lines changed

3 files changed

+150
-22
lines changed

docs/components/aws_iot/aws_tool.md

+25-1
Original file line numberDiff line numberDiff line change
@@ -130,7 +130,7 @@ python tools/scripts/createIoTThings.py delete-thing -p --thing_name <your_thing
130130
131131
## Creating AWS IoT firmware update job (simplified)
132132

133-
The `create-update-simplified` command that (1) creates a Thing and Policy, (2) runs build, (3) creates a bucket, role, and update.
133+
The `create-update-simplified` command (1) creates a Thing and Policy, (2) runs build, (3) creates a bucket, role, and update.
134134
This command also re-uses AWS entities where possible, validating entities being re-used.
135135

136136

@@ -254,6 +254,30 @@ If you want to add a setting, for example `update_name` to the definitions you c
254254

255255
The `target_application` setting is special because it is not defined in the `json` file but can still be mentioned in definitions.
256256

257+
## Cleaning up after AWS IoT firmware update job (simplified)
258+
259+
The `cleanup-simplified` command uses the config file from `create-update-simplified` and deletes all AWS entities described there.
260+
Optionally, this command will check all credential files (such as certificates) to identify other Things created by the script.
261+
These Things are deleted with their certificates.
262+
The script identifies possibly linked AWS entities by using the .json config file to generate entity names. E.g. if the certificates for 'myTestThing' are found, and you have specified that 'policy_name' is '${thing_name}_policy', then the script will attempt to delete 'myTestThing_policy'.
263+
264+
To use this command:
265+
1. Fill the following fields in the `.json` config file:
266+
* `thing_name` with the name of your AWS Thing.
267+
* `role_prefix` with the prefix for your role. This prefix will be pre-pended to your role name with a hyphen by default. For example, with the prefix `Proj` and role name `role`, the completed role name will become `Proj-role`.
268+
2. Set up following [prerequisites](#prerequisites).
269+
3. Run the command below.
270+
271+
```sh
272+
python tools/scripts/createIoTThings.py cleanup-simplified
273+
```
274+
275+
That's it. Your AWS entities created by this script should now be deleted.
276+
277+
The only time this command will fail to find or remove all AWS entities is if:
278+
1. You have created an update, and deleted the role associated before the update is deleted. You need to try re-creating the role (e.g. via re-running `create-update-simplified` with the same config). You may see an error message indicating that you `cannot assume a role` to delete an OTA update.
279+
2. You have run `create-update-simplified` with config A, then used config B which specifies a different role, policy, update, or bucket name <b>format</b> from A. For example, changing `policy_name` from `${thing_name}_policy` in A to `myTestPolicy` in B will mean that `cleanup-simplified` cannot find `${thing_name}_policy` if run with config B. To fix this, run `cleanup-simplified` with each config separately.
280+
257281
## Troubleshooting
258282

259283
##### 1. My AWS credentials are rejected, despite being accepted earlier.

release_changes/202409131621.change

+1
Original file line numberDiff line numberDiff line change
@@ -0,0 +1 @@
1+
tools: Add command for automatically cleaning up all AWS entities created by 'createIoTThings.py create-update-simplified'

tools/scripts/createIoTThings.py

+124-21
Original file line numberDiff line numberDiff line change
@@ -18,6 +18,7 @@
1818

1919
from enum import Enum
2020
import os
21+
import glob
2122
import pathlib
2223
import time
2324
import traceback as tb
@@ -2064,6 +2065,20 @@ def _formatVars(
20642065

20652066

20662067
def _try_delete(target, del_func, **kwargs):
2068+
"""
2069+
This function calls:
2070+
>>> del_func(target, kwargs)
2071+
And informs the user whether deletion has succeeded or failed.
2072+
Example usage:
2073+
>>> _try_delete("my-test-bucket", _delete_bucket, force_delete=True)
2074+
2075+
2076+
Parameters:
2077+
target: the item to delete.
2078+
del_func: the function that will do the deletion, which
2079+
should return a boolean value.
2080+
kwargs: keyword parameters, e.g. 'force_delete=True' in the example.
2081+
"""
20672082
has_deleted = False
20682083
try:
20692084
has_deleted = del_func(target, kwargs)
@@ -3418,33 +3433,80 @@ def delete_certificate(ctx, certificate_id, config_file_path, log_level, force_d
34183433
ctx.exit(1)
34193434

34203435

3436+
def _try_delete_all(
3437+
thing_name: str,
3438+
policy_name: str,
3439+
bucket_name: str,
3440+
iam_role_name: str,
3441+
update_name: str,
3442+
):
3443+
"""
3444+
This function tries to force-delete delete each passed AWS entity (if it exists).
3445+
I.e. if a job is ongoing it will be deleted anyway.
3446+
This function displays an error if an entity exists but deletion fails, otherwise
3447+
it will display a success message.
3448+
It is important that this function deletes the OTA update before
3449+
anything else. E.g. deleting the role or Thing before the update will cause
3450+
deletion to fail.
3451+
3452+
Parameters:
3453+
flags (Flags): contains metadata needed for AWS and OTA updates
3454+
(e.g. credentials).
3455+
"""
3456+
ota_job_name = OTA_JOB_NAME_PREFIX + update_name
3457+
if _does_job_exist(ota_job_name):
3458+
_try_delete(ota_job_name, _delete_ota_update, force_delete=True)
3459+
if _does_bucket_exist(bucket_name):
3460+
_try_delete(bucket_name, _delete_bucket, force_delete=True)
3461+
if _does_role_exist(iam_role_name):
3462+
_try_delete(iam_role_name, _delete_iam_role, force_delete=True)
3463+
if _does_policy_exist(policy_name):
3464+
_try_delete(policy_name, _delete_policy, prune=True)
3465+
if _does_thing_exist(thing_name):
3466+
_try_delete(thing_name, _delete_thing)
3467+
3468+
34213469
# Defines Command-line interface for deleting the Thing,
34223470
# Policy, Bucket, Role, and Update
34233471
# specified by createIoTThings_settings.json.
34243472
@cli.command(cls=StdCommand)
3473+
@click.option(
3474+
"--config_file_path",
3475+
help=".json file defining arguments for creating an OTA update.",
3476+
default="createIoTThings_settings.json",
3477+
)
3478+
@click.option(
3479+
"-e",
3480+
"--extended",
3481+
"extended",
3482+
help="For every certificate found in the credentials directory, "
3483+
"delete the Thing and all related AWS entities without asking first.",
3484+
is_flag=True,
3485+
)
34253486
@click.pass_context
34263487
def cleanup_simplified(
34273488
ctx,
34283489
config_file_path,
3490+
extended,
34293491
log_level,
34303492
):
3493+
unformatted_settings = {}
34313494
settings = {}
34323495
# Read .json file, pass parameters to flags.
34333496
try:
34343497
contents = read_whole_file(config_file_path)
3435-
settings = json.loads(contents)
3436-
settings["format_vars"] = settings["format_vars"].replace(
3437-
"target_application;", ""
3438-
)
3439-
settings = _formatVars(settings, ctx)
3498+
unformatted_settings = json.loads(contents)
3499+
unformatted_settings["format_vars"] = unformatted_settings[
3500+
"format_vars"
3501+
].replace("target_application;", "")
3502+
settings = _formatVars(unformatted_settings, ctx)
34403503
logging.debug("Settings .json file parsed to: " + str(settings))
34413504
except FileNotFoundError:
34423505
logging.error("Config file not found at " + config_file_path)
34433506
ctx.exit(1)
34443507
except json.JSONDecodeError:
34453508
logging.error("Failed to parse .json file: " + config_file_path)
34463509
ctx.exit(1)
3447-
34483510
# Check the required settings exist.
34493511
thing_name = _tryGetSetting(
34503512
"thing_name", settings=settings, ctx=ctx, errorOnFailure=True
@@ -3461,21 +3523,62 @@ def cleanup_simplified(
34613523
update_name = _tryGetSetting(
34623524
"update_name", settings=settings, ctx=ctx, errorOnFailure=True
34633525
)
3464-
ota_job_name = OTA_JOB_NAME_PREFIX + update_name
3465-
if _does_job_exist(ota_job_name):
3466-
_delete_ota_update(ota_update_name=update_name, force_delete=True)
3467-
if _wait_for_job_deleted(ota_job_name):
3468-
logging.info("Deleted OTA update " + update_name + " successfully.")
3469-
else:
3470-
logging.warning("Failed to delete OTA update job.")
3471-
if _does_bucket_exist(bucket_name):
3472-
_try_delete(bucket_name, _delete_bucket, force_delete=True)
3473-
if _does_role_exist(iam_role_name):
3474-
_try_delete(iam_role_name, _delete_iam_role, force_delete=True)
3475-
if _does_policy_exist(policy_name):
3476-
_try_delete(policy_name, _delete_policy, prune=True)
3477-
if _does_thing_exist(thing_name):
3478-
_try_delete(thing_name, _delete_thing)
3526+
_try_delete_all(thing_name, policy_name, bucket_name, iam_role_name, update_name)
3527+
# Identify all certificates in the credentials folder.
3528+
if extended:
3529+
fileDir = os.path.dirname(os.path.realpath("__file__"))
3530+
credentials_path = _tryGetSetting(
3531+
"credentials_path", settings=settings, ctx=ctx, errorOnFailure=True
3532+
)
3533+
for file in glob.iglob(
3534+
os.path.join(fileDir, credentials_path, "thing_certificate_*.pem.crt")
3535+
):
3536+
certificateFile = re.search("thing_certificate_(.*?).pem.crt", file)
3537+
if certificateFile is not None:
3538+
thing_name = re.search("thing_certificate_(.*?).pem.crt", file).group(1)
3539+
# Note that if you have modified a field other than thing_name, the
3540+
# script will not reset that field. E.g. 'policy_name' being hard-coded
3541+
# will mean this script will not attempt to find policies using
3542+
# the policy_name_DEFAULT with the formatter.
3543+
settings["thing_name"] = thing_name
3544+
# re-format settings using the new thing_name
3545+
settings = _formatVars(unformatted_settings, ctx)
3546+
# delete all associated entities.
3547+
policy_name = _tryGetSetting(
3548+
"policy_name", settings=settings, ctx=ctx, errorOnFailure=True
3549+
)
3550+
bucket_name = _tryGetSetting(
3551+
"bucket_name", settings=settings, ctx=ctx, errorOnFailure=True
3552+
)
3553+
iam_role_name = _tryGetSetting(
3554+
"iam_role_name", settings=settings, ctx=ctx, errorOnFailure=True
3555+
)
3556+
update_name = _tryGetSetting(
3557+
"update_name", settings=settings, ctx=ctx, errorOnFailure=True
3558+
)
3559+
_try_delete_all(
3560+
thing_name, policy_name, bucket_name, iam_role_name, update_name
3561+
)
3562+
# delete credential files associated.
3563+
certificateFile = os.path.join(
3564+
fileDir, credentials_path, f"thing_certificate_{thing_name}.pem.crt"
3565+
)
3566+
if os.path.exists(certificateFile):
3567+
os.remove(certificateFile)
3568+
privateKeyFile = os.path.join(
3569+
fileDir, credentials_path, f"thing_private_key_{thing_name}.pem.key"
3570+
)
3571+
if os.path.exists(privateKeyFile):
3572+
os.remove(privateKeyFile)
3573+
publicKeyFile = os.path.join(
3574+
fileDir, credentials_path, f"thing_public_key_{thing_name}.pem.key"
3575+
)
3576+
if os.path.exists(publicKeyFile):
3577+
os.remove(publicKeyFile)
3578+
logging.info(
3579+
"Cleaned up all associated entities (using your config"
3580+
" file options) and files."
3581+
)
34793582
logging.info("All done!")
34803583
ctx.exit(0)
34813584

0 commit comments

Comments
 (0)