# # Copyright (c) Contributors to the Open 3D Engine Project. # For complete copyright and license terms please see the LICENSE at the root of this distribution. # # SPDX-License-Identifier: Apache-2.0 OR MIT # include_guard() include(${LY_ROOT_FOLDER}/cmake/LySet.cmake) # OVERVIEW: # this is the Open 3D Engine Package system. # It allows you to host a package on a server and download it as needed when a target # requests that specific package, or manually whenever you want to do so. # Most users will just call ly_associate_package(...) to associate a package with a target # and the system will automatically get that package when a target asks for it as a # dependency. If you want to manually activate a package, you can use ly_download_associated_package # to bring its find* scripts into scope (and download if needs be) # and you can also use ly_set_package_download_location(..) to change it to install # somewhere besides the default install location. # cache variables: # LY_PACKAGE_SERVER_URLS: # missing packages are downloaded from this list of URLS, first one to match wins. # Besides normal http and https URLs, you can also use FILE urls, # for example "file:///mnt/d/lyengine/packageSource/packages;file:///d:/lyengine/packageSource/packages" # also allowed: # "s3://bucketname" (it will use LYPackage_S3Downloader.cmake to download it from a s3 bucket) set(LY_PACKAGE_SERVER_URLS "https://d3t6xeg4fgfoum.cloudfront.net" CACHE STRING "Server URLS to fetch packages from") # Note: if you define the "LY_PACKAGE_SERVER_URLS" environment variable # it will be added to this value in the front, so that users can set # an env var and use that as an "additional" set of servers beyond the default set. if (DEFINED ENV{LY_PACKAGE_SERVER_URLS}) set(LY_PACKAGE_SERVER_URLS $ENV{LY_PACKAGE_SERVER_URLS} ${LY_PACKAGE_SERVER_URLS}) endif() # If you keep packages after downloading, then they can be moved to a network share # or checked into source control so that others on the same project can avoid re-downloading set(LY_PACKAGE_KEEP_AFTER_DOWNLOADING TRUE CACHE BOOL "If enabled, packages will be kept after downloading them for later re-use") set(LY_PACKAGE_DOWNLOAD_CACHE_LOCATION @LY_3RDPARTY_PATH@/downloaded_packages CACHE PATH "Download location for packages (Defaults to @LY_3RDPARTY_PATH@/downloaded_packages)") if (DEFINED ENV{LY_PACKAGE_DOWNLOAD_CACHE_LOCATION}) set(LY_PACKAGE_DOWNLOAD_CACHE_LOCATION $ENV{LY_PACKAGE_DOWNLOAD_CACHE_LOCATION}) endif() string(CONFIGURE ${LY_PACKAGE_DOWNLOAD_CACHE_LOCATION} LY_PACKAGE_DOWNLOAD_CACHE_LOCATION @ONLY) # LY_PACKAGE_UNPACK_LOCATION - you can change this to any path reachable. set(LY_PACKAGE_UNPACK_LOCATION @LY_3RDPARTY_PATH@/packages CACHE PATH "Unpack location of downloaded packages (Defaults to @LY_3RDPARTY_PATH@/packages)") if (DEFINED ENV{LY_PACKAGE_UNPACK_LOCATION}) set(LY_PACKAGE_UNPACK_LOCATION $ENV{LY_PACKAGE_UNPACK_LOCATION}) endif() string(CONFIGURE ${LY_PACKAGE_UNPACK_LOCATION} LY_PACKAGE_UNPACK_LOCATION @ONLY) # while developing you can set one or both to true to force auto downloads from your local cache set(LY_PACKAGE_VALIDATE_CONTENTS FALSE CACHE BOOL "If enabled, will fully validate every file in every package based on the SHA256SUMS file from the package") set(LY_PACKAGE_VALIDATE_PACKAGE FALSE CACHE BOOL "If enabled, will validate that the downloaded package files hash matches the expected hash even if already downloaded and verified before.") # you can also enable verbose/debug logging from the package system. set(LY_PACKAGE_DEBUG FALSE CACHE BOOL "If enabled, will output detailed information during package operations" ) # ---- below this line, no cache variables or tweakables --------- ly_set(LY_PACKAGE_EXT ".tar.xz") ly_set(LY_PACKAGE_HASH_EXT ".tar.xz.SHA256SUMS") ly_set(LY_PACKAGE_CONTENT_HASH_EXT ".tar.xz.content.SHA256SUMS") set(LY_PACKAGE_DOWNLOAD_RETRY_COUNT 3 CACHE STRING "3") # accounts for it being undefined or blank if ("${LY_PACKAGE_DOWNLOAD_CACHE_LOCATION}" STREQUAL "") message(FATAL_ERROR "ly_package: LY_PACKAGE_DOWNLOAD_CACHE_LOCATION must be defined.") endif() # used to send messages and hide them unless the LY_PACKAGE_DEBUG var is true macro(ly_package_message) if (LY_PACKAGE_DEBUG) message(${ARGN}) endif() endmacro() include(${LY_ROOT_FOLDER}/cmake/LYPackage_S3Downloader.cmake) # Attempts one time to download a file. # sets should_retry to true if the caller should retry due to an intermittent problem # Do not call this function, call download_file instead. function(download_file_internal) set(_oneValueArgs URL TARGET_FILE EXPECTED_HASH RESULTS SHOULD_RETRY) cmake_parse_arguments(download_file_internal "" "${_oneValueArgs}" "" ${ARGN}) if(NOT download_file_internal_URL) message(FATAL_ERROR "no URL arg passed into download_file_internal, it is required.") endif() if(NOT download_file_internal_TARGET_FILE) message(FATAL_ERROR "no TARGET_FILE arg passed into download_file_internal, it is required.") endif() if(NOT download_file_internal_EXPECTED_HASH) message(FATAL_ERROR "no EXPECTED_HASH arg passed into download_file_internal, it is required.") endif() if(NOT download_file_internal_RESULTS) message(FATAL_ERROR "no RESULTS arg passed into download_file_internal, it is required.") endif() if(NOT download_file_internal_SHOULD_RETRY) message(FATAL_ERROR "no SHOULD_RETRY arg passed into download_file_internal, it is required.") endif() set(${download_file_internal_RESULTS} "-1;unknown_error" PARENT_SCOPE) unset(${download_file_internal_SHOULD_RETRY} PARENT_SCOPE) # note that below "results" is a local variable and download_file_internal_RESULTS will be set to it before exit. ly_is_s3_url(${download_file_internal_URL} result_is_s3_bucket) if (result_is_s3_bucket) ly_s3_download("${download_file_internal_URL}" ${download_file_internal_TARGET_FILE} results) else() file(DOWNLOAD "${download_file_internal_URL}" ${download_file_internal_TARGET_FILE} STATUS results TLS_VERIFY ON LOG logic) list(APPEND results ${log}) endif() list(GET results 0 code_returned) # sometimes, the server returns a non-error code but still the file is zero bytes. # in this case, we need to return a failure. We won't ever use a 0 byte file. if (EXISTS ${download_file_internal_TARGET_FILE}) file(SIZE ${download_file_internal_TARGET_FILE} target_size) if(target_size EQUAL 0) if(code_returned EQUAL 0) # the server said "OK" but gave us a bad file. Change this to not OK! ly_package_message("Server gave us a zero byte file but still said ${results}, retrying...") set(code_returned 22) set(results "22;\nThe requested URL returned error: 500 Internal Error - Zero byte file returned despite good exit code\n") set(${download_file_internal_SHOULD_RETRY} TRUE PARENT_SCOPE) endif() endif() else() # the file was not created. If the server still returned OK then we need to change this to not OK if (code_returned EQUAL 0) ly_package_message("Server gave us no file, but said ${results}, retrying...") set(code_returned 22) set(results "22;\nThe requested URL returned error: 500 Internal Error - Zero byte file returned despite good exit code\n") set(${download_file_internal_SHOULD_RETRY} TRUE PARENT_SCOPE) endif() endif() if(code_returned EQUAL 0) # code 0 means success, but we still need to hash the file. file(SHA256 ${download_file_internal_TARGET_FILE} hash_of_downloaded_file) if (NOT "${hash_of_downloaded_file}" STREQUAL "${download_file_internal_EXPECTED_HASH}" ) set(results "1;Downloaded successfully, but the file hash did not match expected hash!") set(code_returned 1) endif() endif() if(code_returned) # non zero means it failed to download # however, cmake will leave the file open, and zero bytes. file(REMOVE ${download_file_internal_TARGET_FILE}) # parse the error. If its 22, it means it was an HTTP error from curl. # we'll go to the bother of parsing the error and replacing it with the actual error code. # since the curl error is pointless. # note that the following code is similar to the code that's build into CMake # in its ExternalData.cmake file that ships. if (${code_returned} EQUAL 22) # extract the http response code if possible! string(REGEX MATCH "The requested URL returned error\\:([^\n]*)[\n]" found_string "${results}") if (found_string) # message the log before you replace it for debugging ly_package_message("${results}") # replace it. set(results ${code_returned} ${found_string}) # if we get here, 'found_string' contains the one line response code from the server # which takes the form of ' response code - desciption of response code' # for example, ' 404 - Not Found'. It will only contain this one line. # See if its a 500 or 503 code specifically, if so, we need to retry. if (found_string MATCHES "500" OR found_string MATCHES "503") ly_package_message("500 or 503 code returned from server, will retry...") set(${download_file_internal_SHOULD_RETRY} TRUE PARENT_SCOPE) endif() endif() endif() endif() set(${download_file_internal_RESULTS} ${results} PARENT_SCOPE) endfunction() # Downloads a file with the ability to retry. # uses download_file_internal to actually download the file. # currently works on file://, s3://, ftp://, http://, https:// ... and whatever # else the file(DOWNLOAD ..) function supports. # if you add another downloader, see that it returns a list, where the first element # in the list is the error code, and then the rest of the list is the error(s), just like # the file(DOWNLOAD ... ) function does. # sets should_retry to a non null value if you should retry this download. function(download_file) set(_oneValueArgs URL TARGET_FILE EXPECTED_HASH RESULTS) cmake_parse_arguments(download_file "" "${_oneValueArgs}" "" ${ARGN}) if(NOT download_file_URL) message(FATAL_ERROR "no URL arg passed into download_file, it is required.") endif() if(NOT download_file_TARGET_FILE) message(FATAL_ERROR "no TARGET_FILE arg passed into download_file, it is required.") endif() if(NOT download_file_EXPECTED_HASH) message(FATAL_ERROR "no EXPECTED_HASH arg passed into download_file, it is required.") endif() if(NOT download_file_RESULTS) message(FATAL_ERROR "no RESULTS arg passed into download_file, it is required.") endif() set(${download_file_RESULTS} "-1;unknown_error" PARENT_SCOPE) foreach(retry_count RANGE 0 ${LY_PACKAGE_DOWNLOAD_RETRY_COUNT}) download_file_internal( URL ${download_file_URL} TARGET_FILE ${download_file_TARGET_FILE} RESULTS results EXPECTED_HASH ${download_file_EXPECTED_HASH} SHOULD_RETRY should_retry) if (NOT should_retry) break() endif() ly_package_message("${retry_count} / ${LY_PACKAGE_DOWNLOAD_RETRY_COUNT} download retry attempts.") endforeach() set(${download_file_RESULTS} ${results} PARENT_SCOPE) endfunction() #! ly_package_internal_download_package - note, the list of 3rd party urls is a list! # given a package name it will loop over the servers in the list and try each one # until one has the file AND has the correct hash for that file. function(ly_package_internal_download_package package_name url_variable) unset(${url_variable} PARENT_SCOPE) unset(error_messages) # to avoid spamming with useless repeated warnings, we save # a global that indicates we already failed to find it if (NOT LY_PACKAGE_SERVER_URLS) message(SEND_ERROR "ly_package: - LY_PACKAGE_SERVER_URLS is empty, cannot download packages. enable LY_PACKAGE_DEBUG for details") return() endif() ly_get_package_expected_hash(${package_name} package_expected_hash) foreach(server_url ${LY_PACKAGE_SERVER_URLS}) set(download_url ${server_url}/${package_name}${LY_PACKAGE_EXT}) ly_package_get_target_cache(${package_name} package_download_cache_location) set(download_target ${package_download_cache_location}/${package_name}${LY_PACKAGE_EXT}) file(REMOVE ${download_target}) ly_package_message(STATUS "ly_package: trying to download ${download_url} to ${download_target}") ly_get_package_expected_hash(${package_name} expected_package_hash) download_file(URL ${download_url} TARGET_FILE ${download_target} EXPECTED_HASH ${expected_package_hash} RESULTS results) list(GET results 0 status_code) if (${status_code} EQUAL 0 AND EXISTS ${download_target}) set(${url_variable} ${server_url} PARENT_SCOPE) ly_package_message(STATUS "ly_package: - downloaded ${server_url} for package ${package_name}") return() else() # remove the status code and treat the rest of the list as the error. list(REMOVE_AT results 0) set(current_error_message "Error from server ${server_url} - ${status_code} - ${results}") #strip whitespace string(REGEX REPLACE "[ \t\r\n]$" "" current_error_message "${current_error_message}") list(APPEND error_messages "${current_error_message}") # we can't keep the file, sometimes it makes a zero-byte file! file(REMOVE ${download_target}) endif() endforeach() # note that we FATAL_ERROR here because otherwise, some of the packages we provide would fall through # and use unknown versions possibly present somewhere in the user's system - but if we wanted that to happen # we wouldn't have used a ly-package-association in the first place! Continuing from there would just cause # a cascade of errors even harder to track down. set(final_error_message "ly_package: - Unable to get package ${package_name} from any download server. Enable LY_PACKAGE_DEBUG to debug.") foreach(error_message ${error_messages}) set(final_error_message "${final_error_message}\n${error_message}") endforeach() message(FATAL_ERROR "${final_error_message}") endfunction() # parse_sha256sums_line # given a line of text that is in the SHA256SUMS format, digest it and output it # as a pair of variables of your choice. # will set output_hash to be the hex digest string # will set output_file_name to be the filename that must hash to that value # will unset them if an error occurs. function(parse_sha256sums_line input_hash_line output_hash output_file_name) # the official SHA256SUMS format is actually # the hash, followed by exactly one space character, followed # by either another space or an asterisk (indicating text or binary, space is text) # followed by the name of the file for the rest of the line (may contain more spaces) # the actual GNU implementations always binary hash, and so will we, so we'll ignore # the hash type character (text or binary) # this is why we capture three groups (hash, type, filename) # but only output 2 groups (1 and 3), hash and filename set(REGEX_EXPRESSION "^([A-Za-z0-9]*) ( |\\*)(.*)$") string(REGEX REPLACE "${REGEX_EXPRESSION}" "\\1;\\3" temp_list "${input_hash_line}") # hash list is now: expected_file_hash;file_name" list(LENGTH temp_list hash_length) if (NOT ${hash_length} EQUAL 2) unset(${output_hash} PARENT_SCOPE) unset(${output_file_name} PARENT_SCOPE) else() list(GET temp_list 0 temp_hash) list(GET temp_list 1 temp_filename) set(${output_hash} ${temp_hash} PARENT_SCOPE) set(${output_file_name} ${temp_filename} PARENT_SCOPE) endif() endfunction() # ly_validate_sha256sums_file -- internal function # given the path to a SHA256SUMS file and a working directory, # verifies the hashes based on the current settings. # --- sets HASH_WAS_VALID TRUE on parent scope if valid, FALSE otherwise. # Note that it does not currently check if extra files are present, only that # each file that is supposed to be there, is there and has the right hash. function(ly_validate_sha256sums_file working_directory path_to_sha256sums_file) set(HASH_WAS_VALID FALSE PARENT_SCOPE) if (NOT EXISTS ${path_to_sha256sums_file}) ly_package_message(STATUS "ly_package: Could not find SHA256SUMS file: ${path_to_sha256sums_file}") return() endif() # lines in a SHA256SUMS file take the form of "hash *filename" # that is, the actual 256hash in hex decimals, whitespace, then an asterisk # and then the file name that must have that hash. set(ANY_HASH_MISMATCHES FALSE) # we only try to do any kind of hashing if the VALIDATE_CONTENTS flag is on # otherwise we just check for the presence of required files. # note that VALIDATE_CONTENTS is forced to true for any package when we download # it the first time. Its only set to true after that if the user forces it to be enabled. file(STRINGS ${path_to_sha256sums_file} hash_data ENCODING UTF-8) foreach(hash_line IN ITEMS ${hash_data}) parse_sha256sums_line("${hash_line}" expected_file_hash file_name) if (NOT expected_file_hash OR NOT file_name) message(SEND_ERROR "ly_package: Invalid format SHA256SUMS file: ${path_to_sha256sums_file} line ${hash_line} - cannot verify hashes. Enable LY_PACKAGE_DEBUG to debug.") return() endif() if (EXISTS ${working_directory}/${file_name}) if (LY_PACKAGE_VALIDATE_CONTENTS) file(SHA256 ${working_directory}/${file_name} existing_hash) if (NOT "${existing_hash}" STREQUAL "${expected_file_hash}" ) ly_package_message(STATUS "ly_package: File hash mismatch: ${working_directory}/${file_name}") set(ANY_HASH_MISMATCHES TRUE) else() ly_package_message(STATUS "ly_package: File hash matches: ${expected_file_hash} - ${file_name}") endif() endif() else() ly_package_message(STATUS "ly_package: Expected file was not found: ${working_directory}/${file_name}") set(ANY_HASH_MISMATCHES TRUE) endif() endforeach() if (${ANY_HASH_MISMATCHES}) ly_package_message(STATUS "ly_package: Validation failed - files were missing, or had hash mismatches.") else() set(HASH_WAS_VALID TRUE PARENT_SCOPE) endif() endfunction() function(ly_package_get_target_folder package_name output_variable_name) # is it grafted onto the tree elsewhere? get_property(overridden_location GLOBAL PROPERTY LY_PACKAGE_DOWNLOAD_LOCATION_${package_name}) if (overridden_location) set(${output_variable_name} ${overridden_location} PARENT_SCOPE) elseif(NOT "${LY_PACKAGE_UNPACK_LOCATION}" STREQUAL "") set(${output_variable_name} ${LY_PACKAGE_UNPACK_LOCATION} PARENT_SCOPE) else() message(WARNING "ly_package: Could not locate the LY_PACKAGE_UNPACK_LOCATION variable" "'${LY_PACKAGE_UNPACK_LOCATION}' please fill it in!" " To compensate, this script will unpack into the build folder") set(${output_variable_name} ${CMAKE_BINARY_DIR} PARENT_SCOPE) endif() endfunction() #! Get the target cache folder function(ly_package_get_target_cache package_name output_variable_name) get_property(overridden_location GLOBAL PROPERTY LY_PACKAGE_DOWNLOAD_CACHE_LOCATION_${package_name}) if (overridden_location) set(${output_variable_name} ${overridden_location} PARENT_SCOPE) else() set(${output_variable_name} ${LY_PACKAGE_DOWNLOAD_CACHE_LOCATION} PARENT_SCOPE) endif() endfunction() #! given the name of a package, validate that all files are present and match as appropriate function(ly_validate_package package_name) ly_package_message(STATUS "ly_package: Validating ${package_name}...") unset(${package_name}_VALIDATED PARENT_SCOPE) ly_package_get_target_folder(${package_name} DOWNLOAD_LOCATION) if (NOT EXISTS "${DOWNLOAD_LOCATION}/${package_name}") ly_package_message(STATUS "ly_package: - ${package_name} is missing from ${DOWNLOAD_LOCATION}") return() endif() set(hash_file_name ${DOWNLOAD_LOCATION}/${package_name}/SHA256SUMS) set(json_file_name ${DOWNLOAD_LOCATION}/${package_name}/PackageInfo.json) if (NOT EXISTS ${hash_file_name}) ly_package_message(STATUS "Hash file missing from package ${package_name} (or package does not exist at all)") return() endif() if (NOT EXISTS ${json_file_name}) ly_package_message(STATUS "Package info file missing from package: ${json_file_name}") return() endif() set(package_stamp_file_name ${DOWNLOAD_LOCATION}/${package_name}.stamp) if (NOT EXISTS ${package_stamp_file_name}) # This can happen because of a previous version not making these stamp files in the correct place. # In order to avoid re-downloading the package we react to a missing stamp file by creating a new one. # This will cause any logic that wants to do things if the package is 'newer' to re-run, which is # safer than the alternative of not running things that do need to run when packages are downloaded. ly_package_message(STATUS "ly_package: Stamp file was missing, restoring: ${package_stamp_file_name}") file(TOUCH ${package_stamp_file_name}) endif() if (LY_PACKAGE_VALIDATE_PACKAGE) # this message is unconditional because its not the default to do this and also its much slower. # The package hash is always checked automatically on first downloard regardless of the value of this # variable, so if this variable is true, a user explicitly asked to do this. message(STATUS "Checking downloaded package ${package_name} because LY_PACKAGE_VALIDATE_PACKAGE is TRUE") ly_package_get_target_cache(${package_name} package_download_cache_location) set(temp_download_target ${package_download_cache_location}/${package_name}${LY_PACKAGE_EXT}) ly_get_package_expected_hash(${package_name} expected_package_hash) if (EXISTS ${temp_download_target}) file(SHA256 ${temp_download_target} existing_hash) endif() if (NOT "${existing_hash}" STREQUAL "${expected_package_hash}" ) # either the hash doesn't match or the file doesn't exist. Either way, we need to force download it again ly_package_message(STATUS "LY_PACKAGE_VALIDATE_PACKAGE : $[package_name}${LY_PACKAGE_EXT} is either missing or has the wrong hash, re-downloading") return() endif() endif() if (NOT LY_PACKAGE_VALIDATE_CONTENTS) ly_package_message(STATUS "Basic validation checks performed only becuase LY_PACKAGE_VALIDATE_CONTENTS is not enabled.") set(${package_name}_VALIDATED TRUE PARENT_SCOPE) ly_package_message(STATUS "ly_package: Validated ${package_name} - Basic Validation OK") return() endif() ly_validate_sha256sums_file( ${DOWNLOAD_LOCATION}/${package_name} ${DOWNLOAD_LOCATION}/${package_name}/SHA256SUMS) if (HASH_WAS_VALID) set(${package_name}_VALIDATED TRUE PARENT_SCOPE) ly_package_message(STATUS "ly_package: Validated ${package_name} - Full Validation OK") endif() endfunction() # ly_force_download_package # forces the download of a third party library regardless of current situation # package_name is like 'zlib-1.2.8-platform', not a file name or URL. # ---> Sets ${package_name}_VALIDATED on parent scope. TRUE only if the package # was successfully validated, including hash of contents. function(ly_force_download_package package_name) unset(${package_name}_FOUND PARENT_SCOPE) unset(${package_name}_VALIDATED PARENT_SCOPE) ly_package_get_target_folder(${package_name} DOWNLOAD_LOCATION) # this function contains a REMOVE_RECURSE. Because of that, we're going to do extra # validation on the inputs. # its not good enough for the variable to just exist but be empty, so we build strings if ("${package_name}" STREQUAL "" OR "${DOWNLOAD_LOCATION}" STREQUAL "") message(FATAL_ERROR "ly_package: ly_force_download_package called with invalid params! Enable LY_PACKAGE_DEBUG to debug.") endif() set(final_folder ${DOWNLOAD_LOCATION}/${package_name}) # is the package already present in the download cache, with the correct hash? ly_package_get_target_cache(${package_name} package_download_cache_location) set(temp_download_target ${package_download_cache_location}/${package_name}${LY_PACKAGE_EXT}) ly_get_package_expected_hash(${package_name} expected_package_hash) # can we reuse the download we already have in our download cache? if (EXISTS ${temp_download_target}) ly_package_message(STATUS "The target ${temp_download_target} exists") file(SHA256 ${temp_download_target} existing_hash) endif() if (NOT "${existing_hash}" STREQUAL "${expected_package_hash}" ) file(REMOVE ${temp_download_target}) # we print this message unconditionally because downloading a package # can take time and we only get here if its missing in the first place, so # this should happen once on the very first configure message(STATUS "Downloading package into ${final_folder}") ly_package_message(STATUS "ly_package: - downloading package '${package_name}' to '${final_folder}'") ly_package_internal_download_package(${package_name} ${temp_download_target}) # The above function will try every download location, with retries, so by the time we get here, the # operation is either done, or has completely failed. if (NOT EXISTS ${temp_download_target}) # the system will have already issued errors, no need to issue more. return() endif() else() ly_package_message(STATUS "ly_package: - package already correct hash ${temp_download_target}, re-using") endif() if (EXISTS ${DOWNLOAD_LOCATION}/${package_name}) ly_package_message(STATUS "ly_package: - removing folder ${DOWNLOAD_LOCATION}/${package_name} to replace it...") file(REMOVE_RECURSE ${DOWNLOAD_LOCATION}/${package_name}) if (EXISTS ${DOWNLOAD_LOCATION}/${package_name}) message(SEND_ERROR "ly_package: -folder ${DOWNLOAD_LOCATION}/${package_name} could not be removed. Check if some program has it open (VSCode, VS, Terminal windows, ...). Enable LY_PACKAGE_DEBUG to debug.") return() endif() endif() file(MAKE_DIRECTORY ${DOWNLOAD_LOCATION}/${package_name}) ly_package_message(STATUS "ly_package: - unpacking package...") execute_process(COMMAND ${CMAKE_COMMAND} -E tar xf ${temp_download_target} WORKING_DIRECTORY ${final_folder} COMMAND_ECHO STDOUT OUTPUT_VARIABLE unpack_result) # For the runtime dependencies cases, we need the timestamps of the files coming from 3rdParty to be newer than the ones # from the output so the new versions get copied over. The untar from the previous step preserves timestamps so they # can produce binaries with older timestamps to the ones that are in the build output. file(GLOB_RECURSE package_files LIST_DIRECTORIES false ${final_folder}/*) file(TOUCH_NOCREATE ${package_files}) if (NOT ${unpack_result} EQUAL 0) message(SEND_ERROR "ly_package: required package {package_name} could not be unpacked. Compile may fail! Enable LY_PACKAGE_DEBUG to debug.") return() else() if (NOT LY_PACKAGE_KEEP_AFTER_DOWNLOADING) ly_package_message(STATUS "ly_package: Removing package after unpacking (LY_PACKAGE_KEEP_AFTER_DOWNLOADING is ${LY_PACKAGE_KEEP_AFTER_DOWNLOADING})") file(REMOVE ${temp_download_target}) endif() endif() # because we just downloaded this file, we are going to force full hashing validation. # future runs will use the setting or default, which is a quicker validation set(LY_PACKAGE_VALIDATE_CONTENTS_old ${LY_PACKAGE_VALIDATE_CONTENTS}) set(LY_PACKAGE_VALIDATE_CONTENTS TRUE) ly_validate_package(${package_name}) set(LY_PACKAGE_VALIDATE_CONTENTS ${LY_PACKAGE_VALIDATE_CONTENTS_old}) set(${package_name}_VALIDATED ${package_name}_VALIDATED PARENT_SCOPE) set(package_stamp_file_name ${DOWNLOAD_LOCATION}/${package_name}.stamp) if (${package_name}_VALIDATED) # This we intentionally print out for each package that was actually downloaded from the internet, one time only: message(STATUS "Installed And Validated package at ${final_folder} - OK") # we also record a stamp file of when we did this, for use in other computations file(TOUCH ${package_stamp_file_name}) else() file(REMOVE ${package_stamp_file_name}) endif() endfunction() #! ly_enable_package: low-level function - adds a package to the auto package download system # Calling this will immediately make sure the package is present locally # and will add the local cache path to the additional module path (CMAKE_MODULE_PATH) # so that findxxxxx works # note that package_name here is the actual package name, not the association name. function(ly_enable_package package_name) # you can call this function as many times as you want, it will only try to validate the property once. # is it grafted onto the tree elsewhere? ly_package_get_target_folder(${package_name} DOWNLOAD_LOCATION) # add it to the prefixes so that we search here first # we add it in front so it can override any later paths, so "last one to declare" wins if (NOT "${DOWNLOAD_LOCATION}/${package_name}" IN_LIST "${CMAKE_MODULE_PATH}") set(CMAKE_MODULE_PATH ${DOWNLOAD_LOCATION}/${package_name} ${CMAKE_MODULE_PATH} PARENT_SCOPE) endif() get_property(existing_state GLOBAL PROPERTY LY_${package_name}_VALIDATED SET) if(NOT ${existing_state}) # note - check is for whether its SET, not whether its TRUE # if we get here, its not SET, so set it to FALSE pre-emptively so that # we don't try to download over and over, if the attempt to download fails. set_property(GLOBAL PROPERTY LY_${package_name}_VALIDATED FALSE) ly_validate_package(${package_name}) # sets VALIDATED in this scope. if (NOT ${package_name}_VALIDATED) # this will also validate it and set VALIDATED in this scope ly_force_download_package(${package_name}) endif() if(${package_name}_VALIDATED) set_property(GLOBAL PROPERTY LY_${package_name}_VALIDATED TRUE) # this message is unconditional as it will help prove that the package even was # attempted to be mounted using our package system. In the absence of this message # its going to be difficult to know why a package is missing in the logs # its also consistent with cmake's other messages, like # 'using Windows Target SDK xxxxx' or 'using clang xyz' message(STATUS "Using package ${DOWNLOAD_LOCATION}/${package_name}") # if the package goes missing, we will reconfigure: # note that this is already likely the case, if you ever use find_package, but this can help when # the package contains nothing cmake-related, for example, its just extra tools or assets or # similar. Packages are required to have PackageInfo.json at the root at minimum. set_property(DIRECTORY APPEND PROPERTY CMAKE_CONFIGURE_DEPENDS ${DOWNLOAD_LOCATION}/${package_name}/PackageInfo.json) endif() endif() endfunction() #! ly_associate_package - Main public function # - allows you to associate an actual package name ('zlib-1.2.8-multiplatform') # with any number of targets that are expected to be inside the package. # Associating packages with targets will cause cmake to download the package (if necessary), # and ensure the path to the package root is added to the find_package search paths. # For example # ly_associate_package(TARGETS zlib PACKAGE_NAME zlib-1.2.8-multiplatform PACKAGE_HASH e6f34b8ac16acf881e3d666ef9fd0c1aee94c3f69283fb6524d35d6f858eebbb) # - this will cause it to automatically download and activate this package if it finds a target that # depends on '3rdParty::zlib' in its runtime or its build time dependency list. # - note that '3rdParty' is implied, do not specify it in the TARGETS list. function(ly_associate_package) set(_oneValueArgs PACKAGE_NAME PACKAGE_HASH) set(_multiValueArgs TARGETS) cmake_parse_arguments(ly_associate_package "" "${_oneValueArgs}" "${_multiValueArgs}" ${ARGN}) if(NOT ly_associate_package_TARGETS) message(FATAL_ERROR "ly_associate_package was called without the TARGETS argument, at least one target is required") endif() if(NOT ly_associate_package_PACKAGE_NAME) message(FATAL_ERROR "ly_associate_package was called without the PACKAGE_NAME argument, this is required") endif() if(NOT ly_associate_package_PACKAGE_HASH) message(FATAL_ERROR "ly_associate_package was called without the PACKAGE_HASH argument, this is required") endif() foreach(find_package_name ${ly_associate_package_TARGETS}) set_property(GLOBAL PROPERTY LY_PACKAGE_ASSOCIATION_${find_package_name} ${ly_associate_package_PACKAGE_NAME}) set_property(GLOBAL PROPERTY LY_PACKAGE_HASH_${ly_associate_package_PACKAGE_NAME} ${ly_associate_package_PACKAGE_HASH}) endforeach() set_property(GLOBAL APPEND PROPERTY LY_PACKAGE_NAMES ${ly_associate_package_PACKAGE_NAME}) set_property(GLOBAL PROPERTY LY_PACKAGE_TARGETS_${ly_associate_package_PACKAGE_NAME} ${ly_associate_package_TARGETS}) endfunction() #! Given a package find_package name (eg, 'zlib' not the actual package name) # will set output_variable to the package id iff the package has a package # association declared, otherwise will unset it. function(ly_get_package_association find_package_name output_variable) unset(${output_variable}) get_property(is_associated GLOBAL PROPERTY LY_PACKAGE_ASSOCIATION_${find_package_name}) if (is_associated) set(${output_variable} ${is_associated} PARENT_SCOPE) endif() endfunction() # given a package name (as in, the actual name of the package, not its associated find libraries) # return the expected download package hash. macro(ly_get_package_expected_hash actual_package_name output_variable) unset(${output_variable}) get_property(package_hash_found GLOBAL PROPERTY LY_PACKAGE_HASH_${actual_package_name}) if (package_hash_found) set(${output_variable} ${package_hash_found}) else() # This is a fatal error because it is a programmer error and ignoring hashes # could be a security problem. message(FATAL_ERROR "ly_get_package_expected_hash could not find a hash for package ${actual_package_name}") endif() endmacro() # ly_set_package_download_location - OPTIONAL. # by default, packages are downloaded to the package root # at LY_PACKAGE_UNPACK_LOCATION - but if a package needs to be placed # elsewhere, use this. # note that package_name is expected to be the actual package name, not # the find_package(...) name! macro(ly_set_package_download_location package_name download_location) set_property(GLOBAL PROPERTY LY_PACKAGE_DOWNLOAD_LOCATION_${package_name} ${download_location}) endmacro() # ly_set_package_download_cache_location - OPTIONAL. # Similar to ly_set_package_download_location, this set the download # cache location for a particular package. # note that package_name is expected to be the actual package name, not # the find_package(...) name! macro(ly_set_package_download_cache_location package_name download_location) set_property(GLOBAL PROPERTY LY_PACKAGE_DOWNLOAD_CACHE_LOCATION_${package_name} ${download_location}) endmacro() # ly_download_associated_package - main public function # this just checks to see if the find_library_name (like 'zlib', not a package name) # is associated with a package, as above. If it is, it makes sure that the package # is brought into scope (and if necessary, downloaded.) macro(ly_download_associated_package find_library_name) unset(package_name) ly_get_package_association(${find_library_name} package_name) if (package_name) # it is an associated package. ly_enable_package(${package_name}) endif() endmacro() # ly_package_is_newer_than(package_name reference output_variable) # will set output_variable to TRUE if and only if the package was downloaded # more recently than the reference file's timestamp. function(ly_package_is_newer_than package_name reference_file output_variable) unset(${output_variable} PARENT_SCOPE) ly_package_get_target_folder(${package_name} DOWNLOAD_LOCATION) set(package_stamp_file_name ${DOWNLOAD_LOCATION}/${package_name}.stamp) if (EXISTS ${package_stamp_file_name} AND ${package_stamp_file_name} IS_NEWER_THAN ${reference_file}) set(${output_variable} TRUE PARENT_SCOPE) endif() endfunction() # if we're in script mode, we dont want to declare package associations if (NOT CMAKE_SCRIPT_MODE_FILE) # include the built in 3rd party packages that are for every platform. # you can put your package associations anywhere, but this provides # a good starting point. include(${LY_ROOT_FOLDER}/cmake/3rdParty/BuiltInPackages.cmake) endif() get_property(O3DE_SCRIPT_ONLY GLOBAL PROPERTY "O3DE_SCRIPT_ONLY") if(PAL_TRAIT_BUILD_HOST_TOOLS AND NOT O3DE_SCRIPT_ONLY) include(${LY_ROOT_FOLDER}/cmake/LYWrappers.cmake) # Importing this globally to handle AUTOMOC, AUTOUIC, AUTORCC ly_parse_third_party_dependencies(3rdParty::Qt) endif()