You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
o3de/cmake/3rdPartyPackages.cmake

719 lines
36 KiB
CMake

#
# 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})
set(download_target ${LY_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()
#! 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")
set(temp_download_target ${LY_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?
set(temp_download_target ${LY_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 waill 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_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)
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()
if(PAL_TRAIT_BUILD_HOST_TOOLS)
include(${LY_ROOT_FOLDER}/cmake/LYWrappers.cmake)
# Importing this globally to handle AUTOMOC, AUTOUIC, AUTORCC
ly_parse_third_party_dependencies(3rdParty::Qt)
endif()