ExternalProject-download.cmake.in 3.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161
  1. #=============================================================================
  2. # Copyright 2008-2013 Kitware, Inc.
  3. # Copyright 2016 Ruslan Baratov
  4. #
  5. # Distributed under the OSI-approved BSD License (the "License");
  6. # see accompanying file Copyright.txt for details.
  7. #
  8. # This software is distributed WITHOUT ANY WARRANTY; without even the
  9. # implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
  10. # See the License for more information.
  11. #=============================================================================
  12. # (To distribute this file outside of CMake, substitute the full
  13. # License text for the above reference.)
  14. cmake_minimum_required(VERSION 3.5)
  15. function(check_file_hash has_hash hash_is_good)
  16. if("${has_hash}" STREQUAL "")
  17. message(FATAL_ERROR "has_hash Can't be empty")
  18. endif()
  19. if("${hash_is_good}" STREQUAL "")
  20. message(FATAL_ERROR "hash_is_good Can't be empty")
  21. endif()
  22. if("@ALGO@" STREQUAL "")
  23. # No check
  24. set("${has_hash}" FALSE PARENT_SCOPE)
  25. set("${hash_is_good}" FALSE PARENT_SCOPE)
  26. return()
  27. endif()
  28. set("${has_hash}" TRUE PARENT_SCOPE)
  29. message(STATUS "verifying file...
  30. file='@LOCAL@'")
  31. file("@ALGO@" "@LOCAL@" actual_value)
  32. if(NOT "${actual_value}" STREQUAL "@EXPECT_VALUE@")
  33. set("${hash_is_good}" FALSE PARENT_SCOPE)
  34. message(STATUS "@ALGO@ hash of
  35. @LOCAL@
  36. does not match expected value
  37. expected: '@EXPECT_VALUE@'
  38. actual: '${actual_value}'")
  39. else()
  40. set("${hash_is_good}" TRUE PARENT_SCOPE)
  41. endif()
  42. endfunction()
  43. function(sleep_before_download attempt)
  44. if(attempt EQUAL 0)
  45. return()
  46. endif()
  47. if(attempt EQUAL 1)
  48. message(STATUS "Retrying...")
  49. return()
  50. endif()
  51. set(sleep_seconds 0)
  52. if(attempt EQUAL 2)
  53. set(sleep_seconds 5)
  54. elseif(attempt EQUAL 3)
  55. set(sleep_seconds 5)
  56. elseif(attempt EQUAL 4)
  57. set(sleep_seconds 15)
  58. elseif(attempt EQUAL 5)
  59. set(sleep_seconds 60)
  60. elseif(attempt EQUAL 6)
  61. set(sleep_seconds 90)
  62. elseif(attempt EQUAL 7)
  63. set(sleep_seconds 300)
  64. else()
  65. set(sleep_seconds 1200)
  66. endif()
  67. message(STATUS "Retry after ${sleep_seconds} seconds (attempt #${attempt}) ...")
  68. execute_process(COMMAND "${CMAKE_COMMAND}" -E sleep "${sleep_seconds}")
  69. endfunction()
  70. if("@LOCAL@" STREQUAL "")
  71. message(FATAL_ERROR "LOCAL can't be empty")
  72. endif()
  73. if("@REMOTE@" STREQUAL "")
  74. message(FATAL_ERROR "REMOTE can't be empty")
  75. endif()
  76. if(EXISTS "@LOCAL@")
  77. check_file_hash(has_hash hash_is_good)
  78. if(has_hash)
  79. if(hash_is_good)
  80. message(STATUS "File already exists and hash match (skip download):
  81. file='@LOCAL@'
  82. @ALGO@='@EXPECT_VALUE@'"
  83. )
  84. return()
  85. else()
  86. message(STATUS "File already exists but hash mismatch. Removing...")
  87. file(REMOVE "@LOCAL@")
  88. endif()
  89. else()
  90. message(STATUS "File already exists but no hash specified (use URL_HASH):
  91. file='@LOCAL@'
  92. Old file will be removed and new file downloaded from URL."
  93. )
  94. file(REMOVE "@LOCAL@")
  95. endif()
  96. endif()
  97. set(retry_number 5)
  98. foreach(i RANGE ${retry_number})
  99. sleep_before_download(${i})
  100. message(STATUS "downloading...
  101. src='@REMOTE@'
  102. dst='@LOCAL@'
  103. timeout='@TIMEOUT_MSG@'")
  104. @TLS_VERIFY_CODE@
  105. @TLS_CAINFO_CODE@
  106. file(
  107. DOWNLOAD
  108. "@REMOTE@" "@LOCAL@"
  109. @SHOW_PROGRESS@
  110. @TIMEOUT_ARGS@
  111. STATUS status
  112. LOG log
  113. )
  114. list(GET status 0 status_code)
  115. list(GET status 1 status_string)
  116. if(status_code EQUAL 0)
  117. check_file_hash(has_hash hash_is_good)
  118. if(has_hash AND NOT hash_is_good)
  119. message(STATUS "Hash mismatch, removing...")
  120. file(REMOVE "@LOCAL@")
  121. else()
  122. message(STATUS "Downloading... done")
  123. return()
  124. endif()
  125. else()
  126. message("error: downloading '@REMOTE@' failed
  127. status_code: ${status_code}
  128. status_string: ${status_string}
  129. log:
  130. --- LOG BEGIN ---
  131. ${log}
  132. --- LOG END ---"
  133. )
  134. endif()
  135. endforeach()
  136. message(FATAL_ERROR "Downloading failed")