appimage.yml 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260
  1. name: Build AppImage
  2. on:
  3. release:
  4. types: [published]
  5. workflow_dispatch:
  6. inputs:
  7. version:
  8. description: 'Version'
  9. required: false
  10. push:
  11. branches-ignore:
  12. - 'dependabot/**'
  13. paths:
  14. - '.github/workflows/appimage.yml'
  15. - 'dist/linux/appimage/**'
  16. - 'dist/linux/common/**'
  17. - 'dist/linux/resources/**'
  18. env:
  19. JAVA_DIST: 'temurin'
  20. JAVA_VERSION: '24.0.1+9'
  21. jobs:
  22. get-version:
  23. uses: ./.github/workflows/get-version.yml
  24. with:
  25. version: ${{ inputs.version }} #okay if not defined
  26. build:
  27. name: Build AppImage
  28. runs-on: ${{ matrix.os }}
  29. needs: [get-version]
  30. strategy:
  31. fail-fast: false
  32. matrix:
  33. include:
  34. - os: ubuntu-latest
  35. appimage-suffix: x86_64
  36. openjfx-url: 'https://download2.gluonhq.com/openjfx/24.0.1/openjfx-24.0.1_linux-x64_bin-jmods.zip'
  37. openjfx-sha: '425fac742b9fbd095b2ce868cff82d1024620f747c94a7144d0a4879e756146c'
  38. - os: ubuntu-24.04-arm
  39. appimage-suffix: aarch64
  40. openjfx-url: 'https://download2.gluonhq.com/openjfx/24.0.1/openjfx-24.0.1_linux-aarch64_bin-jmods.zip'
  41. openjfx-sha: '7e02edd0f4ee5527a27c94b0bbba66fcaaff41009119e45d0eca0f96ddfb6e7b'
  42. steps:
  43. - uses: actions/checkout@v4
  44. - name: Setup Java
  45. uses: actions/setup-java@v4
  46. with:
  47. distribution: ${{ env.JAVA_DIST }}
  48. java-version: ${{ env.JAVA_VERSION }}
  49. check-latest: true
  50. cache: 'maven'
  51. - name: Download OpenJFX jmods
  52. id: download-jmods
  53. run: |
  54. curl -L ${{ matrix.openjfx-url }} -o openjfx-jmods.zip
  55. echo "${{ matrix.openjfx-sha }} openjfx-jmods.zip" | shasum -a256 --check
  56. mkdir -p openjfx-jmods
  57. unzip -j openjfx-jmods.zip \*/javafx.base.jmod \*/javafx.controls.jmod \*/javafx.fxml.jmod \*/javafx.graphics.jmod -d openjfx-jmods
  58. - name: Ensure major jfx version in pom and in jmods is the same
  59. run: |
  60. JMOD_VERSION=$(jmod describe openjfx-jmods/javafx.base.jmod | head -1)
  61. JMOD_VERSION=${JMOD_VERSION#*@}
  62. JMOD_VERSION=${JMOD_VERSION%%.*}
  63. POM_JFX_VERSION=$(mvn help:evaluate "-Dexpression=javafx.version" -q -DforceStdout)
  64. POM_JFX_VERSION=${POM_JFX_VERSION#*@}
  65. POM_JFX_VERSION=${POM_JFX_VERSION%%.*}
  66. if [ $POM_JFX_VERSION -ne $JMOD_VERSION ]; then
  67. >&2 echo "Major JavaFX version in pom.xml (${POM_JFX_VERSION}) != amd64 jmod version (${JMOD_VERSION})"
  68. exit 1
  69. fi
  70. - name: Set version
  71. run : mvn versions:set -DnewVersion=${{ needs.get-version.outputs.semVerStr }}
  72. - name: Run maven
  73. run: mvn -B clean package -Plinux -DskipTests -Djavafx.platform=linux
  74. - name: Patch target dir
  75. run: |
  76. cp LICENSE.txt target
  77. cp target/cryptomator-*.jar target/mods
  78. - name: Run jlink with help option
  79. id: jep-493-check
  80. run: |
  81. JMOD_PATHS="openjfx-jmods"
  82. if ! ${JAVA_HOME}/bin/jlink --help | grep -q "Linking from run-time image enabled"; then
  83. JMOD_PATHS="${JAVA_HOME}/jmods:${JMOD_PATHS}"
  84. fi
  85. echo "jmod_paths=${JMOD_PATHS}" >> "$GITHUB_OUTPUT"
  86. - name: Run jlink
  87. #Remark: no compression is applied for improved build compression later (here appimage)
  88. run: >
  89. ${JAVA_HOME}/bin/jlink
  90. --verbose
  91. --output runtime
  92. --module-path "${{ steps.jep-493-check.outputs.jmod_paths }}"
  93. --add-modules java.base,java.desktop,java.instrument,java.logging,java.naming,java.net.http,java.scripting,java.sql,java.xml,javafx.base,javafx.graphics,javafx.controls,javafx.fxml,jdk.unsupported,jdk.security.auth,jdk.accessibility,jdk.management.jfr,jdk.net,java.compiler
  94. --strip-native-commands
  95. --no-header-files
  96. --no-man-pages
  97. --strip-debug
  98. --compress zip-0
  99. - name: Run jpackage
  100. run: >
  101. ${JAVA_HOME}/bin/jpackage
  102. --verbose
  103. --type app-image
  104. --runtime-image runtime
  105. --input target/libs
  106. --module-path target/mods
  107. --module org.cryptomator.desktop/org.cryptomator.launcher.Cryptomator
  108. --dest appdir
  109. --name Cryptomator
  110. --vendor "Skymatic GmbH"
  111. --copyright "(C) 2016 - 2025 Skymatic GmbH"
  112. --app-version "${{ needs.get-version.outputs.semVerNum }}.${{ needs.get-version.outputs.revNum }}"
  113. --java-options "--enable-preview"
  114. --java-options "--enable-native-access=javafx.graphics,org.cryptomator.jfuse.linux.amd64,org.cryptomator.jfuse.linux.aarch64,org.purejava.appindicator"
  115. --java-options "--sun-misc-unsafe-memory-access=allow"
  116. --java-options "-Xss5m"
  117. --java-options "-Xmx256m"
  118. --java-options "-Dcryptomator.appVersion=\"${{ needs.get-version.outputs.semVerStr }}\""
  119. --java-options "-Dfile.encoding=\"utf-8\""
  120. --java-options "-Djava.net.useSystemProxies=true"
  121. --java-options "-Dcryptomator.logDir=\"@{userhome}/.local/share/Cryptomator/logs\""
  122. --java-options "-Dcryptomator.pluginDir=\"@{userhome}/.local/share/Cryptomator/plugins\""
  123. --java-options "-Dcryptomator.settingsPath=\"@{userhome}/.config/Cryptomator/settings.json:@{userhome}/.Cryptomator/settings.json\""
  124. --java-options "-Dcryptomator.p12Path=\"@{userhome}/.config/Cryptomator/key.p12\""
  125. --java-options "-Dcryptomator.ipcSocketPath=\"@{userhome}/.config/Cryptomator/ipc.socket\""
  126. --java-options "-Dcryptomator.mountPointsDir=\"@{userhome}/.local/share/Cryptomator/mnt\""
  127. --java-options "-Dcryptomator.showTrayIcon=true"
  128. --java-options "-Dcryptomator.integrationsLinux.trayIconsDir=\"@{appdir}/usr/share/icons/hicolor/symbolic/apps\""
  129. --java-options "-Dcryptomator.buildNumber=\"appimage-${{ needs.get-version.outputs.revNum }}\""
  130. --java-options "-Dcryptomator.networking.truststore.p12Path=\"/etc/cryptomator/certs.p12\""
  131. --resource-dir dist/linux/resources
  132. - name: Patch Cryptomator.AppDir
  133. run: |
  134. mv appdir/Cryptomator Cryptomator.AppDir
  135. cp -r dist/linux/appimage/resources/AppDir/* Cryptomator.AppDir/
  136. cp dist/linux/common/org.cryptomator.Cryptomator256.png Cryptomator.AppDir/usr/share/icons/hicolor/256x256/apps/org.cryptomator.Cryptomator.png
  137. cp dist/linux/common/org.cryptomator.Cryptomator512.png Cryptomator.AppDir/usr/share/icons/hicolor/512x512/apps/org.cryptomator.Cryptomator.png
  138. cp dist/linux/common/org.cryptomator.Cryptomator.svg Cryptomator.AppDir/usr/share/icons/hicolor/scalable/apps/org.cryptomator.Cryptomator.svg
  139. cp dist/linux/common/org.cryptomator.Cryptomator.tray.svg Cryptomator.AppDir/usr/share/icons/hicolor/scalable/apps/org.cryptomator.Cryptomator.tray.svg
  140. cp dist/linux/common/org.cryptomator.Cryptomator.tray-unlocked.svg Cryptomator.AppDir/usr/share/icons/hicolor/scalable/apps/org.cryptomator.Cryptomator.tray-unlocked.svg
  141. cp dist/linux/common/org.cryptomator.Cryptomator.tray.svg Cryptomator.AppDir/usr/share/icons/hicolor/symbolic/apps/org.cryptomator.Cryptomator.tray-symbolic.svg
  142. cp dist/linux/common/org.cryptomator.Cryptomator.tray-unlocked.svg Cryptomator.AppDir/usr/share/icons/hicolor/symbolic/apps/org.cryptomator.Cryptomator.tray-unlocked-symbolic.svg
  143. cp dist/linux/common/org.cryptomator.Cryptomator.metainfo.xml Cryptomator.AppDir/usr/share/metainfo/org.cryptomator.Cryptomator.metainfo.xml
  144. cp dist/linux/common/org.cryptomator.Cryptomator.desktop Cryptomator.AppDir/usr/share/applications/org.cryptomator.Cryptomator.desktop
  145. cp dist/linux/common/application-vnd.cryptomator.vault.xml Cryptomator.AppDir/usr/share/mime/packages/application-vnd.cryptomator.vault.xml
  146. ln -s usr/share/icons/hicolor/scalable/apps/org.cryptomator.Cryptomator.svg Cryptomator.AppDir/org.cryptomator.Cryptomator.svg
  147. ln -s usr/share/icons/hicolor/scalable/apps/org.cryptomator.Cryptomator.svg Cryptomator.AppDir/.DirIcon
  148. ln -s usr/share/applications/org.cryptomator.Cryptomator.desktop Cryptomator.AppDir/org.cryptomator.Cryptomator.desktop
  149. ln -s org.cryptomator.Cryptomator.metainfo.xml Cryptomator.AppDir/usr/share/metainfo/org.cryptomator.Cryptomator.appdata.xml
  150. ln -s bin/cryptomator.sh Cryptomator.AppDir/AppRun
  151. - name: Download AppImageKit
  152. run: |
  153. curl -L https://github.com/AppImage/appimagetool/releases/download/continuous/appimagetool-${{ matrix.appimage-suffix }}.AppImage -o appimagetool.AppImage
  154. chmod +x appimagetool.AppImage
  155. ./appimagetool.AppImage --appimage-extract
  156. - name: Prepare GPG-Agent for signing with key 615D449FE6E6A235
  157. run: |
  158. echo "${GPG_PRIVATE_KEY}" | gpg --batch --quiet --import
  159. echo "${GPG_PASSPHRASE}" | gpg --batch --quiet --passphrase-fd 0 --pinentry-mode loopback -u 615D449FE6E6A235 --dry-run --sign README.md
  160. env:
  161. GPG_PRIVATE_KEY: ${{ secrets.RELEASES_GPG_PRIVATE_KEY }}
  162. GPG_PASSPHRASE: ${{ secrets.RELEASES_GPG_PASSPHRASE }}
  163. - name: Build AppImage
  164. run: >
  165. ./squashfs-root/AppRun Cryptomator.AppDir cryptomator-${{ needs.get-version.outputs.semVerStr }}-${{ matrix.appimage-suffix }}.AppImage
  166. -u "gh-releases-zsync|cryptomator|cryptomator|latest|cryptomator-*-${{ matrix.appimage-suffix }}.AppImage.zsync"
  167. --sign --sign-key=615D449FE6E6A235
  168. - name: Create detached GPG signatures
  169. run: |
  170. gpg --batch --quiet --passphrase-fd 0 --pinentry-mode loopback -u 615D449FE6E6A235 --detach-sign -a cryptomator-*.AppImage
  171. gpg --batch --quiet --passphrase-fd 0 --pinentry-mode loopback -u 615D449FE6E6A235 --detach-sign -a cryptomator-*.AppImage.zsync
  172. - name: Upload artifacts
  173. uses: actions/upload-artifact@v4
  174. with:
  175. name: appimage-${{ matrix.appimage-suffix }}
  176. path: |
  177. cryptomator-*.AppImage
  178. cryptomator-*.AppImage.zsync
  179. cryptomator-*.asc
  180. if-no-files-found: error
  181. - name: Publish AppImage on GitHub Releases
  182. if: startsWith(github.ref, 'refs/tags/') && github.event.action == 'published'
  183. uses: softprops/action-gh-release@v2
  184. with:
  185. fail_on_unmatched_files: true
  186. token: ${{ secrets.CRYPTOBOT_RELEASE_TOKEN }}
  187. files: |
  188. cryptomator-*.AppImage
  189. cryptomator-*.zsync
  190. cryptomator-*.asc
  191. create-aur-bin-pr:
  192. name: Create PR for aur-bin repo
  193. needs: [build, get-version]
  194. runs-on: ubuntu-latest
  195. if: github.event_name == 'release'
  196. steps:
  197. - name: Download AppImages
  198. uses: actions/download-artifact@v4
  199. with:
  200. path: downloads/
  201. merge-multiple: true
  202. - name: Compute sha256 hash of AppImages
  203. id: checksums
  204. run: |
  205. X64_SHA256=$(sha256sum downloads/cryptomator-*-x86_64.AppImage | cut -d ' ' -f1)
  206. echo "x64-sha256sum=${X64_SHA256}" >> "$GITHUB_OUTPUT"
  207. AARCH64_SHA256=$(sha256sum downloads/cryptomator-*-aarch64.AppImage | cut -d ' ' -f1)
  208. echo "aarch64-sha256sum=${AARCH64_SHA256}" >> "$GITHUB_OUTPUT"
  209. - uses: actions/checkout@v4
  210. with:
  211. repository: 'cryptomator/aur-bin'
  212. token: ${{ secrets.CRYPTOBOT_PR_TOKEN }}
  213. - name: Install dependencies
  214. run: |
  215. sudo apt-get update
  216. sudo apt-get -y install makepkg pacman-package-manager
  217. - name: Checkout release branch
  218. run: |
  219. git checkout -b release/${{ needs.get-version.outputs.semVerStr }}
  220. - name: Update build file
  221. run: |
  222. sed -i -e 's|^pkgver=.*$|pkgver=${{ needs.get-version.outputs.semVerStr }}|' PKGBUILD
  223. sed -i -e 's|^pkgrel=.*$|pkgrel=1|' PKGBUILD
  224. sed -i -e "s|^sha256sums_x86_64=.*$|sha256sums_x86_64=('${{ steps.checksums.outputs.x64-sha256sum }}'|" PKGBUILD
  225. sed -i -e "s|^sha256sums_aarch64=.*$|sha256sums_aarch64=('${{ steps.checksums.outputs.aarch64-sha256sum}}'|" PKGBUILD
  226. makepkg --printsrcinfo > .SRCINFO
  227. - name: Commit and push
  228. run: |
  229. git config user.name "${{ github.actor }}"
  230. git config user.email "${{ github.actor_id }}+${{ github.actor }}@users.noreply.github.com"
  231. git config push.autoSetupRemote true
  232. git stage .
  233. git commit -m "Prepare release ${{needs.get-version.outputs.semVerStr}}"
  234. git push
  235. - name: Create pull request
  236. id: create-pr
  237. run: |
  238. printf "> [!IMPORTANT]\n> Todos:\n> - [ ] Update build instructions\n> - [ ] Check for JDK update\n> - [ ] Check for JFX update" > pr_body.md
  239. URL=$(gh pr create --title "Release ${{ needs.get-version.outputs.semVerStr }}" --body-file pr_body.md)
  240. echo "PR_URL=$URL" >> "$GITHUB_OUTPUT"
  241. env:
  242. GH_TOKEN: ${{ secrets.CRYPTOBOT_PR_TOKEN }}
  243. - name: Slack Notification
  244. uses: rtCamp/action-slack-notify@v2
  245. env:
  246. SLACK_WEBHOOK: ${{ secrets.SLACK_WEBHOOK_URL }}
  247. SLACK_USERNAME: 'Cryptobot'
  248. SLACK_ICON: false
  249. SLACK_ICON_EMOJI: ':bot:'
  250. SLACK_CHANNEL: 'cryptomator-desktop'
  251. SLACK_TITLE: "AUR-bin release PR for ${{ github.event.repository.name }} ${{ github.event.release.tag_name }} created."
  252. SLACK_MESSAGE: "See <${{ steps.create-pr.outputs.PR_URL }}|PR> on how to proceed."
  253. SLACK_FOOTER: false
  254. MSG_MINIMAL: true