Skip to content

Commit

Permalink
[vcpkg] Add support for installing from HEAD
Browse files Browse the repository at this point in the history
  • Loading branch information
ras0219-msft committed May 3, 2017
1 parent f10861f commit 4633c5e
Show file tree
Hide file tree
Showing 11 changed files with 287 additions and 29 deletions.
10 changes: 5 additions & 5 deletions ports/azure-storage-cpp/portfile.cmake
Original file line number Diff line number Diff line change
@@ -1,12 +1,12 @@
include(vcpkg_common_functions)
set(SOURCE_PATH ${CURRENT_BUILDTREES_DIR}/src/azure-storage-cpp-3.0.0)

vcpkg_download_distfile(ARCHIVE
URLS "https://github.com/Azure/azure-storage-cpp/archive/v3.0.0.tar.gz"
FILENAME "azure-storage-cpp/v3.0.0.tar.gz"
vcpkg_from_github(
OUT_SOURCE_PATH SOURCE_PATH
REPO Azure/azure-storage-cpp
REF v3.0.0
SHA512 45d0d7f8cc350a16cff0371cdd442e851912c89061acfec559482e8f79cebafffd8681b32a30b878e329235cd3aaad5d2ff797d1148302e3109cf5111df14b97
HEAD_REF master
)
vcpkg_extract_source_archive(${ARCHIVE})

vcpkg_apply_patches(
SOURCE_PATH ${SOURCE_PATH}
Expand Down
33 changes: 20 additions & 13 deletions ports/cpprestsdk/portfile.cmake
Original file line number Diff line number Diff line change
@@ -1,20 +1,20 @@
include(vcpkg_common_functions)
set(SOURCE_PATH ${CURRENT_BUILDTREES_DIR}/src/cpprestsdk-2.9.0)

vcpkg_download_distfile(ARCHIVE
URLS "https://github.com/Microsoft/cpprestsdk/archive/v2.9.0.tar.gz"
FILENAME "cpprestsdk-2.9.0.tar.gz"
vcpkg_from_github(
OUT_SOURCE_PATH SOURCE_PATH
REPO Microsoft/cpprestsdk
REF v2.9.0
SHA512 c75de6ad33b3e8d2c6ba7c0955ed851d557f78652fb38a565de0cfbc99e7db89cb6fa405857512e5149df80356c51ae9335abd914c3c593fa6658ac50adf4e29
HEAD_REF master
)
vcpkg_extract_source_archive(${ARCHIVE})


vcpkg_apply_patches(
SOURCE_PATH ${SOURCE_PATH}
PATCHES
${CMAKE_CURRENT_LIST_DIR}/0001_cmake.patch
${CMAKE_CURRENT_LIST_DIR}/0002_no_websocketpp_in_uwp.patch
)
if(NOT VCPKG_USE_HEAD_VERSION)
vcpkg_apply_patches(
SOURCE_PATH ${SOURCE_PATH}
PATCHES
${CMAKE_CURRENT_LIST_DIR}/0001_cmake.patch
${CMAKE_CURRENT_LIST_DIR}/0002_no_websocketpp_in_uwp.patch
)
endif()

set(OPTIONS)
if(NOT VCPKG_CMAKE_SYSTEM_NAME STREQUAL "WindowsStore")
Expand All @@ -26,17 +26,24 @@ endif()

vcpkg_configure_cmake(
SOURCE_PATH ${SOURCE_PATH}/Release
PREFER_NINJA
OPTIONS
${OPTIONS}
-DBUILD_TESTS=OFF
-DBUILD_SAMPLES=OFF
-DCPPREST_EXCLUDE_WEBSOCKETS=OFF
-DCPPREST_EXPORT_DIR=share/cpprestsdk
OPTIONS_DEBUG
-DCASA_INSTALL_HEADERS=OFF
-DCPPREST_INSTALL_HEADERS=OFF
)

vcpkg_install_cmake()

if(VCPKG_USE_HEAD_VERSION)
vcpkg_fixup_cmake_targets()
endif()

file(INSTALL
${SOURCE_PATH}/license.txt
DESTINATION ${CURRENT_PACKAGES_DIR}/share/cpprestsdk RENAME copyright)
Expand Down
2 changes: 2 additions & 0 deletions scripts/cmake/vcpkg_common_functions.cmake
Original file line number Diff line number Diff line change
Expand Up @@ -4,6 +4,8 @@ include(vcpkg_extract_source_archive)
include(vcpkg_execute_required_process)
include(vcpkg_execute_required_process_repeat)
include(vcpkg_find_acquire_program)
include(vcpkg_fixup_cmake_targets)
include(vcpkg_from_github)
include(vcpkg_build_cmake)
include(vcpkg_build_msbuild)
include(vcpkg_build_qmake)
Expand Down
4 changes: 4 additions & 0 deletions scripts/cmake/vcpkg_download_distfile.cmake
Original file line number Diff line number Diff line change
Expand Up @@ -23,6 +23,10 @@ function(vcpkg_download_distfile VAR)
message(STATUS "Using cached ${downloaded_file_path}")
test_hash("cached file" "Please delete the file and retry if this file should be downloaded again.")
else()
if(_VCPKG_NO_DOWNLOADS)
message(FATAL_ERROR "Downloads are disabled, but '${downloaded_file_path}' does not exist.")
endif()

# Tries to download the file.
foreach(url IN LISTS vcpkg_download_distfile_URLS)
message(STATUS "Downloading ${url}...")
Expand Down
31 changes: 24 additions & 7 deletions scripts/cmake/vcpkg_extract_source_archive.cmake
Original file line number Diff line number Diff line change
@@ -1,22 +1,39 @@
include(vcpkg_execute_required_process)

function(vcpkg_extract_source_archive ARCHIVE)
if(NOT ARGC EQUAL 2)
set(WORKING_DIRECTORY ${CURRENT_BUILDTREES_DIR}/src)
function(vcpkg_extract_source_archive_ex)
cmake_parse_arguments(_vesae "" "ARCHIVE;WORKING_DIRECTORY" "" ${ARGN})

if(NOT _vesae_ARCHIVE)
message(FATAL_ERROR "Must specify ARCHIVE parameter to vcpkg_extract_source_archive_ex()")
endif()

if(DEFINED _vesae_WORKING_DIRECTORY)
set(WORKING_DIRECTORY ${_vesae_WORKING_DIRECTORY})
else()
set(WORKING_DIRECTORY ${ARGV1})
set(WORKING_DIRECTORY ${CURRENT_BUILDTREES_DIR}/src)
endif()

get_filename_component(ARCHIVE_FILENAME ${ARCHIVE} NAME)
get_filename_component(ARCHIVE_FILENAME ${_vesae_ARCHIVE} NAME)
if(NOT EXISTS ${WORKING_DIRECTORY}/${ARCHIVE_FILENAME}.extracted)
message(STATUS "Extracting source ${ARCHIVE}")
message(STATUS "Extracting source ${_vesae_ARCHIVE}")
file(MAKE_DIRECTORY ${WORKING_DIRECTORY})
vcpkg_execute_required_process(
COMMAND ${CMAKE_COMMAND} -E tar xjf ${ARCHIVE}
COMMAND ${CMAKE_COMMAND} -E tar xjf ${_vesae_ARCHIVE}
WORKING_DIRECTORY ${WORKING_DIRECTORY}
LOGNAME extract
)
file(WRITE ${WORKING_DIRECTORY}/${ARCHIVE_FILENAME}.extracted)
endif()
message(STATUS "Extracting done")
endfunction()

function(vcpkg_extract_source_archive ARCHIVE)
if(NOT ARGC EQUAL 2)
vcpkg_extract_source_archive_ex(ARCHIVE ${ARCHIVE})
else()
vcpkg_extract_source_archive_ex(
ARCHIVE ${ARCHIVE}
WORKING_DIRECTORY ${ARGV1}
)
endif()
endfunction()
47 changes: 47 additions & 0 deletions scripts/cmake/vcpkg_fixup_cmake_targets.cmake
Original file line number Diff line number Diff line change
@@ -0,0 +1,47 @@
#.rst:
# .. command:: vcpkg_fixup_cmake_targets
#
# Transform all /debug/share/<port>/*targets-debug.cmake files and move them to /share/<port>.
# Removes all /debug/share/<port>/*targets.cmake and /debug/share/<port>/*config.cmake

This comment has been minimized.

Copy link
@KindDragon

KindDragon May 6, 2017

Contributor

Do you plan to support all combination?

<prefix>/                                               (W)
<prefix>/(cmake|CMake)/                                 (W)
<prefix>/<name>*/                                       (W)
<prefix>/<name>*/(cmake|CMake)/                         (W)
<prefix>/(lib/<arch>|lib|share)/cmake/<name>*/          (U)
<prefix>/(lib/<arch>|lib|share)/<name>*/                (U)
<prefix>/(lib/<arch>|lib|share)/<name>*/(cmake|CMake)/  (U)

https://cmake.org/cmake/help/v3.4/command/find_package.html

This comment has been minimized.

Copy link
@KindDragon

KindDragon May 6, 2017

Contributor

gflags for example store *.cmake file in CMake folder

This comment has been minimized.

Copy link
@ras0219-msft

ras0219-msft May 7, 2017

Author Contributor

It really would be ideal to handle all these different cases -- however it's a bit tricky.

The general format is that the <name>-targets.cmake file creates the _IMPORT_PREFIX variable by effectively doing

get_filename_component(_IMPORT_PREFIX "${CMAKE_CURRENT_LIST_FILE}" PATH)
get_filename_component(_IMPORT_PREFIX "${_IMPORT_PREFIX}" PATH) # N of these

This means we could "easily" handle any directories that have the same depth as share/name/ -- name/cmake/ and lib/name/ but not lib/cmake/name/.

I don't mean to say we shouldn't handle the other cases -- we should! -- but it will require more work and testing than one or two lines.

This comment has been minimized.

Copy link
@ras0219-msft

ras0219-msft May 7, 2017

Author Contributor

I've opened #1044 to track this.

This comment has been minimized.

Copy link
@KindDragon

KindDragon May 7, 2017

Contributor

Ty for explanation

#
# ::
# vcpkg_fixup_cmake_targets()
#

function(vcpkg_fixup_cmake_targets)
cmake_parse_arguments(_vfct "" "" "" ${ARGN})

set(DEBUG_SHARE ${CURRENT_PACKAGES_DIR}/debug/share/${PORT})
set(RELEASE_SHARE ${CURRENT_PACKAGES_DIR}/share/${PORT})

if(NOT EXISTS ${DEBUG_SHARE})
message(FATAL_ERROR "'${DEBUG_SHARE}' does not exist")
endif()

file(GLOB UNUSED_FILES "${DEBUG_SHARE}/*[Tt]argets.cmake" "${DEBUG_SHARE}/*[Cc]onfig.cmake")
file(REMOVE ${UNUSED_FILES})

file(GLOB DEBUG_TARGETS "${DEBUG_SHARE}/*[Tt]argets-debug.cmake")

foreach(DEBUG_TARGET ${DEBUG_TARGETS})
get_filename_component(DEBUG_TARGET_NAME ${DEBUG_TARGET} NAME)

file(READ ${DEBUG_TARGET} _contents)
string(REPLACE "\${_IMPORT_PREFIX}" "\${_IMPORT_PREFIX}/debug" _contents "${_contents}")
file(WRITE ${CURRENT_PACKAGES_DIR}/share/${PORT}/${DEBUG_TARGET_NAME} "${_contents}")

file(REMOVE ${DEBUG_TARGET})
endforeach()

# Remove /debug/share/<port>/ if it's empty.
file(GLOB_RECURSE REMAINING_FILES "${DEBUG_SHARE}/*")
if(NOT REMAINING_FILES)
file(REMOVE_RECURSE ${DEBUG_SHARE})
endif()

# Remove /debug/share/ if it's empty.
file(GLOB_RECURSE REMAINING_FILES "${CURRENT_PACKAGES_DIR}/debug/share/*")
if(NOT REMAINING_FILES)
file(REMOVE_RECURSE ${CURRENT_PACKAGES_DIR}/debug/share)
endif()
endfunction()
131 changes: 131 additions & 0 deletions scripts/cmake/vcpkg_from_github.cmake
Original file line number Diff line number Diff line change
@@ -0,0 +1,131 @@
# Usage:
# vcpkg_from_github(
# OUT_SOURCE_PATH <OutVar for SOURCE_PATH (SOURCE_PATH)>
# REPO <Repository (Microsoft/cpprestsdk)>
# REF <stable ref (v2.0.0)>
# SHA512 <SHA for REF (45d0d7f8cc350...)>
# HEAD_REF <unstable branch (master)>
# )
#
# Notes:
# This will export VCPKG_HEAD_VERSION variable during head builds.
function(vcpkg_from_github)
set(oneValueArgs OUT_SOURCE_PATH REPO REF SHA512 HEAD_REF)
set(multipleValuesArgs)
cmake_parse_arguments(_vdud "" "${oneValueArgs}" "${multipleValuesArgs}" ${ARGN})

if(NOT _vdud_OUT_SOURCE_PATH)
message(FATAL_ERROR "OUT_SOURCE_PATH must be specified.")
endif()

if((_vdud_REF AND NOT _vdud_SHA512) OR (NOT _vdud_REF AND _vdud_SHA512))
message(FATAL_ERROR "SHA512 must be specified if REF is specified.")
endif()

if(NOT _vdud_REPO)
message(FATAL_ERROR "The GitHub repository must be specified.")
endif()

if(NOT _vdud_REF AND NOT _vdud_HEAD_REF)
message(FATAL_ERROR "At least one of REF and HEAD_REF must be specified.")
endif()

string(REGEX REPLACE ".*/" "" REPO_NAME ${_vdud_REPO})
string(REGEX REPLACE "/.*" "" ORG_NAME ${_vdud_REPO})

macro(set_SOURCE_PATH BASE BASEREF)
set(SOURCE_PATH "${BASE}/${REPO_NAME}-${BASEREF}")
if(EXISTS ${SOURCE_PATH})
set(${_vdud_OUT_SOURCE_PATH} "${SOURCE_PATH}" PARENT_SCOPE)
else()
# Sometimes GitHub strips a leading 'v' off the REF.
string(REGEX REPLACE "^v" "" REF ${BASEREF})
set(SOURCE_PATH "${BASE}/${REPO_NAME}-${REF}")
if(EXISTS ${SOURCE_PATH})
set(${_vdud_OUT_SOURCE_PATH} "${SOURCE_PATH}" PARENT_SCOPE)
else()
message(FATAL_ERROR "Could not determine source path: '${BASE}/${REPO_NAME}-${BASEREF}' does not exist")
endif()
endif()
endmacro()

if(VCPKG_USE_HEAD_VERSION AND NOT _vdud_HEAD_REF)
message(STATUS "Package does not specify HEAD_REF. Falling back to non-HEAD version.")
set(VCPKG_USE_HEAD_VERSION OFF)
endif()

# Handle --no-head scenarios
if(NOT VCPKG_USE_HEAD_VERSION)
if(NOT _vdud_REF)
message(FATAL_ERROR "Package does not specify REF. It must built using --head.")
endif()

vcpkg_download_distfile(ARCHIVE
URLS "https://github.com/${ORG_NAME}/${REPO_NAME}/archive/${_vdud_REF}.tar.gz"
SHA512 "${_vdud_SHA512}"
FILENAME "${ORG_NAME}-${REPO_NAME}-${_vdud_REF}.tar.gz"
)
vcpkg_extract_source_archive_ex(ARCHIVE "${ARCHIVE}")
set_SOURCE_PATH(${CURRENT_BUILDTREES_DIR}/src ${_vdud_REF})
return()
endif()

# The following is for --head scenarios
set(URL "https://github.com/${ORG_NAME}/${REPO_NAME}/archive/${_vdud_HEAD_REF}.tar.gz")
set(downloaded_file_path "${DOWNLOADS}/${ORG_NAME}-${REPO_NAME}-${_vdud_HEAD_REF}.tar.gz")

if(_VCPKG_NO_DOWNLOADS)
if(NOT EXISTS ${downloaded_file_path} OR NOT EXISTS ${downloaded_file_path}.version)
message(FATAL_ERROR "Downloads are disabled, but '${downloaded_file_path}' does not exist.")
endif()
message(STATUS "Using cached ${downloaded_file_path}")
else()
if(EXISTS ${downloaded_file_path})
message(STATUS "Purging cached ${downloaded_file_path} to fetch latest (use --no-downloads to suppress)")
file(REMOVE ${downloaded_file_path})
endif()
if(EXISTS ${downloaded_file_path}.version)
file(REMOVE ${downloaded_file_path}.version)
endif()
if(EXISTS ${CURRENT_BUILDTREES_DIR}/src/head)
file(REMOVE_RECURSE ${CURRENT_BUILDTREES_DIR}/src/head)
endif()

# Try to download the file and version information from github.
message(STATUS "Downloading ${URL}...")
file(DOWNLOAD "https://api.github.com/repos/${ORG_NAME}/${REPO_NAME}/git/refs/heads/${_vdud_HEAD_REF}"
${downloaded_file_path}.version
STATUS download_status
)
list(GET download_status 0 status_code)
if (NOT "${status_code}" STREQUAL "0")
file(REMOVE ${downloaded_file_path}.version)
message(FATAL_ERROR "Downloading version info for ${URL}... Failed. Status: ${download_status}")
endif()

file(DOWNLOAD ${URL} ${downloaded_file_path} STATUS download_status)
list(GET download_status 0 status_code)
if (NOT "${status_code}" STREQUAL "0")
file(REMOVE ${downloaded_file_path})
message(FATAL_ERROR "Downloading ${URL}... Failed. Status: ${download_status}")
else()
message(STATUS "Downloading ${URL}... OK")
endif()
endif()

vcpkg_extract_source_archive_ex(
ARCHIVE "${downloaded_file_path}"
WORKING_DIRECTORY "${CURRENT_BUILDTREES_DIR}/src/head"
)

# Parse the github refs response with regex.
# TODO: use some JSON swiss-army-knife utility instead.
file(READ "${downloaded_file_path}.version" _contents)
string(REGEX MATCH "\"sha\": \"[a-f0-9]+\"" x "${_contents}")
string(REGEX REPLACE "\"sha\": \"([a-f0-9]+)\"" "\\1" _version ${x})

# exports VCPKG_HEAD_VERSION to the caller. This will get picked up by ports.cmake after the build.
set(VCPKG_HEAD_VERSION ${_version} PARENT_SCOPE)

set_SOURCE_PATH(${CURRENT_BUILDTREES_DIR}/src/head ${_vdud_HEAD_REF})
endfunction()
3 changes: 3 additions & 0 deletions scripts/ports.cmake
Original file line number Diff line number Diff line change
Expand Up @@ -88,6 +88,9 @@ if(CMD MATCHES "^BUILD$")
if (DEFINED VCPKG_POLICY_EMPTY_INCLUDE_FOLDER)
file(APPEND ${BUILD_INFO_FILE_PATH} "PolicyEmptyIncludeFolder: ${VCPKG_POLICY_EMPTY_INCLUDE_FOLDER}\n")
endif()
if (DEFINED VCPKG_HEAD_VERSION)
file(APPEND ${BUILD_INFO_FILE_PATH} "Version: ${VCPKG_HEAD_VERSION}\n")
endif()
elseif(CMD MATCHES "^CREATE$")
file(TO_NATIVE_PATH ${VCPKG_ROOT_DIR} NATIVE_VCPKG_ROOT_DIR)
file(TO_NATIVE_PATH ${DOWNLOADS} NATIVE_DOWNLOADS)
Expand Down
6 changes: 6 additions & 0 deletions toolsrc/include/vcpkg_Build.h
Original file line number Diff line number Diff line change
Expand Up @@ -6,6 +6,7 @@
#include "StatusParagraphs.h"
#include "VcpkgPaths.h"
#include "vcpkg_Files.h"
#include "vcpkg_optional.h"
#include <map>
#include <string>
#include <unordered_map>
Expand Down Expand Up @@ -50,6 +51,9 @@ namespace vcpkg::Build
const SourceParagraph& src;
const Triplet& triplet;
fs::path port_dir;

bool use_head_version;
bool no_downloads;
};

ExtendedBuildResult build_package(const VcpkgPaths& paths,
Expand All @@ -63,6 +67,8 @@ namespace vcpkg::Build
PostBuildLint::LinkageType crt_linkage;
PostBuildLint::LinkageType library_linkage;

Optional<std::string> version;

std::map<PostBuildLint::BuildPolicies, bool> policies;
};

Expand Down
Loading

0 comments on commit 4633c5e

Please sign in to comment.