diff --git a/ggdeploymentd/src/dependency_resolver.c b/ggdeploymentd/src/dependency_resolver.c new file mode 100644 index 000000000..42247b8ff --- /dev/null +++ b/ggdeploymentd/src/dependency_resolver.c @@ -0,0 +1,927 @@ +// aws-greengrass-lite - AWS IoT Greengrass runtime for constrained devices +// Copyright Amazon.com, Inc. or its affiliates. All Rights Reserved. +// SPDX-License-Identifier: Apache-2.0 + +#include "dependency_resolver.h" +#include "component_manager.h" +#include "deployment_configuration.h" +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include + +static GglError get_device_thing_groups(GglBuffer *response) { + GglByteVec data_endpoint = GGL_BYTE_VEC(config.data_endpoint); + + GglError ret = get_data_endpoint(&data_endpoint); + if (ret != GGL_ERR_OK) { + GGL_LOGE("Failed to get dataplane endpoint."); + return ret; + } + + GglByteVec region = GGL_BYTE_VEC(config.region); + ret = get_region(®ion); + if (ret != GGL_ERR_OK) { + GGL_LOGE("Failed to get region."); + return ret; + } + + GglByteVec port = GGL_BYTE_VEC(config.port); + ret = get_data_port(&port); + if (ret != GGL_ERR_OK) { + GGL_LOGE("Failed to get dataplane port."); + return ret; + } + + GglByteVec pkey_path = GGL_BYTE_VEC(config.pkey_path); + ret = get_private_key_path(&pkey_path); + if (ret != GGL_ERR_OK) { + GGL_LOGE("Failed to get private key path."); + return ret; + } + + GglByteVec cert_path = GGL_BYTE_VEC(config.cert_path); + ret = get_cert_path(&cert_path); + if (ret != GGL_ERR_OK) { + GGL_LOGE("Failed to get certificate path."); + return ret; + } + + GglByteVec rootca_path = GGL_BYTE_VEC(config.rootca_path); + ret = get_rootca_path(&rootca_path); + if (ret != GGL_ERR_OK) { + GGL_LOGE("Failed to get certificate path."); + return ret; + } + + CertificateDetails cert_details + = { .gghttplib_cert_path = config.cert_path, + .gghttplib_root_ca_path = config.rootca_path, + .gghttplib_p_key_path = config.pkey_path }; + + uint8_t thing_name_arr[128]; + GglBuffer thing_name = GGL_BUF(thing_name_arr); + ret = get_thing_name(&thing_name); + if (ret != GGL_ERR_OK) { + GGL_LOGE("Failed to get thing name."); + return ret; + } + + uint8_t uri_path_buf[PATH_MAX]; + GglByteVec uri_path_vec = GGL_BYTE_VEC(uri_path_buf); + ret = ggl_byte_vec_append( + &uri_path_vec, GGL_STR("greengrass/v2/coreDevices/") + ); + ggl_byte_vec_chain_append(&ret, &uri_path_vec, thing_name); + ggl_byte_vec_chain_append(&ret, &uri_path_vec, GGL_STR("/thingGroups")); + if (ret != GGL_ERR_OK) { + GGL_LOGE("Failed to create thing groups call uri."); + return ret; + } + + ret = gg_dataplane_call( + data_endpoint.buf, + port.buf, + uri_path_vec.buf, + cert_details, + NULL, + response + ); + if (ret != GGL_ERR_OK) { + GGL_LOGE( + "The listThingGroupsForCoreDevice call failed with response %.*s.", + (int) response->len, + response->data + ); + return ret; + } + + GGL_LOGD( + "Received response from thingGroups dataplane call: %.*s", + (int) response->len, + response->data + ); + + return GGL_ERR_OK; +} + +static GglError generate_resolve_component_candidates_body( + GglBuffer component_name, + GglBuffer component_requirements, + GglByteVec *body_vec +) { + GglError byte_vec_ret = GGL_ERR_OK; + ggl_byte_vec_chain_append( + &byte_vec_ret, body_vec, GGL_STR("{\"componentCandidates\": [") + ); + + ggl_byte_vec_chain_append( + &byte_vec_ret, body_vec, GGL_STR("{\"componentName\": \"") + ); + ggl_byte_vec_chain_append(&byte_vec_ret, body_vec, component_name); + ggl_byte_vec_chain_append( + &byte_vec_ret, + body_vec, + GGL_STR("\",\"versionRequirements\": {\"requirements\": \"") + ); + ggl_byte_vec_chain_append(&byte_vec_ret, body_vec, component_requirements); + ggl_byte_vec_chain_append(&byte_vec_ret, body_vec, GGL_STR("\"}}")); + + // TODO: Include architecture requirements if any + ggl_byte_vec_chain_append( + &byte_vec_ret, + body_vec, + GGL_STR("],\"platform\": { \"attributes\": { \"os\" : \"linux\", " + "\"runtime\" : \"aws_nucleus_lite\" " + "},\"name\": \"linux\"}}") + ); + ggl_byte_vec_chain_push(&byte_vec_ret, body_vec, '\0'); + + GGL_LOGD("Body for call: %s", body_vec->buf.data); + + return GGL_ERR_OK; +} + +static GglError resolve_component_with_cloud( + GglBuffer component_name, + GglBuffer version_requirements, + GglBuffer *response +) { + static char resolve_candidates_body_buf[2048]; + GglByteVec body_vec = GGL_BYTE_VEC(resolve_candidates_body_buf); + GglError ret = generate_resolve_component_candidates_body( + component_name, version_requirements, &body_vec + ); + if (ret != GGL_ERR_OK) { + GGL_LOGE("Failed to generate body for resolveComponentCandidates call"); + return ret; + } + + GglByteVec data_endpoint = GGL_BYTE_VEC(config.data_endpoint); + ret = get_data_endpoint(&data_endpoint); + if (ret != GGL_ERR_OK) { + GGL_LOGE("Failed to get dataplane endpoint."); + return ret; + } + + GglByteVec region = GGL_BYTE_VEC(config.region); + ret = get_region(®ion); + if (ret != GGL_ERR_OK) { + GGL_LOGE("Failed to get region."); + return ret; + } + + GglByteVec port = GGL_BYTE_VEC(config.port); + ret = get_data_port(&port); + if (ret != GGL_ERR_OK) { + GGL_LOGE("Failed to get dataplane port."); + return ret; + } + + GglByteVec pkey_path = GGL_BYTE_VEC(config.pkey_path); + ret = get_private_key_path(&pkey_path); + if (ret != GGL_ERR_OK) { + GGL_LOGE("Failed to get private key path."); + return ret; + } + + GglByteVec cert_path = GGL_BYTE_VEC(config.cert_path); + ret = get_cert_path(&cert_path); + if (ret != GGL_ERR_OK) { + GGL_LOGE("Failed to get certificate path."); + return ret; + } + + GglByteVec rootca_path = GGL_BYTE_VEC(config.rootca_path); + ret = get_rootca_path(&rootca_path); + if (ret != GGL_ERR_OK) { + GGL_LOGE("Failed to get certificate path."); + return ret; + } + + CertificateDetails cert_details + = { .gghttplib_cert_path = config.cert_path, + .gghttplib_root_ca_path = config.rootca_path, + .gghttplib_p_key_path = config.pkey_path }; + + ret = gg_dataplane_call( + data_endpoint.buf, + port.buf, + GGL_STR("greengrass/v2/resolveComponentCandidates"), + cert_details, + resolve_candidates_body_buf, + response + ); + if (ret != GGL_ERR_OK) { + GGL_LOGE( + "Cloud resolution for the component failed with response %.*s.", + (int) response->len, + response->data + ); + return ret; + } + + GGL_LOGD( + "Received response from resolveComponentCandidates: %.*s", + (int) response->len, + response->data + ); + + return GGL_ERR_OK; +} + +static GglError parse_dataplane_response_and_save_recipe( + GglBuffer dataplane_response, + GglDeploymentHandlerThreadArgs *args, + GglBuffer *cloud_version +) { + GglObject json_candidates_response_obj; + // TODO: Figure out a better size. This response can be big. + uint8_t candidates_response_mem[100 * sizeof(GglObject)]; + GglBumpAlloc balloc = ggl_bump_alloc_init(GGL_BUF(candidates_response_mem)); + GglError ret = ggl_json_decode_destructive( + dataplane_response, &balloc.alloc, &json_candidates_response_obj + ); + if (ret != GGL_ERR_OK) { + GGL_LOGE("Error when parsing resolveComponentCandidates response to " + "json."); + return ret; + } + + if (json_candidates_response_obj.type != GGL_TYPE_MAP) { + GGL_LOGE("resolveComponentCandidates response did not parse into a " + "map."); + return ret; + } + + GglObject *resolved_component_versions; + if (!ggl_map_get( + json_candidates_response_obj.map, + GGL_STR("resolvedComponentVersions"), + &resolved_component_versions + )) { + GGL_LOGE("Missing resolvedComponentVersions."); + return ret; + } + if (resolved_component_versions->type != GGL_TYPE_LIST) { + GGL_LOGE("resolvedComponentVersions response is not a list."); + return ret; + } + + bool first_component = true; + GGL_LIST_FOREACH(resolved_version, resolved_component_versions->list) { + if (!first_component) { + GGL_LOGE( + "resolveComponentCandidates returned information for more than " + "one component." + ); + return GGL_ERR_INVALID; + } + first_component = false; + + if (resolved_version->type != GGL_TYPE_MAP) { + GGL_LOGE("Resolved version is not of type map."); + return ret; + } + + GglObject *cloud_component_arn; + GglObject *cloud_component_name; + GglObject *cloud_component_version; + GglObject *vendor_guidance; + GglObject *recipe_file_content; + + ret = ggl_map_validate( + resolved_version->map, + GGL_MAP_SCHEMA( + { GGL_STR("arn"), true, GGL_TYPE_BUF, &cloud_component_arn }, + { GGL_STR("componentName"), + true, + GGL_TYPE_BUF, + &cloud_component_name }, + { GGL_STR("componentVersion"), + true, + GGL_TYPE_BUF, + &cloud_component_version }, + { GGL_STR("vendorGuidance"), + false, + GGL_TYPE_BUF, + &vendor_guidance }, + { GGL_STR("recipe"), true, GGL_TYPE_BUF, &recipe_file_content }, + ) + ); + if (ret != GGL_ERR_OK) { + return ret; + } + assert(cloud_component_version->buf.len <= NAME_MAX); + + memcpy( + cloud_version->data, + cloud_component_version->buf.data, + cloud_component_version->buf.len + ); + cloud_version->len = cloud_component_version->buf.len; + + if (vendor_guidance != NULL) { + if (ggl_buffer_eq(vendor_guidance->buf, GGL_STR("DISCONTINUED"))) { + GGL_LOGW("The component version has been discontinued by " + "its " + "publisher. You can deploy this component version, " + "but " + "we recommend that you use a different version of " + "this " + "component"); + } + } + + if (recipe_file_content->buf.len == 0) { + GGL_LOGE("Recipe is empty."); + } + + ggl_base64_decode_in_place(&recipe_file_content->buf); + recipe_file_content->buf.data[recipe_file_content->buf.len] = '\0'; + + GGL_LOGD( + "Decoded recipe data as: %.*s", + (int) recipe_file_content->buf.len, + recipe_file_content->buf.data + ); + + static uint8_t recipe_name_buf[PATH_MAX]; + GglByteVec recipe_name_vec = GGL_BYTE_VEC(recipe_name_buf); + ret = ggl_byte_vec_append(&recipe_name_vec, cloud_component_name->buf); + ggl_byte_vec_chain_append(&ret, &recipe_name_vec, GGL_STR("-")); + ggl_byte_vec_chain_append( + &ret, &recipe_name_vec, cloud_component_version->buf + ); + // TODO: Actual support for .json files. We're writing a .json + // to a .yaml and relying on yaml being an almost-superset of + // json. + ggl_byte_vec_chain_append(&ret, &recipe_name_vec, GGL_STR(".yaml")); + if (ret != GGL_ERR_OK) { + GGL_LOGE("Failed to create recipe file name."); + return ret; + } + + static uint8_t recipe_dir_buf[PATH_MAX]; + GglByteVec recipe_dir_vec = GGL_BYTE_VEC(recipe_dir_buf); + ret = ggl_byte_vec_append( + &recipe_dir_vec, + ggl_buffer_from_null_term((char *) args->root_path.data) + ); + ggl_byte_vec_chain_append( + &ret, &recipe_dir_vec, GGL_STR("/packages/recipes/") + ); + if (ret != GGL_ERR_OK) { + GGL_LOGE("Failed to create recipe directory name."); + return ret; + } + + // Write file + int root_dir_fd = -1; + ret = ggl_dir_open(recipe_dir_vec.buf, O_PATH, true, &root_dir_fd); + if (ret != GGL_ERR_OK) { + GGL_LOGE("Failed to open dir when writing cloud recipe."); + return ret; + } + + int fd = -1; + ret = ggl_file_openat( + root_dir_fd, + recipe_name_vec.buf, + O_CREAT | O_WRONLY | O_TRUNC, + (mode_t) 0644, + &fd + ); + if (ret != GGL_ERR_OK) { + GGL_LOGE("Failed to open file at the dir when writing cloud " + "recipe."); + return ret; + } + + ret = ggl_file_write(fd, recipe_file_content->buf); + if (ret != GGL_ERR_OK) { + GGL_LOGE("Write to cloud recipe file failed"); + return ret; + } + + GGL_LOGD("Saved recipe under the name %s", recipe_name_vec.buf.data); + + ret = ggl_gg_config_write( + GGL_BUF_LIST(GGL_STR("services"), cloud_component_name->buf, ), + GGL_OBJ_MAP(GGL_MAP({ GGL_STR("arn"), *cloud_component_arn })), + 1 + ); + if (ret != GGL_ERR_OK) { + GGL_LOGE("Write of arn to config failed"); + return ret; + } + } + + return GGL_ERR_OK; +} + +GglError resolve_dependencies( + GglMap *root_components, + GglBuffer thing_group_name, + GglDeploymentHandlerThreadArgs *args, + GglAlloc *alloc, + GglKVVec *resolved_components_kv_vec +) { + assert(root_components->len != 0); + + GglError ret; + + // TODO: Decide on size + GglKVVec components_to_resolve = GGL_KV_VEC((GglKV[64]) { 0 }); + + static uint8_t version_requirements_mem[2048] = { 0 }; + GglBumpAlloc version_requirements_balloc + = ggl_bump_alloc_init(GGL_BUF(version_requirements_mem)); + + // Root components from current deployment + // TODO: Add current deployment's thing group to components map to config + for (GglKV *pair = (root_components)->pairs; + pair < &(root_components)->pairs[(root_components)->len]; + pair = &pair[1]) { + if (pair->val.type != GGL_TYPE_MAP) { + GGL_LOGE("Incorrect formatting for cloud deployment components " + "field."); + return GGL_ERR_INVALID; + } + + GglObject *val; + GglBuffer component_version = { 0 }; + if (ggl_map_get(pair->val.map, GGL_STR("version"), &val)) { + if (val->type != GGL_TYPE_BUF) { + GGL_LOGE("Received invalid argument."); + return GGL_ERR_INVALID; + } + component_version = val->buf; + } + + ret = ggl_kv_vec_push( + &components_to_resolve, + (GglKV) { pair->key, GGL_OBJ_BUF(component_version) } + ); + if (ret != GGL_ERR_OK) { + return ret; + } + } + + // At this point, components_to_resolve should be only a map of root + // component names to their version requirements from the deployment. + ret = ggl_gg_config_write( + GGL_BUF_LIST( + GGL_STR("services"), + GGL_STR("DeploymentService"), + GGL_STR("thingGroupsToRootComponents"), + thing_group_name + ), + GGL_OBJ_MAP(components_to_resolve.map), + 0 + ); + + if (ret != GGL_ERR_OK) { + GGL_LOGE( + "Failed to write thing group to root components map to ggconfigd." + ); + return ret; + } + + // Get list of thing groups + static uint8_t list_thing_groups_response_buf[2048] = { 0 }; + GglBuffer list_thing_groups_response + = GGL_BUF(list_thing_groups_response_buf); + + ret = get_device_thing_groups(&list_thing_groups_response); + if (ret != GGL_ERR_OK) { + return ret; + } + + // TODO: Add a schema and only parse the fields we need to save memory + GglObject json_thing_groups_object; + uint8_t thing_groups_response_mem[100 * sizeof(GglObject)]; + GglBumpAlloc thing_groups_json_balloc + = ggl_bump_alloc_init(GGL_BUF(thing_groups_response_mem)); + ret = ggl_json_decode_destructive( + list_thing_groups_response, + &thing_groups_json_balloc.alloc, + &json_thing_groups_object + ); + if (ret != GGL_ERR_OK) { + GGL_LOGE("Error when parsing listThingGroups response to " + "json."); + return ret; + } + + if (json_thing_groups_object.type != GGL_TYPE_MAP) { + GGL_LOGE("listThingGroups response did not parse into a " + "map."); + return ret; + } + + GglObject *thing_groups_list; + if (!ggl_map_get( + json_thing_groups_object.map, + GGL_STR("thingGroups"), + &thing_groups_list + )) { + GGL_LOGE("Missing thingGroups."); + return ret; + } + if (thing_groups_list->type != GGL_TYPE_LIST) { + GGL_LOGE("thingGroups response is not a list."); + return ret; + } + + // TODO: We want to also add root components from local deployments, not + // only thing group deployments. + GGL_LIST_FOREACH(thing_group_item, thing_groups_list->list) { + if (thing_group_item->type != GGL_TYPE_MAP) { + GGL_LOGE("Thing group item is not of type map."); + return ret; + } + + GglObject *thing_group_name_from_item; + + ret = ggl_map_validate( + thing_group_item->map, + GGL_MAP_SCHEMA( + { GGL_STR("thingGroupName"), + true, + GGL_TYPE_BUF, + &thing_group_name_from_item }, + ) + ); + if (ret != GGL_ERR_OK) { + return ret; + } + + if (!ggl_buffer_eq(thing_group_name_from_item->buf, thing_group_name)) { + GglObject group_root_components_read_value; + ret = ggl_gg_config_read( + GGL_BUF_LIST( + GGL_STR("services"), + GGL_STR("DeploymentService"), + GGL_STR("thingGroupsToRootComponents"), + thing_group_name_from_item->buf + ), + alloc, + &group_root_components_read_value + ); + if (ret != GGL_ERR_OK) { + GGL_LOGI( + "No info found in config for root components for thing " + "group %.*s, assuming no components are part of this thing " + "group.", + (int) thing_group_name_from_item->buf.len, + thing_group_name_from_item->buf.data + ); + } else { + if (group_root_components_read_value.type != GGL_TYPE_MAP) { + GGL_LOGE( + "Did not read a map from config for thing group to " + "root components map" + ); + return GGL_ERR_INVALID; + } + + GGL_MAP_FOREACH( + root_component_pair, group_root_components_read_value.map + ) { + // If component is already in the root component list, it + // must be the same version as the one already in the list + // or we have a conflict. + GglObject *existing_root_component_version; + ret = ggl_map_validate( + components_to_resolve.map, + GGL_MAP_SCHEMA( + { root_component_pair->key, + false, + GGL_TYPE_BUF, + &existing_root_component_version }, + ) + ); + if (ret != GGL_ERR_OK) { + return ret; + } + + bool need_to_add_root_component = true; + + if (existing_root_component_version != NULL) { + if (ggl_buffer_eq( + existing_root_component_version->buf, + root_component_pair->val.buf + )) { + need_to_add_root_component = false; + } else { + GGL_LOGE( + "There is a version conflict for component " + "%.*s, where two deployments are asking for " + "versions %.*s and %.*s. Please check that " + "this root component does not have conflicting " + "versions across your deployments.", + (int) root_component_pair->key.len, + root_component_pair->key.data, + (int) root_component_pair->val.buf.len, + root_component_pair->val.buf.data, + (int) existing_root_component_version->buf.len, + existing_root_component_version->buf.data + ); + return GGL_ERR_INVALID; + } + } + + if (need_to_add_root_component) { + GglBuffer root_component_name_buf; + ret = ggl_buf_clone( + root_component_pair->key, + alloc, + &root_component_name_buf + ); + if (ret != GGL_ERR_OK) { + return ret; + } + + GglBuffer root_component_version_buf; + ret = ggl_buf_clone( + root_component_pair->val.buf, + &version_requirements_balloc.alloc, + &root_component_version_buf + ); + if (ret != GGL_ERR_OK) { + return ret; + } + + ret = ggl_kv_vec_push( + &components_to_resolve, + (GglKV) { root_component_name_buf, + GGL_OBJ_BUF(root_component_version_buf) } + ); + GGL_LOGD( + "Added %.*s to the list of root components to " + "resolve " + "from " + "the thing group %.*s", + (int) root_component_name_buf.len, + root_component_name_buf.data, + (int) thing_group_name_from_item->buf.len, + thing_group_name_from_item->buf.data + ); + } + } + } + } + } + + GGL_MAP_FOREACH(pair, components_to_resolve.map) { + // We assume that we have not resolved a component yet if we are finding + // it in this map. + uint8_t resolved_version_arr[NAME_MAX]; + GglBuffer resolved_version = GGL_BUF(resolved_version_arr); + bool found_local_candidate = resolve_component_version( + pair->key, pair->val.buf, &resolved_version + ); + + if (!found_local_candidate) { + // Resolve with cloud and download recipe + static uint8_t resolve_component_candidates_response_buf[16384] + = { 0 }; + GglBuffer resolve_component_candidates_response + = GGL_BUF(resolve_component_candidates_response_buf); + + ret = resolve_component_with_cloud( + pair->key, pair->val.buf, &resolve_component_candidates_response + ); + if (ret != GGL_ERR_OK) { + return ret; + } + + bool is_empty_response = ggl_buffer_eq( + resolve_component_candidates_response, GGL_STR("{}") + ); + + if (is_empty_response) { + GGL_LOGI( + "Cloud version resolution failed for component %.*s.", + (int) pair->key.len, + pair->val.buf.data + ); + return GGL_ERR_FAILURE; + } + + ret = parse_dataplane_response_and_save_recipe( + resolve_component_candidates_response, args, &resolved_version + ); + if (ret != GGL_ERR_OK) { + return ret; + } + } + + // Add resolved component to list of resolved components + GglBuffer val_buf; + ret = ggl_buf_clone(resolved_version, alloc, &val_buf); + if (ret != GGL_ERR_OK) { + return ret; + } + + ret = ggl_kv_vec_push( + resolved_components_kv_vec, + (GglKV) { pair->key, GGL_OBJ_BUF(val_buf) } + ); + if (ret != GGL_ERR_OK) { + GGL_LOGE( + "Error while adding component to list of resolved component" + ); + return ret; + } + + // Find dependencies from recipe and add them to the list of components + // to resolve. If the dependency is for a component that is already + // resolved, verify that new requirements are satisfied and fail + // deployment if not. + + // Get actual recipe read + GglObject recipe_obj; + static uint8_t recipe_mem[8192] = { 0 }; + GglBumpAlloc balloc = ggl_bump_alloc_init(GGL_BUF(recipe_mem)); + ret = ggl_recipe_get_from_file( + args->root_path_fd, + pair->key, + resolved_version, + &balloc.alloc, + &recipe_obj + ); + if (ret != GGL_ERR_OK) { + return ret; + } + GglObject *component_dependencies = NULL; + + if (recipe_obj.type != GGL_TYPE_MAP) { + GGL_LOGE("Recipe object did not parse into a map."); + return GGL_ERR_INVALID; + } + + ret = ggl_map_validate( + recipe_obj.map, + GGL_MAP_SCHEMA( + { GGL_STR("ComponentDependencies"), + false, + GGL_TYPE_MAP, + &component_dependencies }, + ) + ); + if (ret != GGL_ERR_OK) { + return ret; + } + if (component_dependencies) { + GGL_MAP_FOREACH(dependency, component_dependencies->map) { + if (dependency->val.type != GGL_TYPE_MAP) { + GGL_LOGE( + "Component dependency in recipe does not have map data" + ); + return GGL_ERR_INVALID; + } + GglObject *dep_version_requirement = NULL; + ret = ggl_map_validate( + dependency->val.map, + GGL_MAP_SCHEMA( + { GGL_STR("VersionRequirement"), + true, + GGL_TYPE_BUF, + &dep_version_requirement }, + ) + ); + if (ret != GGL_ERR_OK) { + return ret; + } + + // If we already resolved the component version, check that it + // still satisfies the new requirement and fail otherwise. + GglObject *already_resolved_version; + ret = ggl_map_validate( + resolved_components_kv_vec->map, + GGL_MAP_SCHEMA( + { dependency->key, + false, + GGL_TYPE_BUF, + &already_resolved_version }, + ) + ); + if (ret != GGL_ERR_OK) { + return ret; + } + if (already_resolved_version) { + bool meets_requirements = is_in_range( + already_resolved_version->buf, + dep_version_requirement->buf + ); + if (!meets_requirements) { + GGL_LOGE("Already resolved component does not meet new " + "dependency requirement, failing dependency " + "resolution."); + return GGL_ERR_FAILURE; + } + } + + if (!already_resolved_version) { + // If we haven't resolved it yet, check if we have an + // existing requirement and append the new requirement if + // so. + GglObject *existing_requirements; + ret = ggl_map_validate( + components_to_resolve.map, + GGL_MAP_SCHEMA( + { dependency->key, + false, + GGL_TYPE_BUF, + &existing_requirements }, + ) + ); + if (ret != GGL_ERR_OK) { + return ret; + } + if (existing_requirements) { + uint8_t new_req_buf[PATH_MAX]; + GglByteVec new_req_vec = GGL_BYTE_VEC(new_req_buf); + ret = ggl_byte_vec_append( + &new_req_vec, existing_requirements->buf + ); + ggl_byte_vec_chain_push(&ret, &new_req_vec, ' '); + ggl_byte_vec_chain_append( + &ret, &new_req_vec, dep_version_requirement->buf + ); + if (ret != GGL_ERR_OK) { + GGL_LOGE("Failed to create new requirements for " + "dependency version."); + return ret; + } + + uint8_t *new_req = GGL_ALLOCN( + &version_requirements_balloc.alloc, + uint8_t, + new_req_vec.buf.len + ); + if (new_req == NULL) { + GGL_LOGE("Ran out of memory while trying to create " + "new requirements"); + return GGL_ERR_NOMEM; + } + + memcpy( + new_req, new_req_vec.buf.data, new_req_vec.buf.len + ); + *existing_requirements = GGL_OBJ_BUF((GglBuffer + ) { .data = new_req, .len = new_req_vec.buf.len }); + } + + // If we haven't resolved it yet, and it doesn't have an + // existing requirement, add it. + if (!existing_requirements) { + GglBuffer name_key_buf; + ret = ggl_buf_clone( + dependency->key, alloc, &name_key_buf + ); + if (ret != GGL_ERR_OK) { + return ret; + } + + GglBuffer vers_key_buf; + ret = ggl_buf_clone( + dep_version_requirement->buf, + &version_requirements_balloc.alloc, + &vers_key_buf + ); + if (ret != GGL_ERR_OK) { + return ret; + } + + ret = ggl_kv_vec_push( + &components_to_resolve, + (GglKV) { name_key_buf, GGL_OBJ_BUF(vers_key_buf) } + ); + if (ret != GGL_ERR_OK) { + return ret; + } + } + } + } + } + } + return GGL_ERR_OK; +} diff --git a/ggdeploymentd/src/dependency_resolver.h b/ggdeploymentd/src/dependency_resolver.h new file mode 100644 index 000000000..f032c7f37 --- /dev/null +++ b/ggdeploymentd/src/dependency_resolver.h @@ -0,0 +1,23 @@ +// aws-greengrass-lite - AWS IoT Greengrass runtime for constrained devices +// Copyright Amazon.com, Inc. or its affiliates. All Rights Reserved. +// SPDX-License-Identifier: Apache-2.0 + +#ifndef GGDEPLOYMENTD_DEPENDENCY_RESOLVER_H +#define GGDEPLOYMENTD_DEPENDENCY_RESOLVER_H + +#include "deployment_handler.h" +#include +#include +#include +#include +#include + +GglError resolve_dependencies( + GglMap *root_components, + GglBuffer thing_group_name, + GglDeploymentHandlerThreadArgs *args, + GglAlloc *alloc, + GglKVVec *resolved_components_kv_vec +); + +#endif diff --git a/ggdeploymentd/src/deployment_configuration.c b/ggdeploymentd/src/deployment_configuration.c new file mode 100644 index 000000000..60f615c48 --- /dev/null +++ b/ggdeploymentd/src/deployment_configuration.c @@ -0,0 +1,232 @@ +// aws-greengrass-lite - AWS IoT Greengrass runtime for constrained devices +// Copyright Amazon.com, Inc. or its affiliates. All Rights Reserved. +// SPDX-License-Identifier: Apache-2.0 + +#include "deployment_configuration.h" +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include + +DeploymentConfiguration config; + +GglError get_data_endpoint(GglByteVec *endpoint) { + uint8_t resp_mem[128] = { 0 }; + GglBuffer resp = GGL_BUF(resp_mem); + + GglError ret = ggl_gg_config_read_str( + GGL_BUF_LIST( + GGL_STR("services"), + GGL_STR("aws.greengrass.Nucleus-Lite"), + GGL_STR("configuration"), + GGL_STR("iotDataEndpoint") + ), + &resp + ); + + if (ret != GGL_ERR_OK) { + GGL_LOGW("Failed to get dataplane endpoint from config."); + return ret; + } + + return ggl_byte_vec_append(endpoint, resp); +} + +GglError get_data_port(GglByteVec *port) { + uint8_t resp_mem[16] = { 0 }; + GglBuffer resp = GGL_BUF(resp_mem); + + GglError ret = ggl_gg_config_read_str( + GGL_BUF_LIST( + GGL_STR("services"), + GGL_STR("aws.greengrass.Nucleus-Lite"), + GGL_STR("configuration"), + GGL_STR("greengrassDataPlanePort") + ), + &resp + ); + + if (ret != GGL_ERR_OK) { + GGL_LOGW("Failed to get dataplane port from config."); + return ret; + } + + return ggl_byte_vec_append(port, resp); +} + +GglError get_region(GglByteVec *region) { + uint8_t resp_mem[30] = { 0 }; + GglBuffer resp = GGL_BUF(resp_mem); + + GglError ret = ggl_gg_config_read_str( + GGL_BUF_LIST( + GGL_STR("services"), + GGL_STR("aws.greengrass.Nucleus-Lite"), + GGL_STR("configuration"), + GGL_STR("awsRegion") + ), + &resp + ); + if (ret != GGL_ERR_OK) { + GGL_LOGW("Failed to get region from config."); + return ret; + } + + ggl_byte_vec_append(region, resp); + return ret; +} + +GglError get_thing_name(GglBuffer *thing_name) { + uint8_t resp_mem[128] = { 0 }; + GglBuffer resp = GGL_BUF(resp_mem); + + GglError ret = ggl_gg_config_read_str( + GGL_BUF_LIST(GGL_STR("system"), GGL_STR("thingName")), &resp + ); + if (ret != GGL_ERR_OK) { + GGL_LOGW("Failed to get thing name from config."); + return ret; + } + + if (thing_name->len < resp.len) { + assert(false); + return GGL_ERR_FAILURE; + } + memcpy(thing_name->data, resp.data, resp.len); + thing_name->len = resp.len; + return GGL_ERR_OK; +} + +GglError get_root_ca_path(GglBuffer *root_ca_path) { + uint8_t resp_mem[PATH_MAX] = { 0 }; + GglBuffer resp = GGL_BUF(resp_mem); + + GglError ret = ggl_gg_config_read_str( + GGL_BUF_LIST(GGL_STR("system"), GGL_STR("rootCaPath")), &resp + ); + if (ret != GGL_ERR_OK) { + GGL_LOGW("Failed to get rootCaPath from config."); + return ret; + } + + if (root_ca_path->len < resp.len) { + assert(false); + return GGL_ERR_FAILURE; + } + memcpy(root_ca_path->data, resp.data, resp.len); + root_ca_path->len = resp.len; + return GGL_ERR_OK; +} + +GglError get_tes_cred_url(GglBuffer *tes_cred_url) { + uint8_t resp_mem[128] = { 0 }; + GglBuffer resp = GGL_BUF(resp_mem); + + GglError ret = ggl_gg_config_read_str( + GGL_BUF_LIST( + GGL_STR("services"), + GGL_STR("aws.greengrass.Nucleus-Lite"), + GGL_STR("configuration"), + GGL_STR("tesCredUrl") + ), + &resp + ); + if (ret != GGL_ERR_OK) { + GGL_LOGW("Failed to get tesCredUrl from config."); + return ret; + } + + if (tes_cred_url->len < resp.len) { + assert(false); + return GGL_ERR_FAILURE; + } + memcpy(tes_cred_url->data, resp.data, resp.len); + tes_cred_url->len = resp.len; + return GGL_ERR_OK; +} + +GglError get_posix_user(GglBuffer *posix_user) { + uint8_t resp_mem[128] = { 0 }; + GglBuffer resp = GGL_BUF(resp_mem); + + GglError ret = ggl_gg_config_read_str( + GGL_BUF_LIST( + GGL_STR("services"), + GGL_STR("aws.greengrass.Nucleus-Lite"), + GGL_STR("configuration"), + GGL_STR("runWithDefault"), + GGL_STR("posixUser") + ), + &resp + ); + if (ret != GGL_ERR_OK) { + GGL_LOGW("Failed to get posixUser from config."); + return ret; + } + + if (posix_user->len < resp.len) { + assert(false); + return GGL_ERR_FAILURE; + } + memcpy(posix_user->data, resp.data, resp.len); + posix_user->len = resp.len; + return GGL_ERR_OK; +} + +GglError get_private_key_path(GglByteVec *pkey_path) { + uint8_t resp_mem[PATH_MAX] = { 0 }; + GglBuffer resp = GGL_BUF(resp_mem); + + GglError ret = ggl_gg_config_read_str( + GGL_BUF_LIST(GGL_STR("system"), GGL_STR("privateKeyPath")), &resp + ); + + if (ret != GGL_ERR_OK) { + GGL_LOGW("Failed to get private key path from config."); + return ret; + } + + ggl_byte_vec_append(pkey_path, resp); + return ret; +} + +GglError get_cert_path(GglByteVec *cert_path) { + uint8_t resp_mem[PATH_MAX] = { 0 }; + GglBuffer resp = GGL_BUF(resp_mem); + + GglError ret = ggl_gg_config_read_str( + GGL_BUF_LIST(GGL_STR("system"), GGL_STR("certificateFilePath")), &resp + ); + + if (ret != GGL_ERR_OK) { + GGL_LOGW("Failed to get certificate path from config."); + return ret; + } + + ggl_byte_vec_append(cert_path, resp); + return ret; +} + +GglError get_rootca_path(GglByteVec *rootca_path) { + uint8_t resp_mem[PATH_MAX] = { 0 }; + GglBuffer resp = GGL_BUF(resp_mem); + + GglError ret = ggl_gg_config_read_str( + GGL_BUF_LIST(GGL_STR("system"), GGL_STR("rootCaPath")), &resp + ); + + if (ret != GGL_ERR_OK) { + GGL_LOGW("Failed to get rootca path from config."); + return ret; + } + + ggl_byte_vec_append(rootca_path, resp); + return ret; +} diff --git a/ggdeploymentd/src/deployment_configuration.h b/ggdeploymentd/src/deployment_configuration.h new file mode 100644 index 000000000..42dc9a8f1 --- /dev/null +++ b/ggdeploymentd/src/deployment_configuration.h @@ -0,0 +1,34 @@ +// aws-greengrass-lite - AWS IoT Greengrass runtime for constrained devices +// Copyright Amazon.com, Inc. or its affiliates. All Rights Reserved. +// SPDX-License-Identifier: Apache-2.0 + +#ifndef GGDEPLOYMENTD_DEPLOYMENT_CONFIGURATION_H +#define GGDEPLOYMENTD_DEPLOYMENT_CONFIGURATION_H + +#include +#include +#include + +typedef struct { + char data_endpoint[128]; + char cert_path[128]; + char rootca_path[128]; + char pkey_path[128]; + char region[24]; + char port[16]; +} DeploymentConfiguration; + +extern DeploymentConfiguration config; + +GglError get_data_endpoint(GglByteVec *endpoint); +GglError get_data_port(GglByteVec *port); +GglError get_region(GglByteVec *region); +GglError get_thing_name(GglBuffer *thing_name); +GglError get_root_ca_path(GglBuffer *root_ca_path); +GglError get_tes_cred_url(GglBuffer *tes_cred_url); +GglError get_posix_user(GglBuffer *posix_user); +GglError get_private_key_path(GglByteVec *pkey_path); +GglError get_cert_path(GglByteVec *cert_path); +GglError get_rootca_path(GglByteVec *rootca_path); + +#endif diff --git a/ggdeploymentd/src/deployment_handler.c b/ggdeploymentd/src/deployment_handler.c index 383fe75ed..1cf1ab905 100644 --- a/ggdeploymentd/src/deployment_handler.c +++ b/ggdeploymentd/src/deployment_handler.c @@ -3,7 +3,8 @@ // SPDX-License-Identifier: Apache-2.0 #include "deployment_handler.h" -#include "component_manager.h" +#include "dependency_resolver.h" +#include "deployment_configuration.h" #include "deployment_model.h" #include "deployment_queue.h" #include "iot_jobs_listener.h" @@ -11,7 +12,6 @@ #include #include #include -#include #include #include #include @@ -31,7 +31,6 @@ #include #include #include -#include #include #include #include @@ -46,15 +45,6 @@ #define MAX_DECODE_BUF_LEN 4096 #define MAX_COMP_NAME_BUF_SIZE 10000 -static struct DeploymentConfiguration { - char data_endpoint[128]; - char cert_path[128]; - char rootca_path[128]; - char pkey_path[128]; - char region[24]; - char port[16]; -} config; - typedef struct TesCredentials { GglBuffer aws_region; GglBuffer access_key_id; @@ -92,289 +82,6 @@ static GglError merge_dir_to(GglBuffer source, char *dir) { return ggl_process_call(cp); } -static GglError get_thing_name(char **thing_name) { - static uint8_t resp_mem[129] = { 0 }; - GglBuffer resp = GGL_BUF(resp_mem); - resp.len -= 1; - - GglError ret = ggl_gg_config_read_str( - GGL_BUF_LIST(GGL_STR("system"), GGL_STR("thingName")), &resp - ); - if (ret != GGL_ERR_OK) { - GGL_LOGW("Failed to get thing name from config."); - return ret; - } - resp.data[resp.len] = '\0'; - - *thing_name = (char *) resp.data; - return GGL_ERR_OK; -} - -static GglError get_region(GglByteVec *region) { - static uint8_t resp_mem[128] = { 0 }; - GglBuffer resp = GGL_BUF(resp_mem); - - GglError ret = ggl_gg_config_read_str( - GGL_BUF_LIST( - GGL_STR("services"), - GGL_STR("aws.greengrass.Nucleus-Lite"), - GGL_STR("configuration"), - GGL_STR("awsRegion") - ), - &resp - ); - if (ret != GGL_ERR_OK) { - GGL_LOGW("Failed to get region from config."); - return ret; - } - - ggl_byte_vec_chain_append(&ret, region, resp); - ggl_byte_vec_chain_push(&ret, region, '\0'); - if (ret == GGL_ERR_OK) { - region->buf.len--; - } - return ret; -} - -static GglError get_root_ca_path(char **root_ca_path) { - static uint8_t resp_mem[129] = { 0 }; - GglBuffer resp = GGL_BUF(resp_mem); - resp.len -= 1; - - GglError ret = ggl_gg_config_read_str( - GGL_BUF_LIST(GGL_STR("system"), GGL_STR("rootCaPath")), &resp - ); - if (ret != GGL_ERR_OK) { - GGL_LOGW("Failed to get rootCaPath from config."); - return ret; - } - resp.data[resp.len] = '\0'; - - *root_ca_path = (char *) resp.data; - return GGL_ERR_OK; -} - -static GglError get_tes_cred_url(char **tes_cred_url) { - static uint8_t resp_mem[129] = { 0 }; - GglBuffer resp = GGL_BUF(resp_mem); - resp.len -= 1; - - GglError ret = ggl_gg_config_read_str( - GGL_BUF_LIST( - GGL_STR("services"), - GGL_STR("aws.greengrass.Nucleus-Lite"), - GGL_STR("configuration"), - GGL_STR("tesCredUrl") - ), - &resp - ); - if (ret != GGL_ERR_OK) { - GGL_LOGW("Failed to get tesCredUrl from config."); - return ret; - } - resp.data[resp.len] = '\0'; - - *tes_cred_url = (char *) resp.data; - return GGL_ERR_OK; -} - -static GglError get_posix_user(char **posix_user) { - static uint8_t resp_mem[129] = { 0 }; - GglBuffer resp = GGL_BUF(resp_mem); - resp.len -= 1; - - GglError ret = ggl_gg_config_read_str( - GGL_BUF_LIST( - GGL_STR("services"), - GGL_STR("aws.greengrass.Nucleus-Lite"), - GGL_STR("configuration"), - GGL_STR("runWithDefault"), - GGL_STR("posixUser") - ), - &resp - ); - if (ret != GGL_ERR_OK) { - GGL_LOGW("Failed to get posixUser from config."); - return ret; - } - resp.data[resp.len] = '\0'; - - *posix_user = (char *) resp.data; - return GGL_ERR_OK; -} - -static GglError get_data_endpoint(GglByteVec *endpoint) { - GglMap params - = GGL_MAP({ GGL_STR("key_path"), - GGL_OBJ_LIST(GGL_LIST( - GGL_OBJ_BUF(GGL_STR("services")), - GGL_OBJ_BUF(GGL_STR("aws.greengrass.Nucleus-Lite")), - GGL_OBJ_BUF(GGL_STR("configuration")), - GGL_OBJ_BUF(GGL_STR("iotDataEndpoint")) - )) }); - - static uint8_t resp_mem[128] = { 0 }; - GglBumpAlloc balloc - = ggl_bump_alloc_init((GglBuffer) { .data = resp_mem, .len = 127 }); - - GglObject resp; - GglError ret = ggl_call( - GGL_STR("gg_config"), - GGL_STR("read"), - params, - NULL, - &balloc.alloc, - &resp - ); - if (ret != GGL_ERR_OK) { - GGL_LOGW("Failed to get dataplane endpoint from config."); - return ret; - } - if (resp.type != GGL_TYPE_BUF) { - GGL_LOGE("Configuration dataplane endpoint is not a string."); - return GGL_ERR_INVALID; - } - - return ggl_byte_vec_append(endpoint, resp.buf); -} - -static GglError get_data_port(GglByteVec *port) { - GglMap params - = GGL_MAP({ GGL_STR("key_path"), - GGL_OBJ_LIST(GGL_LIST( - GGL_OBJ_BUF(GGL_STR("services")), - GGL_OBJ_BUF(GGL_STR("aws.greengrass.Nucleus-Lite")), - GGL_OBJ_BUF(GGL_STR("configuration")), - GGL_OBJ_BUF(GGL_STR("greengrassDataPlanePort")) - )) }); - - static uint8_t resp_mem[128] = { 0 }; - GglBumpAlloc balloc - = ggl_bump_alloc_init((GglBuffer) { .data = resp_mem, .len = 127 }); - - GglObject resp; - GglError ret = ggl_call( - GGL_STR("gg_config"), - GGL_STR("read"), - params, - NULL, - &balloc.alloc, - &resp - ); - if (ret != GGL_ERR_OK) { - GGL_LOGW("Failed to get dataplane port from config."); - return ret; - } - if (resp.type != GGL_TYPE_BUF) { - GGL_LOGE("Configuration dataplane port is not a string."); - return GGL_ERR_INVALID; - } - - return ggl_byte_vec_append(port, resp.buf); -} - -static GglError get_private_key_path(GglByteVec *pkey_path) { - GglMap params = GGL_MAP({ GGL_STR("key_path"), - GGL_OBJ_LIST(GGL_LIST( - GGL_OBJ_BUF(GGL_STR("system")), - GGL_OBJ_BUF(GGL_STR("privateKeyPath")) - )) }); - - uint8_t resp_mem[128] = { 0 }; - GglBumpAlloc balloc - = ggl_bump_alloc_init((GglBuffer) { .data = resp_mem, .len = 127 }); - - GglObject resp; - GglError ret = ggl_call( - GGL_STR("gg_config"), - GGL_STR("read"), - params, - NULL, - &balloc.alloc, - &resp - ); - if (ret != GGL_ERR_OK) { - GGL_LOGW("Failed to get private key path from config."); - return ret; - } - if (resp.type != GGL_TYPE_BUF) { - GGL_LOGE("Configuration private key path is not a string."); - return GGL_ERR_INVALID; - } - - ggl_byte_vec_chain_append(&ret, pkey_path, resp.buf); - ggl_byte_vec_chain_push(&ret, pkey_path, '\0'); - return ret; -} - -static GglError get_cert_path(GglByteVec *cert_path) { - GglMap params = GGL_MAP({ GGL_STR("key_path"), - GGL_OBJ_LIST(GGL_LIST( - GGL_OBJ_BUF(GGL_STR("system")), - GGL_OBJ_BUF(GGL_STR("certificateFilePath")) - )) }); - - static uint8_t resp_mem[128] = { 0 }; - GglBumpAlloc balloc - = ggl_bump_alloc_init((GglBuffer) { .data = resp_mem, .len = 127 }); - - GglObject resp; - GglError ret = ggl_call( - GGL_STR("gg_config"), - GGL_STR("read"), - params, - NULL, - &balloc.alloc, - &resp - ); - if (ret != GGL_ERR_OK) { - GGL_LOGW("Failed to get certificate path from config."); - return ret; - } - if (resp.type != GGL_TYPE_BUF) { - GGL_LOGE("Configuration certificate path is not a string."); - return GGL_ERR_INVALID; - } - - ggl_byte_vec_chain_append(&ret, cert_path, resp.buf); - ggl_byte_vec_chain_push(&ret, cert_path, '\0'); - return ret; -} - -static GglError get_rootca_path(GglByteVec *rootca_path) { - GglMap params = GGL_MAP({ GGL_STR("key_path"), - GGL_OBJ_LIST(GGL_LIST( - GGL_OBJ_BUF(GGL_STR("system")), - GGL_OBJ_BUF(GGL_STR("rootCaPath")) - )) }); - - static uint8_t resp_mem[128] = { 0 }; - GglBumpAlloc balloc - = ggl_bump_alloc_init((GglBuffer) { .data = resp_mem, .len = 127 }); - - GglObject resp; - GglError ret = ggl_call( - GGL_STR("gg_config"), - GGL_STR("read"), - params, - NULL, - &balloc.alloc, - &resp - ); - if (ret != GGL_ERR_OK) { - GGL_LOGW("Failed to get rootca path from config."); - return ret; - } - if (resp.type != GGL_TYPE_BUF) { - GGL_LOGE("Configuration rootca path is not a string."); - return GGL_ERR_INVALID; - } - - ggl_byte_vec_chain_append(&ret, rootca_path, resp.buf); - ggl_byte_vec_chain_push(&ret, rootca_path, '\0'); - return ret; -} - static GglError get_tes_credentials(TesCredentials *tes_creds) { GglObject *aws_access_key_id = NULL; GglObject *aws_secret_access_key = NULL; @@ -745,905 +452,6 @@ static GglError get_recipe_artifacts( return GGL_ERR_OK; } -static GglError get_device_thing_groups(GglBuffer *response) { - GglByteVec data_endpoint = GGL_BYTE_VEC(config.data_endpoint); - GglError ret = get_data_endpoint(&data_endpoint); - if (ret != GGL_ERR_OK) { - GGL_LOGE("Failed to get dataplane endpoint."); - return ret; - } - - GglByteVec region = GGL_BYTE_VEC(config.region); - ret = get_region(®ion); - if (ret != GGL_ERR_OK) { - GGL_LOGE("Failed to get region."); - return ret; - } - - GglByteVec port = GGL_BYTE_VEC(config.port); - ret = get_data_port(&port); - if (ret != GGL_ERR_OK) { - GGL_LOGE("Failed to get dataplane port."); - return ret; - } - - GglByteVec pkey_path = GGL_BYTE_VEC(config.pkey_path); - ret = get_private_key_path(&pkey_path); - if (ret != GGL_ERR_OK) { - GGL_LOGE("Failed to get private key path."); - return ret; - } - - GglByteVec cert_path = GGL_BYTE_VEC(config.cert_path); - ret = get_cert_path(&cert_path); - if (ret != GGL_ERR_OK) { - GGL_LOGE("Failed to get certificate path."); - return ret; - } - - GglByteVec rootca_path = GGL_BYTE_VEC(config.rootca_path); - ret = get_rootca_path(&rootca_path); - if (ret != GGL_ERR_OK) { - GGL_LOGE("Failed to get certificate path."); - return ret; - } - - CertificateDetails cert_details - = { .gghttplib_cert_path = config.cert_path, - .gghttplib_root_ca_path = config.rootca_path, - .gghttplib_p_key_path = config.pkey_path }; - - char *thing_name = NULL; - ret = get_thing_name(&thing_name); - if (ret != GGL_ERR_OK) { - GGL_LOGE("Failed to get thing name."); - return ret; - } - - static uint8_t uri_path_buf[PATH_MAX]; - GglByteVec uri_path_vec = GGL_BYTE_VEC(uri_path_buf); - ret = ggl_byte_vec_append( - &uri_path_vec, GGL_STR("greengrass/v2/coreDevices/") - ); - ggl_byte_vec_chain_append( - &ret, &uri_path_vec, ggl_buffer_from_null_term(thing_name) - ); - ggl_byte_vec_chain_append(&ret, &uri_path_vec, GGL_STR("/thingGroups")); - if (ret != GGL_ERR_OK) { - GGL_LOGE("Failed to create thing groups call uri."); - return ret; - } - - ret = gg_dataplane_call( - data_endpoint.buf, - port.buf, - uri_path_vec.buf, - cert_details, - NULL, - response - ); - if (ret != GGL_ERR_OK) { - GGL_LOGE( - "The listThingGroupsForCoreDevice call failed with response %.*s.", - (int) response->len, - response->data - ); - return ret; - } - - GGL_LOGD( - "Received response from thingGroups dataplane call: %.*s", - (int) response->len, - response->data - ); - - return GGL_ERR_OK; -} - -static GglError generate_resolve_component_candidates_body( - GglBuffer component_name, - GglBuffer component_requirements, - GglByteVec *body_vec -) { - GglError byte_vec_ret = GGL_ERR_OK; - ggl_byte_vec_chain_append( - &byte_vec_ret, body_vec, GGL_STR("{\"componentCandidates\": [") - ); - - ggl_byte_vec_chain_append( - &byte_vec_ret, body_vec, GGL_STR("{\"componentName\": \"") - ); - ggl_byte_vec_chain_append(&byte_vec_ret, body_vec, component_name); - ggl_byte_vec_chain_append( - &byte_vec_ret, - body_vec, - GGL_STR("\",\"versionRequirements\": {\"requirements\": \"") - ); - ggl_byte_vec_chain_append(&byte_vec_ret, body_vec, component_requirements); - ggl_byte_vec_chain_append(&byte_vec_ret, body_vec, GGL_STR("\"}}")); - - // TODO: Include architecture requirements if any - ggl_byte_vec_chain_append( - &byte_vec_ret, - body_vec, - GGL_STR("],\"platform\": { \"attributes\": { \"os\" : \"linux\", " - "\"runtime\" : \"aws_nucleus_lite\" " - "},\"name\": \"linux\"}}") - ); - ggl_byte_vec_chain_push(&byte_vec_ret, body_vec, '\0'); - - GGL_LOGD("Body for call: %s", body_vec->buf.data); - - return GGL_ERR_OK; -} - -static GglError resolve_component_with_cloud( - GglBuffer component_name, - GglBuffer version_requirements, - GglBuffer *response -) { - static char resolve_candidates_body_buf[2048]; - GglByteVec body_vec = GGL_BYTE_VEC(resolve_candidates_body_buf); - GglError ret = generate_resolve_component_candidates_body( - component_name, version_requirements, &body_vec - ); - if (ret != GGL_ERR_OK) { - GGL_LOGE("Failed to generate body for resolveComponentCandidates call"); - return ret; - } - - GglByteVec data_endpoint = GGL_BYTE_VEC(config.data_endpoint); - ret = get_data_endpoint(&data_endpoint); - if (ret != GGL_ERR_OK) { - GGL_LOGE("Failed to get dataplane endpoint."); - return ret; - } - - GglByteVec region = GGL_BYTE_VEC(config.region); - ret = get_region(®ion); - if (ret != GGL_ERR_OK) { - GGL_LOGE("Failed to get region."); - return ret; - } - - GglByteVec port = GGL_BYTE_VEC(config.port); - ret = get_data_port(&port); - if (ret != GGL_ERR_OK) { - GGL_LOGE("Failed to get dataplane port."); - return ret; - } - - GglByteVec pkey_path = GGL_BYTE_VEC(config.pkey_path); - ret = get_private_key_path(&pkey_path); - if (ret != GGL_ERR_OK) { - GGL_LOGE("Failed to get private key path."); - return ret; - } - - GglByteVec cert_path = GGL_BYTE_VEC(config.cert_path); - ret = get_cert_path(&cert_path); - if (ret != GGL_ERR_OK) { - GGL_LOGE("Failed to get certificate path."); - return ret; - } - - GglByteVec rootca_path = GGL_BYTE_VEC(config.rootca_path); - ret = get_rootca_path(&rootca_path); - if (ret != GGL_ERR_OK) { - GGL_LOGE("Failed to get certificate path."); - return ret; - } - - CertificateDetails cert_details - = { .gghttplib_cert_path = config.cert_path, - .gghttplib_root_ca_path = config.rootca_path, - .gghttplib_p_key_path = config.pkey_path }; - - ret = gg_dataplane_call( - data_endpoint.buf, - port.buf, - GGL_STR("greengrass/v2/resolveComponentCandidates"), - cert_details, - resolve_candidates_body_buf, - response - ); - if (ret != GGL_ERR_OK) { - GGL_LOGE( - "Cloud resolution for the component failed with response %.*s.", - (int) response->len, - response->data - ); - return ret; - } - - GGL_LOGD( - "Received response from resolveComponentCandidates: %.*s", - (int) response->len, - response->data - ); - - return GGL_ERR_OK; -} - -static GglError parse_dataplane_response_and_save_recipe( - GglBuffer dataplane_response, - GglDeploymentHandlerThreadArgs *args, - GglBuffer *cloud_version -) { - GglObject json_candidates_response_obj; - // TODO: Figure out a better size. This response can be big. - uint8_t candidates_response_mem[100 * sizeof(GglObject)]; - GglBumpAlloc balloc = ggl_bump_alloc_init(GGL_BUF(candidates_response_mem)); - GglError ret = ggl_json_decode_destructive( - dataplane_response, &balloc.alloc, &json_candidates_response_obj - ); - if (ret != GGL_ERR_OK) { - GGL_LOGE("Error when parsing resolveComponentCandidates response to " - "json."); - return ret; - } - - if (json_candidates_response_obj.type != GGL_TYPE_MAP) { - GGL_LOGE("resolveComponentCandidates response did not parse into a " - "map."); - return ret; - } - - GglObject *resolved_component_versions; - if (!ggl_map_get( - json_candidates_response_obj.map, - GGL_STR("resolvedComponentVersions"), - &resolved_component_versions - )) { - GGL_LOGE("Missing resolvedComponentVersions."); - return ret; - } - if (resolved_component_versions->type != GGL_TYPE_LIST) { - GGL_LOGE("resolvedComponentVersions response is not a list."); - return ret; - } - - bool first_component = true; - GGL_LIST_FOREACH(resolved_version, resolved_component_versions->list) { - if (!first_component) { - GGL_LOGE( - "resolveComponentCandidates returned information for more than " - "one component." - ); - return GGL_ERR_INVALID; - } - first_component = false; - - if (resolved_version->type != GGL_TYPE_MAP) { - GGL_LOGE("Resolved version is not of type map."); - return ret; - } - - GglObject *cloud_component_arn; - GglObject *cloud_component_name; - GglObject *cloud_component_version; - GglObject *vendor_guidance; - GglObject *recipe_file_content; - - ret = ggl_map_validate( - resolved_version->map, - GGL_MAP_SCHEMA( - { GGL_STR("arn"), true, GGL_TYPE_BUF, &cloud_component_arn }, - { GGL_STR("componentName"), - true, - GGL_TYPE_BUF, - &cloud_component_name }, - { GGL_STR("componentVersion"), - true, - GGL_TYPE_BUF, - &cloud_component_version }, - { GGL_STR("vendorGuidance"), - false, - GGL_TYPE_BUF, - &vendor_guidance }, - { GGL_STR("recipe"), true, GGL_TYPE_BUF, &recipe_file_content }, - ) - ); - if (ret != GGL_ERR_OK) { - return ret; - } - assert(cloud_component_version->buf.len <= NAME_MAX); - - memcpy( - cloud_version->data, - cloud_component_version->buf.data, - cloud_component_version->buf.len - ); - cloud_version->len = cloud_component_version->buf.len; - - if (vendor_guidance != NULL) { - if (ggl_buffer_eq(vendor_guidance->buf, GGL_STR("DISCONTINUED"))) { - GGL_LOGW("The component version has been discontinued by " - "its " - "publisher. You can deploy this component version, " - "but " - "we recommend that you use a different version of " - "this " - "component"); - } - } - - if (recipe_file_content->buf.len == 0) { - GGL_LOGE("Recipe is empty."); - } - - ggl_base64_decode_in_place(&recipe_file_content->buf); - recipe_file_content->buf.data[recipe_file_content->buf.len] = '\0'; - - GGL_LOGD( - "Decoded recipe data as: %.*s", - (int) recipe_file_content->buf.len, - recipe_file_content->buf.data - ); - - static uint8_t recipe_name_buf[PATH_MAX]; - GglByteVec recipe_name_vec = GGL_BYTE_VEC(recipe_name_buf); - ret = ggl_byte_vec_append(&recipe_name_vec, cloud_component_name->buf); - ggl_byte_vec_chain_append(&ret, &recipe_name_vec, GGL_STR("-")); - ggl_byte_vec_chain_append( - &ret, &recipe_name_vec, cloud_component_version->buf - ); - // TODO: Actual support for .json files. We're writing a .json - // to a .yaml and relying on yaml being an almost-superset of - // json. - ggl_byte_vec_chain_append(&ret, &recipe_name_vec, GGL_STR(".yaml")); - if (ret != GGL_ERR_OK) { - GGL_LOGE("Failed to create recipe file name."); - return ret; - } - - static uint8_t recipe_dir_buf[PATH_MAX]; - GglByteVec recipe_dir_vec = GGL_BYTE_VEC(recipe_dir_buf); - ret = ggl_byte_vec_append( - &recipe_dir_vec, - ggl_buffer_from_null_term((char *) args->root_path.data) - ); - ggl_byte_vec_chain_append( - &ret, &recipe_dir_vec, GGL_STR("/packages/recipes/") - ); - if (ret != GGL_ERR_OK) { - GGL_LOGE("Failed to create recipe directory name."); - return ret; - } - - // Write file - int root_dir_fd = -1; - ret = ggl_dir_open(recipe_dir_vec.buf, O_PATH, true, &root_dir_fd); - if (ret != GGL_ERR_OK) { - GGL_LOGE("Failed to open dir when writing cloud recipe."); - return ret; - } - - int fd = -1; - ret = ggl_file_openat( - root_dir_fd, - recipe_name_vec.buf, - O_CREAT | O_WRONLY | O_TRUNC, - (mode_t) 0644, - &fd - ); - if (ret != GGL_ERR_OK) { - GGL_LOGE("Failed to open file at the dir when writing cloud " - "recipe."); - return ret; - } - - ret = ggl_file_write(fd, recipe_file_content->buf); - if (ret != GGL_ERR_OK) { - GGL_LOGE("Write to cloud recipe file failed"); - return ret; - } - - GGL_LOGD("Saved recipe under the name %s", recipe_name_vec.buf.data); - - ret = ggl_gg_config_write( - GGL_BUF_LIST(GGL_STR("services"), cloud_component_name->buf, ), - GGL_OBJ_MAP(GGL_MAP({ GGL_STR("arn"), *cloud_component_arn })), - 1 - ); - if (ret != GGL_ERR_OK) { - GGL_LOGE("Write of arn to config failed"); - return ret; - } - } - - return GGL_ERR_OK; -} - -static GglError resolve_dependencies( - GglMap root_components, - GglBuffer thing_group_name, - GglDeploymentHandlerThreadArgs *args, - GglAlloc *alloc, - GglKVVec *resolved_components_kv_vec -) { - assert(root_components.len != 0); - - GglError ret; - - // TODO: Decide on size - GglKVVec components_to_resolve = GGL_KV_VEC((GglKV[64]) { 0 }); - - static uint8_t version_requirements_mem[2048] = { 0 }; - GglBumpAlloc version_requirements_balloc - = ggl_bump_alloc_init(GGL_BUF(version_requirements_mem)); - - // Root components from current deployment - // TODO: Add current deployment's thing group to components map to config - GGL_MAP_FOREACH(pair, root_components) { - if (pair->val.type != GGL_TYPE_MAP) { - GGL_LOGE("Incorrect formatting for cloud deployment components " - "field."); - return GGL_ERR_INVALID; - } - - GglObject *val; - GglBuffer component_version = { 0 }; - if (ggl_map_get(pair->val.map, GGL_STR("version"), &val)) { - if (val->type != GGL_TYPE_BUF) { - GGL_LOGE("Received invalid argument."); - return GGL_ERR_INVALID; - } - component_version = val->buf; - } - - ret = ggl_kv_vec_push( - &components_to_resolve, - (GglKV) { pair->key, GGL_OBJ_BUF(component_version) } - ); - if (ret != GGL_ERR_OK) { - return ret; - } - } - - // At this point, components_to_resolve should be only a map of root - // component names to their version requirements from the deployment. - ret = ggl_gg_config_write( - GGL_BUF_LIST( - GGL_STR("services"), - GGL_STR("DeploymentService"), - GGL_STR("thingGroupsToRootComponents"), - thing_group_name - ), - GGL_OBJ_MAP(components_to_resolve.map), - 0 - ); - - if (ret != GGL_ERR_OK) { - GGL_LOGE( - "Failed to write thing group to root components map to ggconfigd." - ); - return ret; - } - - // Get list of thing groups - static uint8_t list_thing_groups_response_buf[2048] = { 0 }; - GglBuffer list_thing_groups_response - = GGL_BUF(list_thing_groups_response_buf); - - ret = get_device_thing_groups(&list_thing_groups_response); - if (ret != GGL_ERR_OK) { - return ret; - } - - // TODO: Add a schema and only parse the fields we need to save memory - GglObject json_thing_groups_object; - uint8_t thing_groups_response_mem[100 * sizeof(GglObject)]; - GglBumpAlloc thing_groups_json_balloc - = ggl_bump_alloc_init(GGL_BUF(thing_groups_response_mem)); - ret = ggl_json_decode_destructive( - list_thing_groups_response, - &thing_groups_json_balloc.alloc, - &json_thing_groups_object - ); - if (ret != GGL_ERR_OK) { - GGL_LOGE("Error when parsing listThingGroups response to " - "json."); - return ret; - } - - if (json_thing_groups_object.type != GGL_TYPE_MAP) { - GGL_LOGE("listThingGroups response did not parse into a " - "map."); - return ret; - } - - GglObject *thing_groups_list; - if (!ggl_map_get( - json_thing_groups_object.map, - GGL_STR("thingGroups"), - &thing_groups_list - )) { - GGL_LOGE("Missing thingGroups."); - return ret; - } - if (thing_groups_list->type != GGL_TYPE_LIST) { - GGL_LOGE("thingGroups response is not a list."); - return ret; - } - - // TODO: We want to also add root components from local deployments, not - // only thing group deployments. - GGL_LIST_FOREACH(thing_group_item, thing_groups_list->list) { - if (thing_group_item->type != GGL_TYPE_MAP) { - GGL_LOGE("Thing group item is not of type map."); - return ret; - } - - GglObject *thing_group_name_from_item; - - ret = ggl_map_validate( - thing_group_item->map, - GGL_MAP_SCHEMA( - { GGL_STR("thingGroupName"), - true, - GGL_TYPE_BUF, - &thing_group_name_from_item }, - ) - ); - if (ret != GGL_ERR_OK) { - return ret; - } - - if (!ggl_buffer_eq(thing_group_name_from_item->buf, thing_group_name)) { - GglObject group_root_components_read_value; - ret = ggl_gg_config_read( - GGL_BUF_LIST( - GGL_STR("services"), - GGL_STR("DeploymentService"), - GGL_STR("thingGroupsToRootComponents"), - thing_group_name_from_item->buf - ), - alloc, - &group_root_components_read_value - ); - if (ret != GGL_ERR_OK) { - GGL_LOGI( - "No info found in config for root components for thing " - "group %.*s, assuming no components are part of this thing " - "group.", - (int) thing_group_name_from_item->buf.len, - thing_group_name_from_item->buf.data - ); - } else { - if (group_root_components_read_value.type != GGL_TYPE_MAP) { - GGL_LOGE( - "Did not read a map from config for thing group to " - "root components map" - ); - return GGL_ERR_INVALID; - } - - GGL_MAP_FOREACH( - root_component_pair, group_root_components_read_value.map - ) { - // If component is already in the root component list, it - // must be the same version as the one already in the list - // or we have a conflict. - GglObject *existing_root_component_version; - ret = ggl_map_validate( - components_to_resolve.map, - GGL_MAP_SCHEMA( - { root_component_pair->key, - false, - GGL_TYPE_BUF, - &existing_root_component_version }, - ) - ); - if (ret != GGL_ERR_OK) { - return ret; - } - - bool need_to_add_root_component = true; - - if (existing_root_component_version != NULL) { - if (ggl_buffer_eq( - existing_root_component_version->buf, - root_component_pair->val.buf - )) { - need_to_add_root_component = false; - } else { - GGL_LOGE( - "There is a version conflict for component " - "%.*s, where two deployments are asking for " - "versions %.*s and %.*s. Please check that " - "this root component does not have conflicting " - "versions across your deployments.", - (int) root_component_pair->key.len, - root_component_pair->key.data, - (int) root_component_pair->val.buf.len, - root_component_pair->val.buf.data, - (int) existing_root_component_version->buf.len, - existing_root_component_version->buf.data - ); - return GGL_ERR_INVALID; - } - } - - if (need_to_add_root_component) { - GglBuffer root_component_name_buf; - ret = ggl_buf_clone( - root_component_pair->key, - alloc, - &root_component_name_buf - ); - if (ret != GGL_ERR_OK) { - return ret; - } - - GglBuffer root_component_version_buf; - ret = ggl_buf_clone( - root_component_pair->val.buf, - &version_requirements_balloc.alloc, - &root_component_version_buf - ); - if (ret != GGL_ERR_OK) { - return ret; - } - - ret = ggl_kv_vec_push( - &components_to_resolve, - (GglKV) { root_component_name_buf, - GGL_OBJ_BUF(root_component_version_buf) } - ); - GGL_LOGD( - "Added %.*s to the list of root components to " - "resolve " - "from " - "the thing group %.*s", - (int) root_component_name_buf.len, - root_component_name_buf.data, - (int) thing_group_name_from_item->buf.len, - thing_group_name_from_item->buf.data - ); - } - } - } - } - } - - GGL_MAP_FOREACH(pair, components_to_resolve.map) { - // We assume that we have not resolved a component yet if we are finding - // it in this map. - uint8_t resolved_version_arr[NAME_MAX]; - GglBuffer resolved_version = GGL_BUF(resolved_version_arr); - bool found_local_candidate = resolve_component_version( - pair->key, pair->val.buf, &resolved_version - ); - - if (!found_local_candidate) { - // Resolve with cloud and download recipe - static uint8_t resolve_component_candidates_response_buf[16384] - = { 0 }; - GglBuffer resolve_component_candidates_response - = GGL_BUF(resolve_component_candidates_response_buf); - - ret = resolve_component_with_cloud( - pair->key, pair->val.buf, &resolve_component_candidates_response - ); - if (ret != GGL_ERR_OK) { - return ret; - } - - bool is_empty_response = ggl_buffer_eq( - resolve_component_candidates_response, GGL_STR("{}") - ); - - if (is_empty_response) { - GGL_LOGI( - "Cloud version resolution failed for component %.*s.", - (int) pair->key.len, - pair->val.buf.data - ); - return GGL_ERR_FAILURE; - } - - ret = parse_dataplane_response_and_save_recipe( - resolve_component_candidates_response, args, &resolved_version - ); - if (ret != GGL_ERR_OK) { - return ret; - } - } - - // Add resolved component to list of resolved components - GglBuffer val_buf; - ret = ggl_buf_clone(resolved_version, alloc, &val_buf); - if (ret != GGL_ERR_OK) { - return ret; - } - - ret = ggl_kv_vec_push( - resolved_components_kv_vec, - (GglKV) { pair->key, GGL_OBJ_BUF(val_buf) } - ); - if (ret != GGL_ERR_OK) { - GGL_LOGE( - "Error while adding component to list of resolved component" - ); - return ret; - } - - // Find dependencies from recipe and add them to the list of components - // to resolve. If the dependency is for a component that is already - // resolved, verify that new requirements are satisfied and fail - // deployment if not. - - // Get actual recipe read - GglObject recipe_obj; - static uint8_t recipe_mem[8192] = { 0 }; - GglBumpAlloc balloc = ggl_bump_alloc_init(GGL_BUF(recipe_mem)); - ret = ggl_recipe_get_from_file( - args->root_path_fd, - pair->key, - resolved_version, - &balloc.alloc, - &recipe_obj - ); - if (ret != GGL_ERR_OK) { - return ret; - } - GglObject *component_dependencies = NULL; - - if (recipe_obj.type != GGL_TYPE_MAP) { - GGL_LOGE("Recipe object did not parse into a map."); - return GGL_ERR_INVALID; - } - - ret = ggl_map_validate( - recipe_obj.map, - GGL_MAP_SCHEMA( - { GGL_STR("ComponentDependencies"), - false, - GGL_TYPE_MAP, - &component_dependencies }, - ) - ); - if (ret != GGL_ERR_OK) { - return ret; - } - if (component_dependencies) { - GGL_MAP_FOREACH(dependency, component_dependencies->map) { - if (dependency->val.type != GGL_TYPE_MAP) { - GGL_LOGE( - "Component dependency in recipe does not have map data" - ); - return GGL_ERR_INVALID; - } - GglObject *dep_version_requirement = NULL; - ret = ggl_map_validate( - dependency->val.map, - GGL_MAP_SCHEMA( - { GGL_STR("VersionRequirement"), - true, - GGL_TYPE_BUF, - &dep_version_requirement }, - ) - ); - if (ret != GGL_ERR_OK) { - return ret; - } - - // If we already resolved the component version, check that it - // still satisfies the new requirement and fail otherwise. - GglObject *already_resolved_version; - ret = ggl_map_validate( - resolved_components_kv_vec->map, - GGL_MAP_SCHEMA( - { dependency->key, - false, - GGL_TYPE_BUF, - &already_resolved_version }, - ) - ); - if (ret != GGL_ERR_OK) { - return ret; - } - if (already_resolved_version) { - bool meets_requirements = is_in_range( - already_resolved_version->buf, - dep_version_requirement->buf - ); - if (!meets_requirements) { - GGL_LOGE("Already resolved component does not meet new " - "dependency requirement, failing dependency " - "resolution."); - return GGL_ERR_FAILURE; - } - } - - if (!already_resolved_version) { - // If we haven't resolved it yet, check if we have an - // existing requirement and append the new requirement if - // so. - GglObject *existing_requirements; - ret = ggl_map_validate( - components_to_resolve.map, - GGL_MAP_SCHEMA( - { dependency->key, - false, - GGL_TYPE_BUF, - &existing_requirements }, - ) - ); - if (ret != GGL_ERR_OK) { - return ret; - } - if (existing_requirements) { - uint8_t new_req_buf[PATH_MAX]; - GglByteVec new_req_vec = GGL_BYTE_VEC(new_req_buf); - ret = ggl_byte_vec_append( - &new_req_vec, existing_requirements->buf - ); - ggl_byte_vec_chain_push(&ret, &new_req_vec, ' '); - ggl_byte_vec_chain_append( - &ret, &new_req_vec, dep_version_requirement->buf - ); - if (ret != GGL_ERR_OK) { - GGL_LOGE("Failed to create new requirements for " - "dependency version."); - return ret; - } - - uint8_t *new_req = GGL_ALLOCN( - &version_requirements_balloc.alloc, - uint8_t, - new_req_vec.buf.len - ); - if (new_req == NULL) { - GGL_LOGE("Ran out of memory while trying to create " - "new requirements"); - return GGL_ERR_NOMEM; - } - - memcpy( - new_req, new_req_vec.buf.data, new_req_vec.buf.len - ); - *existing_requirements = GGL_OBJ_BUF((GglBuffer - ) { .data = new_req, .len = new_req_vec.buf.len }); - } - - // If we haven't resolved it yet, and it doesn't have an - // existing requirement, add it. - if (!existing_requirements) { - GglBuffer name_key_buf; - ret = ggl_buf_clone( - dependency->key, alloc, &name_key_buf - ); - if (ret != GGL_ERR_OK) { - return ret; - } - - GglBuffer vers_key_buf; - ret = ggl_buf_clone( - dep_version_requirement->buf, - &version_requirements_balloc.alloc, - &vers_key_buf - ); - if (ret != GGL_ERR_OK) { - return ret; - } - - ret = ggl_kv_vec_push( - &components_to_resolve, - (GglKV) { name_key_buf, GGL_OBJ_BUF(vers_key_buf) } - ); - if (ret != GGL_ERR_OK) { - return ret; - } - } - } - } - } - } - return GGL_ERR_OK; -} - static GglError open_component_artifacts_dir( int artifact_store_fd, GglBuffer component_name, @@ -1908,7 +716,7 @@ static void handle_deployment( GglBumpAlloc resolve_dependencies_balloc = ggl_bump_alloc_init(GGL_BUF(resolve_dependencies_mem)); GglError ret = resolve_dependencies( - deployment->cloud_root_components_to_add, + &deployment->cloud_root_components_to_add, deployment->thing_group, args, &resolve_dependencies_balloc.alloc, @@ -2043,54 +851,58 @@ static void handle_deployment( return; } - char *thing_name = NULL; + uint8_t thing_name_arr[128]; + GglBuffer thing_name = GGL_BUF(thing_name_arr); ret = get_thing_name(&thing_name); if (ret != GGL_ERR_OK) { GGL_LOGE("Failed to get thing name."); return; } - char *root_ca_path = NULL; + uint8_t root_ca_path_arr[PATH_MAX]; + GglBuffer root_ca_path = GGL_BUF(root_ca_path_arr); ret = get_root_ca_path(&root_ca_path); if (ret != GGL_ERR_OK) { GGL_LOGE("Failed to get rootCaPath."); return; } - char *tes_cred_url = NULL; + uint8_t tes_cred_arr[128]; + GglBuffer tes_cred_url = GGL_BUF(tes_cred_arr); ret = get_tes_cred_url(&tes_cred_url); if (ret != GGL_ERR_OK) { GGL_LOGE("Failed to get tes credentials url."); return; } - char *posix_user = NULL; + uint8_t posix_user_arr[128]; + GglBuffer posix_user = GGL_BUF(posix_user_arr); ret = get_posix_user(&posix_user); if (ret != GGL_ERR_OK) { GGL_LOGE("Failed to get posix_user."); return; } - if (strlen(posix_user) < 1) { + if (posix_user.len < 1) { GGL_LOGE("Run with default posix user is not set."); return; } bool colon_found = false; char *group; - for (size_t j = 0; j < strlen(posix_user); j++) { - if (posix_user[j] == ':') { - posix_user[j] = '\0'; + for (size_t j = 0; j < posix_user.len; j++) { + if (posix_user.data[j] == ':') { + posix_user.data[j] = '\0'; colon_found = true; - group = &posix_user[j + 1]; + group = (char *) &posix_user.data[j + 1]; break; } } if (!colon_found) { - group = posix_user; + group = (char *) posix_user.data; } static Recipe2UnitArgs recipe2unit_args; memset(&recipe2unit_args, 0, sizeof(Recipe2UnitArgs)); - recipe2unit_args.user = posix_user; + recipe2unit_args.user = (char *) posix_user.data; recipe2unit_args.group = group; recipe2unit_args.component_name = pair->key; @@ -2524,54 +1336,58 @@ static void handle_deployment( return; } - char *thing_name = NULL; + uint8_t thing_name_arr[128]; + GglBuffer thing_name = GGL_BUF(thing_name_arr); ret = get_thing_name(&thing_name); if (ret != GGL_ERR_OK) { GGL_LOGE("Failed to get thing name."); return; } - GglByteVec region = GGL_BYTE_VEC(config.region); - ret = get_region(®ion); + uint8_t root_ca_path_arr[PATH_MAX]; + GglBuffer root_ca_path = GGL_BUF(root_ca_path_arr); + ret = get_root_ca_path(&root_ca_path); if (ret != GGL_ERR_OK) { - GGL_LOGE("Failed to get region."); + GGL_LOGE("Failed to get rootCaPath."); return; } - char *root_ca_path = NULL; - ret = get_root_ca_path(&root_ca_path); + uint8_t tes_cred_arr[128]; + GglBuffer tes_cred_url = GGL_BUF(tes_cred_arr); + ret = get_tes_cred_url(&tes_cred_url); if (ret != GGL_ERR_OK) { - GGL_LOGE("Failed to get rootCaPath."); + GGL_LOGE("Failed to get tes credentials url."); return; } - char *posix_user = NULL; + uint8_t posix_user_arr[128]; + GglBuffer posix_user = GGL_BUF(posix_user_arr); ret = get_posix_user(&posix_user); if (ret != GGL_ERR_OK) { GGL_LOGE("Failed to get posix_user."); return; } - if (strlen(posix_user) < 1) { + if (posix_user.len < 1) { GGL_LOGE("Run with default posix user is not set."); return; } bool colon_found = false; char *group; - for (size_t i = 0; i < strlen(posix_user); i++) { - if (posix_user[i] == ':') { - posix_user[i] = '\0'; + for (size_t j = 0; j < posix_user.len; j++) { + if (posix_user.data[j] == ':') { + posix_user.data[j] = '\0'; colon_found = true; - group = &posix_user[i + 1]; + group = (char *) &posix_user.data[j + 1]; break; } } if (!colon_found) { - group = posix_user; + group = (char *) posix_user.data; } static Recipe2UnitArgs recipe2unit_args; memset(&recipe2unit_args, 0, sizeof(Recipe2UnitArgs)); - recipe2unit_args.user = posix_user; + recipe2unit_args.user = (char *) posix_user.data; recipe2unit_args.group = group; recipe2unit_args.component_name = pair->key;