build-psiphon-framework.sh 9.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299
  1. #!/usr/bin/env bash
  2. set -e -u -x
  3. if [ -z ${1+x} ]; then BUILD_TAGS=""; else BUILD_TAGS="$1"; fi
  4. # Modify this value as we use newer Go versions.
  5. GO_VERSION_REQUIRED="1.17.1"
  6. # At this time, gomobile doesn't support modules
  7. export GO111MODULE=off
  8. # Reset the PATH to macOS default. This is mainly so we don't execute the wrong
  9. # gomobile executable.
  10. PATH=/usr/local/bin:/usr/bin:/bin:/usr/sbin:/sbin:/usr/local/go/bin
  11. # $GOROOT/bin allows build automation to provide various Go versions dynamically.
  12. # As gomobile would be installed at $GOPATH/bin, there is minimal risk that
  13. # adding $GOROOT/bin will run an unexpected gomobile binary.
  14. PATH=$GOROOT/bin:$PATH
  15. BASE_DIR=$(cd "$(dirname "$0")" ; pwd -P)
  16. cd "${BASE_DIR}"
  17. # The location of the final framework build
  18. BUILD_DIR="${BASE_DIR}/build"
  19. # Ensure go is installed
  20. which go 2>&1 > /dev/null
  21. if [[ $? != 0 ]]; then
  22. echo "Go is not installed in the path, aborting"
  23. exit 1
  24. fi
  25. UMBRELLA_FRAMEWORK_XCODE_PROJECT=${BASE_DIR}/PsiphonTunnel/PsiphonTunnel.xcodeproj/
  26. # Exporting these seems necessary for subcommands to pick them up.
  27. export GOPATH=${PWD}/go-ios-build
  28. export PATH=${GOPATH}/bin:${PATH}
  29. # The GOPATH we're using is temporary, so make sure there isn't one from a previous run.
  30. rm -rf "${GOPATH}"
  31. GOMOBILE_PINNED_REV=92f3b9caf7ba8f4f9c10074225afcba0cba47a62
  32. GOMOBILE_PATH=${GOPATH}/src/golang.org/x/mobile/cmd/gomobile
  33. TUNNEL_CORE_SRC_DIR=${GOPATH}/src/github.com/Psiphon-Labs/psiphon-tunnel-core
  34. PATH=${PATH}:${GOPATH}/bin
  35. mkdir -p "${GOPATH}"
  36. if [[ $? != 0 ]]; then
  37. echo "FAILURE: mkdir -p ${GOPATH}"
  38. exit 1
  39. fi
  40. # Symlink the current source directory into GOPATH, so that we're building the
  41. # code in this local repo, rather than pulling from Github and building that.
  42. mkdir -p "${GOPATH}/src/github.com/Psiphon-Labs"
  43. if [[ $? != 0 ]]; then
  44. echo "mkdir -p ${GOPATH}/src/github.com/Psiphon-Labs"
  45. exit 1
  46. fi
  47. ln -s "${BASE_DIR}/../.." "${GOPATH}/src/github.com/Psiphon-Labs/psiphon-tunnel-core"
  48. if [[ $? != 0 ]]; then
  49. echo "ln -s ../.. ${GOPATH}/src/github.com/Psiphon-Labs/psiphon-tunnel-core"
  50. exit 1
  51. fi
  52. # Builds Psi.framework library for the given platform.
  53. #
  54. # - Parameter 1: Possible values are "ios" and "simulator"
  55. # - Parameter 2: Variable name to set output path to.
  56. function gomobile_build_for_platform() {
  57. # Possible values are "ios" and "simulator"
  58. local PLATFORM=$1
  59. local TARGETS=""
  60. # gomobile pinned version 92f3b9c list of
  61. # valid archs are "arm", "arm64", "386", "amd64".
  62. # https://github.com/golang/mobile/blob/92f3b9caf7ba8f4f9c10074225afcba0cba47a62/cmd/gomobile/env.go#L26
  63. #
  64. # As of Go 1.15, "ios/arm" is no longer supported: https://golang.org/doc/go1.15#darwin
  65. case "${PLATFORM}" in
  66. ios)
  67. TARGETS="ios/arm64"
  68. ;;
  69. simulator)
  70. TARGETS="ios/amd64"
  71. ;;
  72. *)
  73. echo "FATAL ERROR: Unknown platform ${PLATFORM}"
  74. exit 1
  75. ;;
  76. esac
  77. echo "Build library for platform ${PLATFORM}"
  78. local FRAMEWORK="Psi"
  79. # Since frameworks for all platforms share the same name "Psi.framework",
  80. # each framework should be in its own directory.
  81. local INTERMEDIATE_OUPUT_DIR="${BUILD_DIR}/${PLATFORM}-psi-framework"
  82. local INTERMEDIATE_OUPUT_FILE="${FRAMEWORK}.framework"
  83. # local FRAMEWORK_BINARY="${INTERMEDIATE_OUPUT_DIR}/${INTERMEDIATE_OUPUT_FILE}/Versions/A/${FRAMEWORK}"
  84. local GOBIND_OUT="${INTERMEDIATE_OUPUT_DIR}/${INTERMEDIATE_OUPUT_FILE}"
  85. # We're using a generated-code prefix to workaround https://github.com/golang/go/issues/18693
  86. # CGO_CFLAGS_ALLOW is to workaround https://github.com/golang/go/issues/23742 in Go 1.9.4
  87. CGO_CFLAGS_ALLOW="-fmodules|-fblocks" "${GOPATH}"/bin/gomobile bind -v -x \
  88. -target "${TARGETS}" \
  89. -prefix Go \
  90. -tags="${BUILD_TAGS}" \
  91. -ldflags="${LDFLAGS}" \
  92. -o "${GOBIND_OUT}" github.com/Psiphon-Labs/psiphon-tunnel-core/MobileLibrary/psi
  93. rc=$?; if [[ $rc != 0 ]]; then
  94. echo "FAILURE: gomobile bind failed".
  95. exit $rc
  96. fi
  97. # Sets parameter $2 to value of GOBIND_OUT.
  98. eval "$2=${GOBIND_OUT}"
  99. }
  100. #
  101. # Check Go version
  102. #
  103. GO_VERSION=$(go version | sed -E -n 's/.*go([0-9]\.[0-9]+(\.[0-9]+)?).*/\1/p')
  104. if [[ ${GO_VERSION} != "${GO_VERSION_REQUIRED}" ]]; then
  105. echo "FAILURE: go version mismatch; require ${GO_VERSION_REQUIRED}; got ${GO_VERSION}"
  106. exit 1
  107. fi
  108. #
  109. # Get and install gomobile, using our pinned revision
  110. #
  111. go get -u golang.org/x/mobile/cmd/gomobile
  112. cd "${GOPATH}"/src/golang.org/x/mobile/cmd/gomobile
  113. git checkout master
  114. git checkout -b pinned ${GOMOBILE_PINNED_REV}
  115. go install
  116. "${GOPATH}"/bin/gomobile init -v -x
  117. if [[ $? != 0 ]]; then
  118. echo "FAILURE: ${GOPATH}/bin/gomobile init"
  119. exit 1
  120. fi
  121. #
  122. # gomobile bind
  123. #
  124. # Ensure BUILD* variables reflect the tunnel-core repo
  125. cd "${TUNNEL_CORE_SRC_DIR}"
  126. BUILDINFOFILE="${BASE_DIR}/psiphon-tunnel-core_buildinfo.txt"
  127. BUILDDATE=$(date +%Y-%m-%dT%H:%M:%S%z)
  128. BUILDREPO=$(git config --get remote.origin.url)
  129. BUILDREV=$(git rev-parse --short HEAD)
  130. GOVERSION=$(go version | perl -ne '/go version (.*?) / && print $1')
  131. GOMOBILEVERSION=$("${GOPATH}"/bin/gomobile version | perl -ne '/gomobile version (.*?) / && print $1')
  132. # see DEPENDENCIES comment in MobileLibrary/Android/make.bash
  133. cd "${GOPATH}"/src/github.com/Psiphon-Labs/psiphon-tunnel-core/MobileLibrary/psi
  134. DEPENDENCIES=$(echo -n "{" && GOOS=darwin go list -tags "${BUILD_TAGS}" -f '{{range $dep := .Deps}}{{printf "%s\n" $dep}}{{end}}' | GOOS=darwin xargs go list -tags "${BUILD_TAGS}" -f '{{if not .Standard}}{{.ImportPath}}{{end}}' | xargs -I pkg bash -c 'cd $GOPATH/src/$0 && if echo -n "$0" | grep -vEq "^github.com/Psiphon-Labs/psiphon-tunnel-core/" ; then echo -n "\"$0\":\"$(git rev-parse --short HEAD)\"," ; fi' pkg | sed 's/,$//' | tr -d '\n' && echo -n "}")
  135. LDFLAGS="\
  136. -s \
  137. -w \
  138. -X github.com/Psiphon-Labs/psiphon-tunnel-core/psiphon/common/buildinfo.buildDate=${BUILDDATE} \
  139. -X github.com/Psiphon-Labs/psiphon-tunnel-core/psiphon/common/buildinfo.buildRepo=${BUILDREPO} \
  140. -X github.com/Psiphon-Labs/psiphon-tunnel-core/psiphon/common/buildinfo.buildRev=${BUILDREV} \
  141. -X github.com/Psiphon-Labs/psiphon-tunnel-core/psiphon/common/buildinfo.goVersion=${GOVERSION} \
  142. -X github.com/Psiphon-Labs/psiphon-tunnel-core/psiphon/common/buildinfo.gomobileVersion=${GOMOBILEVERSION} \
  143. -X github.com/Psiphon-Labs/psiphon-tunnel-core/psiphon/common/buildinfo.dependencies=${DEPENDENCIES} \
  144. "
  145. echo -e "${BUILDDATE}\n${BUILDREPO}\n${BUILDREV}\n" > "$BUILDINFOFILE"
  146. echo ""
  147. echo "Variables for ldflags:"
  148. echo " Build date: ${BUILDDATE}"
  149. echo " Build repo: ${BUILDREPO}"
  150. echo " Build revision: ${BUILDREV}"
  151. echo " Go version: ${GOVERSION}"
  152. echo " Gomobile version: ${GOMOBILEVERSION}"
  153. echo ""
  154. #
  155. # Clean previous output
  156. #
  157. rm -rf "${BUILD_DIR}"
  158. #
  159. # Builds Psi.framework for each platform/variant.
  160. #
  161. IOS_PSI_FRAMEWORK=""
  162. gomobile_build_for_platform "ios" IOS_PSI_FRAMEWORK
  163. SIMULATOR_PSI_FRAMEWORK=""
  164. gomobile_build_for_platform "simulator" SIMULATOR_PSI_FRAMEWORK
  165. #
  166. # Xcode archive for each platform.
  167. #
  168. # Xcode project requires Psi.framework bundle at $PSI_FRAMEWORK.
  169. # Except for macOS, Apple does not support umbrella frameworks that
  170. # contain other frameworks. So for building framework for each platform/variant,
  171. # the Psi.framework should be copied to the path at $PSI_FRAMEWORK_PATH.
  172. PSI_FRAMEWORK_PATH="${BASE_DIR}/PsiphonTunnel/PsiphonTunnel"
  173. PSI_FRAMEWORK="${PSI_FRAMEWORK_PATH}/Psi.framework"
  174. # Build PsiphonTunnel framework for iOS.
  175. echo "$IOS_PSI_FRAMEWORK"
  176. echo "$SIMULATOR_PSI_FRAMEWORK"
  177. # Copies iOS Psi.framework
  178. rm -rf "${PSI_FRAMEWORK}"
  179. cp -r "${IOS_PSI_FRAMEWORK}" "${PSI_FRAMEWORK_PATH}"
  180. IOS_ARCHIVE="${BUILD_DIR}/ios.xcarchive"
  181. xcodebuild clean archive \
  182. -project "${UMBRELLA_FRAMEWORK_XCODE_PROJECT}" \
  183. -scheme "PsiphonTunnel" \
  184. -configuration "Release" \
  185. -sdk iphoneos \
  186. -archivePath "${IOS_ARCHIVE}" \
  187. CODE_SIGN_IDENTITY="" \
  188. CODE_SIGNING_REQUIRED="NO" \
  189. CODE_SIGN_ENTITLEMENTS="" \
  190. CODE_SIGNING_ALLOWED="NO" \
  191. STRIP_BITCODE_FROM_COPIED_FILES="NO" \
  192. BUILD_LIBRARY_FOR_DISTRIBUTION="YES" \
  193. ONLY_ACTIVE_ARCH="NO" \
  194. SKIP_INSTALL="NO" \
  195. EXCLUDED_ARCHS="armv7"
  196. # Build PsiphonTunnel framework for simulator.
  197. #
  198. # Note:
  199. # - Excludes 32-bit Intel: EXCLUDED_ARCHS="i386".
  200. # - Excludes ARM Macs: EXCLUDED_ARCHS="arm64".
  201. # Copies simulator Psi.framework
  202. rm -rf "${PSI_FRAMEWORK}"
  203. cp -r "${SIMULATOR_PSI_FRAMEWORK}" "${PSI_FRAMEWORK_PATH}"
  204. SIMULATOR_ARCHIVE="${BUILD_DIR}/simulator.xcarchive"
  205. xcodebuild clean archive \
  206. -project "${UMBRELLA_FRAMEWORK_XCODE_PROJECT}" \
  207. -scheme "PsiphonTunnel" \
  208. -configuration "Release" \
  209. -sdk iphonesimulator \
  210. -archivePath "${SIMULATOR_ARCHIVE}" \
  211. CODE_SIGN_IDENTITY="" \
  212. CODE_SIGNING_REQUIRED="NO" \
  213. CODE_SIGN_ENTITLEMENTS="" \
  214. CODE_SIGNING_ALLOWED="NO" \
  215. STRIP_BITCODE_FROM_COPIED_FILES="NO" \
  216. BUILD_LIBRARY_FOR_DISTRIBUTION="YES" \
  217. ONLY_ACTIVE_ARCH="NO" \
  218. SKIP_INSTALL="NO" \
  219. EXCLUDED_ARCHS="arm64 i386"
  220. #
  221. # Building PsiphonTunnel.xcframework
  222. #
  223. xcodebuild -create-xcframework \
  224. -framework "${IOS_ARCHIVE}/Products/Library/Frameworks/PsiphonTunnel.framework" \
  225. -debug-symbols "${IOS_ARCHIVE}/dSYMs/PsiphonTunnel.framework.dSYM" \
  226. -framework "${SIMULATOR_ARCHIVE}/Products/Library/Frameworks/PsiphonTunnel.framework" \
  227. -debug-symbols "${SIMULATOR_ARCHIVE}/dSYMs/PsiphonTunnel.framework.dSYM" \
  228. -output "${BUILD_DIR}/PsiphonTunnel.xcframework"
  229. # Jenkins loses symlinks from the framework directory, which results in a build
  230. # artifact that is invalid to use in an App Store app. Instead, we will zip the
  231. # resulting build and use that as the artifact.
  232. cd "${BUILD_DIR}"
  233. zip --recurse-paths --symlinks build.zip ./PsiphonTunnel.xcframework --exclude "*.DS_Store"
  234. echo "BUILD DONE"