From b31c3913477a7a47644851803075eabe1aacc08b Mon Sep 17 00:00:00 2001 From: hrb-hub <181954414+hrb-hub@users.noreply.github.com> Date: Wed, 13 Nov 2024 14:34:11 +0100 Subject: [PATCH] WIP: Reorganize build/release pipelines Part of #6830 Co-authored-by: paw --- app-ios/fastlane/Fastfile | 28 +- ci/Android.Jenkinsfile | 69 +--- ci/Desktop.Jenkinsfile | 237 ++++++++------ ci/Ios.Jenkinsfile | 71 +---- ...ublish-CalendarMobileArtifacts.Jenkinsfile | 4 +- ci/Publish-MailDesktopArtifacts.Jenkinsfile | 177 +++++++++++ ci/Publish-MailMobileArtifacts.Jenkinsfile | 299 ++++++++++++++++++ ci/Release.Jenkinsfile | 142 ++++++--- ci/jenkins-lib/util.groovy | 13 +- 9 files changed, 744 insertions(+), 296 deletions(-) create mode 100644 ci/Publish-MailDesktopArtifacts.Jenkinsfile create mode 100644 ci/Publish-MailMobileArtifacts.Jenkinsfile diff --git a/app-ios/fastlane/Fastfile b/app-ios/fastlane/Fastfile index a873aea71fce..072c70f0f7f4 100644 --- a/app-ios/fastlane/Fastfile +++ b/app-ios/fastlane/Fastfile @@ -16,8 +16,8 @@ default_platform(:ios) platform :ios do - desc "Push a new prod release to AppStore" - lane :appstore_prod do |options| + desc "Build a new Mail prod release with AppStore configuration" + lane :build_mail_prod do |options| match( app_identifier: ["de.tutao.tutanota", "de.tutao.tutanota.TutanotaShareExtension", "de.tutao.tutanota.TutanotaNotificationExtension"], type: "appstore", @@ -37,18 +37,22 @@ platform :ios do include_symbols: true, verbose: true ) - if options[:submit] - upload_to_app_store( - skip_screenshots: true, - submit_for_review: false, - precheck_include_in_app_purchases: false, - # must use force as long as we don't automatically create html previews - force: true, - api_key_path: ENV["API_KEY_JSON_FILE_PATH"] - ) - end end + desc "Publish a Mail artifact to AppStore" + lane :publish_mail_prod do |options| + sh 'echo "Uploading mail artifact ' + options[:file] + '"' + + upload_to_app_store( + skip_screenshots: true, + submit_for_review: false, + precheck_include_in_app_purchases: false, + # must use force as long as we don't automatically create html previews + force: true, + api_key_path: ENV["API_KEY_JSON_FILE_PATH"] + ) + end + desc "Build a new prod release for ad-hoc" lane :adhoc_prod do |options| match( diff --git a/ci/Android.Jenkinsfile b/ci/Android.Jenkinsfile index 49cf94df7125..c06f089aec76 100644 --- a/ci/Android.Jenkinsfile +++ b/ci/Android.Jenkinsfile @@ -5,7 +5,6 @@ pipeline { PATH = "${env.NODE_PATH}:${env.PATH}:/home/jenkins/emsdk/upstream/bin/:/home/jenkins/emsdk/:/home/jenkins/emsdk/upstream/emscripten" ANDROID_SDK_ROOT = "/opt/android-sdk-linux" ANDROID_HOME = "/opt/android-sdk-linux" - GITHUB_RELEASE_PAGE = "https://github.com/tutao/tutanota/releases/tag/tutanota-android-release-${VERSION}" } agent { @@ -19,15 +18,8 @@ pipeline { parameters { booleanParam( name: 'RELEASE', defaultValue: false, - description: "Build a test and release version of the app. " + - "Uploads both to Nexus and creates a new release on google play, " + - "which must be manually published from play.google.com/console" + description: "Build a test and release version of the app. Uploads both to Nexus." ) - persistentText( - name: "releaseNotes", - defaultValue: "", - description: "release notes for this build" - ) } stages { @@ -124,26 +116,6 @@ pipeline { assetFilePath: "${WORKSPACE}/build/app-android/tutanota-app-tutao-releaseTest-${VERSION}.apk", fileExtension: 'apk' ) - - catchError(stageResult: 'UNSTABLE', buildResult: 'SUCCESS', message: 'Failed to upload android test app to Play Store') { - // This doesn't publish to the main app on play store, - // instead it gets published to the hidden "tutanota-test" app - // this happens because the AppId is set to de.tutao.tutanota.test by the android build - // and play store knows which app to publish just based on the id - androidApkUpload( - googleCredentialsId: 'android-app-publisher-credentials', - apkFilesPattern: "build/app-android/tutanota-app-tutao-releaseTest-${VERSION}.apk", - trackName: 'internal', - rolloutPercentage: '100%', - recentChangeList: [ - [ - language: "en-US", - text : "see: ${GITHUB_RELEASE_PAGE}" - ] - ] - ) // androidApkUpload - } // catchError - } } } // stage testing @@ -163,49 +135,10 @@ pipeline { assetFilePath: "${WORKSPACE}/${filePath}", fileExtension: 'apk' ) - - androidApkUpload( - googleCredentialsId: 'android-app-publisher-credentials', - apkFilesPattern: "${filePath}", - trackName: 'production', - // Don't publish the app to users directly - // It will require manual intervention at play.google.com/console - rolloutPercentage: '0%', - recentChangeList: [ - [ - language: "en-US", - text : "see: ${GITHUB_RELEASE_PAGE}" - ] - ] - ) } } } // stage production } } - stage('Tag and publish release page') { - when { - expression { return params.RELEASE } - } - steps { - // Needed to upload it - unstash 'apk-production' - - script { - def filePath = "build/app-android/tutanota-app-tutao-release-${VERSION}.apk" - - writeFile file: "notes.txt", text: params.releaseNotes - catchError(stageResult: 'UNSTABLE', buildResult: 'SUCCESS', message: 'Failed to create github release page for android') { - withCredentials([string(credentialsId: 'github-access-token', variable: 'GITHUB_TOKEN')]) { - sh """node buildSrc/createReleaseDraft.js --name '${VERSION} (Android)' \ - --tag 'tutanota-android-release-${VERSION}' \ - --uploadFile '${WORKSPACE}/${filePath}' \ - --notes notes.txt""" - } // withCredentials - } // catchError - sh "rm notes.txt" - } // script - } - } } } diff --git a/ci/Desktop.Jenkinsfile b/ci/Desktop.Jenkinsfile index 2908a83c4abe..64d560de171f 100644 --- a/ci/Desktop.Jenkinsfile +++ b/ci/Desktop.Jenkinsfile @@ -10,15 +10,25 @@ pipeline { parameters { booleanParam( - name: 'RELEASE', + name: 'UPLOAD', defaultValue: false, - description: "Prepare a release version (doesn't publish to production, this is done manually)" + description: "Upload built clients to Nexus" + ) + booleanParam( + name: 'WINDOWS', + defaultValue: false, + description: "Build Windows client" + ) + booleanParam( + name: 'MAC', + defaultValue: false, + description: "Build Mac client" + ) + booleanParam( + name: 'LINUX', + defaultValue: false, + description: "Build Linux client" ) - persistentText( - name: "releaseNotes", - defaultValue: "", - description: "release notes for this build" - ) } agent { @@ -26,6 +36,17 @@ pipeline { } stages { + stage("Checking params") { + steps { + script{ + if(!params.WINDOWS && !params.MAC && !params.LINUX) { + currentBuild.result = 'ABORTED' + error('No artifacts were selected.') + } + } + echo "Params OKAY" + } + } stage('Check Github') { steps { script { @@ -57,6 +78,7 @@ pipeline { } stage('Native modules') { + when { expression { return params.WINDOWS } } agent { label 'win-native' } @@ -73,6 +95,7 @@ pipeline { stage('Build desktop clients') { parallel { stage('Windows') { + when { expression { return params.WINDOWS } } environment { PATH = "${env.NODE_PATH}:${env.PATH}" } @@ -101,6 +124,7 @@ pipeline { } stage('Mac') { + when { expression { return params.MAC } } environment { PATH = "${env.NODE_MAC_PATH}:${env.PATH}" } @@ -117,14 +141,14 @@ pipeline { ]) { sh 'security unlock-keychain -p $FASTLANE_KEYCHAIN_PASSWORD' script { - def stage = params.RELEASE ? 'release' : 'prod' + def stage = params.UPLOAD ? 'release' : 'prod' sh ''' export APPLEID=${APPLEIDVAR}; export APPLEIDPASS=${APPLEIDPASSVAR}; export APPLETEAMID=${APPLETEAMIDVAR}; node desktop --existing --architecture universal --platform mac ''' + "${stage}" dir('artifacts') { - if (params.RELEASE) { + if (params.UPLOAD) { stash includes: 'desktop-test/*', name:'mac_installer_test' } stash includes: 'desktop/*', name:'mac_installer' @@ -135,6 +159,7 @@ pipeline { } stage('Linux') { + when { expression { return params.LINUX } } agent { dockerfile { filename 'linux-build.dockerfile' @@ -158,8 +183,7 @@ pipeline { } } - stage('Preparation for build deb and publish') { - when { expression { return params.RELEASE } } + stage('Preparation for sign clients and upload to Nexus') { agent { label 'master' } @@ -170,8 +194,8 @@ pipeline { } } } - stage('Build deb and publish') { - when { expression { return params.RELEASE } } + stage('Sign clients and upload to Nexus') { + when { expression { return params.UPLOAD } } agent { dockerfile { filename 'linux-build.dockerfile' @@ -181,95 +205,116 @@ pipeline { args "--network host -v /run:/run:rw,z -v /opt/repository:/opt/repository:rw,z --device=${env.DEVICE_PATH}" } // docker } - environment { PATH = "${env.NODE_PATH}:${env.PATH}" } - steps { - sh 'npm ci' - sh 'npm run build-packages' - sh 'rm -rf ./build/*' - - dir('build') { - unstash 'linux_installer' - unstash 'mac_installer' - unstash 'win_installer' - unstash 'linux_installer_test' - unstash 'mac_installer_test' - unstash 'win_installer_test' - } - - withCredentials([string(credentialsId: 'HSM_USER_PIN', variable: 'PW')]) { - sh '''export HSM_USER_PIN=${PW}; node buildSrc/signDesktopClients.js''' - } + environment { + PATH = "${env.NODE_PATH}:${env.PATH}" + } + stages { + stage('Preparation for sign and upload') { + steps { + sh 'npm ci' + sh 'npm run build-packages' + sh 'rm -rf ./build/*' + } + } + stage('Sign and upload') { + parallel { + stage('Windows') { + when { expression { return params.WINDOWS } } + steps { + dir('build') { + unstash 'win_installer' + unstash 'win_installer_test' + } - sh 'node buildSrc/publish.js desktop' + withCredentials([string(credentialsId: 'HSM_USER_PIN', variable: 'PW')]) { + sh '''export HSM_USER_PIN=${PW}; node buildSrc/signDesktopClients.js''' + } - script { // create release draft - def desktopLinux = "build/desktop/tutanota-desktop-linux.AppImage" - def desktopWin = "build/desktop/tutanota-desktop-win.exe" - def desktopMac = "build/desktop/tutanota-desktop-mac.dmg" + script { + def util = load "ci/jenkins-lib/util.groovy" + util.publishToNexus( + groupId: "app", + artifactId: "desktop-win-test", + version: "${VERSION}", + assetFilePath: "${WORKSPACE}/build/desktop-test/tutanota-desktop-test-win.exe", + fileExtension: 'exe' + ) + util.publishToNexus( + groupId: "app", + artifactId: "desktop-win", + version: "${VERSION}", + assetFilePath: "${WORKSPACE}/build/desktop/tutanota-desktop-win.exe", + fileExtension: 'exe' + ) + } + } + } // windows + stage('Mac') { + when { expression { return params.MAC } } + steps { + dir('build') { + unstash 'mac_installer' + unstash 'mac_installer_test' + } - writeFile file: "notes.txt", text: params.releaseNotes - catchError(stageResult: 'UNSTABLE', buildResult: 'SUCCESS', message: 'Failed to create github release page for desktop') { - withCredentials([string(credentialsId: 'github-access-token', variable: 'GITHUB_TOKEN')]) { - sh """node buildSrc/createReleaseDraft.js --name '${VERSION} (Desktop)' \ - --tag 'tutanota-desktop-release-${VERSION}' \ - --uploadFile '${WORKSPACE}/${desktopLinux}' \ - --uploadFile '${WORKSPACE}/${desktopWin}' \ - --uploadFile '${WORKSPACE}/${desktopMac}' \ - --notes notes.txt""" - } // withCredentials - } // catchError - sh "rm notes.txt" - } // script release draft + withCredentials([string(credentialsId: 'HSM_USER_PIN', variable: 'PW')]) { + sh '''export HSM_USER_PIN=${PW}; node buildSrc/signDesktopClients.js''' + } - script { // upload to nexus - def util = load "ci/jenkins-lib/util.groovy" + script { + def util = load "ci/jenkins-lib/util.groovy" + util.publishToNexus( + groupId: "app", + artifactId: "desktop-mac-test", + version: "${VERSION}", + assetFilePath: "${WORKSPACE}/build/desktop-test/tutanota-desktop-test-mac.dmg", + fileExtension: 'dmg' + ) + util.publishToNexus( + groupId: "app", + artifactId: "desktop-mac", + version: "${VERSION}", + assetFilePath: "${WORKSPACE}/build/desktop/tutanota-desktop-mac.dmg", + fileExtension: 'dmg' + ) + } + } + } // mac + stage('Linux') { + when { expression { return params.LINUX } } + steps { + dir('build') { + unstash 'linux_installer' + unstash 'linux_installer_test' + } - util.publishToNexus( - groupId: "app", - artifactId: "desktop-linux-test", - version: "${VERSION}", - assetFilePath: "${WORKSPACE}/build/desktop-test/tutanota-desktop-test-linux.AppImage", - fileExtension: 'AppImage' - ) - util.publishToNexus( - groupId: "app", - artifactId: "desktop-win-test", - version: "${VERSION}", - assetFilePath: "${WORKSPACE}/build/desktop-test/tutanota-desktop-test-win.exe", - fileExtension: 'exe' - ) - util.publishToNexus( - groupId: "app", - artifactId: "desktop-mac-test", - version: "${VERSION}", - assetFilePath: "${WORKSPACE}/build/desktop-test/tutanota-desktop-test-mac.dmg", - fileExtension: 'dmg' - ) - util.publishToNexus( - groupId: "app", - artifactId: "desktop-linux", - version: "${VERSION}", - assetFilePath: "${WORKSPACE}/build/desktop/tutanota-desktop-linux.AppImage", - fileExtension: 'AppImage' - ) - util.publishToNexus( - groupId: "app", - artifactId: "desktop-win", - version: "${VERSION}", - assetFilePath: "${WORKSPACE}/build/desktop/tutanota-desktop-win.exe", - fileExtension: 'exe' - ) - util.publishToNexus( - groupId: "app", - artifactId: "desktop-mac", - version: "${VERSION}", - assetFilePath: "${WORKSPACE}/build/desktop/tutanota-desktop-mac.dmg", - fileExtension: 'dmg' - ) - } // script upload to nexus + withCredentials([string(credentialsId: 'HSM_USER_PIN', variable: 'PW')]) { + sh '''export HSM_USER_PIN=${PW}; node buildSrc/signDesktopClients.js''' + } - } // steps - } // stage build deb & publish + script { + def util = load "ci/jenkins-lib/util.groovy" + util.publishToNexus( + groupId: "app", + artifactId: "desktop-linux-test", + version: "${VERSION}", + assetFilePath: "${WORKSPACE}/build/desktop-test/tutanota-desktop-test-linux.AppImage", + fileExtension: 'AppImage' + ) + util.publishToNexus( + groupId: "app", + artifactId: "desktop-linux", + version: "${VERSION}", + assetFilePath: "${WORKSPACE}/build/desktop/tutanota-desktop-linux.AppImage", + fileExtension: 'AppImage' + ) + } + } + } // linux + } // parallel + } // stage sign and upload + } // stages + } // stage sign clients and upload to Nexus } // stages } // pipeline diff --git a/ci/Ios.Jenkinsfile b/ci/Ios.Jenkinsfile index 3f3ec3c1a47d..3d0a098463bf 100644 --- a/ci/Ios.Jenkinsfile +++ b/ci/Ios.Jenkinsfile @@ -2,7 +2,6 @@ pipeline { environment { NODE_MAC_PATH = "/usr/local/opt/node@20/bin/" VERSION = sh(returnStdout: true, script: "${env.NODE_PATH}/node -p -e \"require('./package.json').version\" | tr -d \"\n\"") - RELEASE_NOTES_PATH = "app-ios/fastlane/metadata/default/release_notes.txt" } agent { @@ -13,8 +12,8 @@ pipeline { booleanParam( name: 'RELEASE', defaultValue: false, - description: "Build testing and production version, and upload them to nexus/testflight/appstore. " + - "The production version will need to be released manually from appstoreconnect.apple.com." + description: "Upload staging/prod to Nexus and send staging version to testflight. " + + "The production version must be sent to appstore using the publish job" ) booleanParam( name: 'PROD', @@ -24,11 +23,6 @@ pipeline { name: 'STAGING', defaultValue: true ) - persistentText( - name: "releaseNotes", - defaultValue: "", - description: "release notes for this build" - ) } stages { @@ -99,10 +93,11 @@ pipeline { generateXCodeProjects() util.runFastlane("de.tutao.tutanota", "adhoc_prod") if (params.RELEASE) { - writeReleaseNotesForAppStore() - util.runFastlane("de.tutao.tutanota", "appstore_prod submit:true") + util.runFastlane("de.tutao.tutanota", "build_mail_prod") + stash includes: "app-ios/releases/tutanota-${VERSION}.ipa", name: 'ipa-production' + } else { + stash includes: "app-ios/releases/tutanota-${VERSION}-adhoc.ipa", name: 'ipa-production' } - stash includes: "app-ios/releases/tutanota-${VERSION}-adhoc.ipa", name: 'ipa-production' } } } @@ -131,42 +126,12 @@ pipeline { if (params.PROD) { unstash 'ipa-production' catchError(stageResult: 'UNSTABLE', buildResult: 'SUCCESS', message: 'There was an error when uploading to Nexus') { - publishToNexus("ios", "tutanota-${VERSION}-adhoc.ipa") + publishToNexus("ios", "tutanota-${VERSION}.ipa") } } } } } - - stage('Tag and create github release page') { - environment { - PATH = "${env.NODE_PATH}:${env.PATH}" - } - when { - expression { return params.RELEASE } - } - agent { - label 'linux' - } - steps { - script { - - catchError(stageResult: 'UNSTABLE', buildResult: 'SUCCESS', message: 'Failed to create github release page for ios') { - def tag = "tutanota-ios-release-${VERSION}" - // need to run npm ci to install dependencies of releaseNotes.js - sh "npm ci" - - writeFile file: "notes.txt", text: params.releaseNotes - withCredentials([string(credentialsId: 'github-access-token', variable: 'GITHUB_TOKEN')]) { - sh """node buildSrc/createReleaseDraft.js --name '${VERSION} (iOS)' \ - --tag 'tutanota-ios-release-${VERSION}' \ - --notes notes.txt""" - } // withCredentials - sh "rm notes.txt" - } // catchError - } - } - } } } @@ -174,7 +139,7 @@ void stubClientDirectory() { script { sh "pwd" sh "echo $PATH" - sh "mkdir build-calendar-app" + sh "mkdir build-calendar-app" sh "mkdir build" } } @@ -209,26 +174,6 @@ void generateCalendarProject() { generateXCodeProject("app-ios", "calendar-project") } - -void writeReleaseNotesForAppStore() { - script { - catchError(stageResult: 'UNSTABLE', buildResult: 'SUCCESS', message: 'Failed to create github release notes for ios') { - // need to run npm ci to install dependencies of releaseNotes.js - sh "npm ci" - writeFile file: "notes.txt", text: params.releaseNotes - withCredentials([string(credentialsId: 'github-access-token', variable: 'GITHUB_TOKEN')]) { - sh """node buildSrc/createReleaseDraft.js --name '${VERSION} (iOS)' \ - --tag 'tutanota-ios-release-${VERSION}'\ - --notes notes.txt \ - --toFile ${RELEASE_NOTES_PATH}""" - } - sh "rm notes.txt" - } - } - - sh "echo Created release notes for fastlane ${RELEASE_NOTES_PATH}" -} - void publishToNexus(String artifactId, String ipaFileName) { def util = load "ci/jenkins-lib/util.groovy" util.publishToNexus(groupId: "app", diff --git a/ci/Publish-CalendarMobileArtifacts.Jenkinsfile b/ci/Publish-CalendarMobileArtifacts.Jenkinsfile index 99d2c2d33ca7..864bfab981c0 100644 --- a/ci/Publish-CalendarMobileArtifacts.Jenkinsfile +++ b/ci/Publish-CalendarMobileArtifacts.Jenkinsfile @@ -128,12 +128,12 @@ pipeline { } steps { script { - catchError(stageResult: 'UNSTABLE', buildResult: 'SUCCESS', message: 'Failed to upload android app to GitHub') { + catchError(stageResult: 'UNSTABLE', buildResult: 'SUCCESS', message: 'Failed to upload iOS app to GitHub') { writeReleaseNotes("ios", "iOS", "${env.VERSION}", "") } } // script } // steps - } // stage Android App + } // stage iOS App } } stage("Publishing Artifacts to Stores") { diff --git a/ci/Publish-MailDesktopArtifacts.Jenkinsfile b/ci/Publish-MailDesktopArtifacts.Jenkinsfile new file mode 100644 index 000000000000..e4fc3a41d397 --- /dev/null +++ b/ci/Publish-MailDesktopArtifacts.Jenkinsfile @@ -0,0 +1,177 @@ +pipeline { + environment { + // on m1 macs, this is a symlink that must be updated. see wiki. + VERSION = sh(returnStdout: true, script: "${env.NODE_PATH}/node -p -e \"require('./package.json').version\" | tr -d \"\n\"") + TMPDIR ='/tmp' + LINUX_IMAGE_PATH = 'build/desktop/tutanota-desktop-linux.AppImage' + } + + parameters { + booleanParam( + name: 'DEB', + defaultValue: false, + description: "build deb client" + ) + booleanParam( + name: 'PUBLISH_NOTES', + defaultValue: false, + description: "publish release notes draft" + ) + persistentText( + name: 'releaseNotes', + defaultValue: '', + description: "release notes for this build" + ) + } + + agent { + label 'master' + } + + stages { + stage('Check Github') { + steps { + script { + def util = load "ci/jenkins-lib/util.groovy" + util.checkGithub() + } + } + } // check github + stage('Preparation for build deb and publish notes') { + agent { + label 'master' + } + steps { + script { + def devicePath = sh(script: 'lsusb | grep Nitro | sed -nr \'s|Bus (.*) Device ([^:]*):.*|/dev/bus/usb/\\1/\\2|p\'', returnStdout: true).trim() + env.DEVICE_PATH = devicePath + } + } + } // preparation for build deb and publish notes + stage ('Build deb and publish release notes draft') { + when { expression { return params.PUBLISH_NOTES || params.DEB } } + agent { + dockerfile { + filename 'linux-build.dockerfile' + label 'master' + dir 'ci/containers' + additionalBuildArgs '--format docker' + args "--network host -v /run:/run:rw,z -v /opt/repository:/opt/repository:rw,z --device=${env.DEVICE_PATH}" + } // docker + } // agent + stages { + stage('Publish release notes draft') { + when { expression { return params.PUBLISH_NOTES } } + steps { + script { + def desktopLinux = env.LINUX_IMAGE_PATH + def desktopWin = "build/desktop/tutanota-desktop-win.exe" + def desktopMac = "build/desktop/tutanota-desktop-mac.dmg" + + def util = load "ci/jenkins-lib/util.groovy" + + util.downloadFromNexus( groupId: "app", + artifactId: "desktop-win", + version: "${VERSION}", + outFile: "${WORKSPACE}/${desktopWin}", + fileExtension: 'exe') + if (!fileExists("${desktopWin}")) { + currentBuild.result = 'ABORTED' + error("Unable to find file ${desktopWin}") + } + + util.downloadFromNexus( groupId: "app", + artifactId: "desktop-mac", + version: "${VERSION}", + outFile: "${WORKSPACE}/${desktopMac}", + fileExtension: 'dmg') + if (!fileExists("${desktopMac}")) { + currentBuild.result = 'ABORTED' + error("Unable to find file ${desktopMac}") + } + + util.downloadFromNexus( groupId: "app", + artifactId: "desktop-linux", + version: "${VERSION}", + outFile: "${WORKSPACE}/${desktopLinux}", + fileExtension: 'AppImage') + if (!fileExists("${desktopLinux}")) { + currentBuild.result = 'ABORTED' + error("Unable to find file ${desktopLinux}") + } + + writeFile file: "notes.txt", text: params.releaseNotes + catchError(stageResult: 'UNSTABLE', buildResult: 'SUCCESS', message: 'Failed to create github release page for desktop') { + withCredentials([string(credentialsId: 'github-access-token', variable: 'GITHUB_TOKEN')]) { + sh """node buildSrc/createReleaseDraft.js --name '${VERSION} (Desktop)' \ + --tag 'tutanota-desktop-release-${VERSION}' \ + --uploadFile '${WORKSPACE}/${desktopLinux}' \ + --uploadFile '${WORKSPACE}/${desktopWin}' \ + --uploadFile '${WORKSPACE}/${desktopMac}' \ + --notes notes.txt""" + } // withCredentials + } // catchError + sh "rm notes.txt" + + stash includes: desktopLinux, name: 'linux_image' + } // script release draft + } // steps + } // publish + stage('Build webapp') { + when { expression { return params.DEB } } + steps { + sh 'npm ci' + sh 'npm run build-packages' + sh 'node webapp.js release' + + // excluding web-specific and mobile specific parts which we don't need in desktop + stash includes: 'build/**', excludes: '**/braintree.html, **/index.html, **/app.html, **/desktop.html, **/index-index.js, **/index-app.js, **/index-desktop.js, **/sw.js', name: 'web_base' + } + } + stage('Build deb') { + when { expression { return params.DEB } } + steps { + script { + def desktopLinux = env.LINUX_IMAGE_PATH + def desktopLinuxTest = "build/desktop-test/tutanota-desktop-test-linux.AppImage" + + def util = load "ci/jenkins-lib/util.groovy" + + if (params.PUBLISH_NOTES) { + unstash 'linux_image' + } else { + util.downloadFromNexus( groupId: "app", + artifactId: "desktop-linux", + version: "${VERSION}", + outFile: "${WORKSPACE}/${desktopLinux}", + fileExtension: 'AppImage') + } + if (!fileExists("${desktopLinux}")) { + currentBuild.result = 'ABORTED' + error("Unable to find file ${desktopLinux}") + } + + util.downloadFromNexus( groupId: "app", + artifactId: "desktop-linux-test", + version: "${VERSION}", + outFile: "${WORKSPACE}/${desktopLinuxTest}", + fileExtension: 'AppImage') + if (!fileExists("${desktopLinuxTest}")) { + currentBuild.result = 'ABORTED' + error("Unable to find file ${desktopLinuxTest}") + } + } // script build deb + + sh 'node -v' + sh 'npm -v' + sh 'npm ci' + sh 'npm run build-packages' + unstash 'web_base' + sh 'node buildSrc/publish.js desktop' + sh 'rm -rf ./build/*' + } // steps build deb + } // build deb + } // stages + } // build deb and publish notes + } // stages +} // pipeline \ No newline at end of file diff --git a/ci/Publish-MailMobileArtifacts.Jenkinsfile b/ci/Publish-MailMobileArtifacts.Jenkinsfile new file mode 100644 index 000000000000..3bc621febe24 --- /dev/null +++ b/ci/Publish-MailMobileArtifacts.Jenkinsfile @@ -0,0 +1,299 @@ +import groovy.transform.Field +@Field def releaseNotes + +pipeline { + environment { + PATH="${env.NODE_PATH}:${env.PATH}" + VERSION = sh(returnStdout: true, script: "${env.NODE_PATH}/node -p -e \"require('./package.json').version\" | tr -d \"\n\"") + IOS_RELEASE_NOTES_PATH = "app-ios/fastlane/metadata/default/release_notes.txt" + } + + parameters { + booleanParam( + name: 'googlePlayStore', + defaultValue: false, + description: "Uploads android artifacts (apk) to Google PlayStore as a Draft on the public track." + ) + booleanParam( + name: 'appleAppStore', + defaultValue: false, + description: "Uploads iOS artifacts to Apple App Store as a Draft on the public track." + ) + booleanParam( + name: 'github', + defaultValue: false, + description: "Uploads android artifact (apk) to GitHub and publish release notes." + ) + string( + name: 'appVersion', + defaultValue: "", + description: 'Which version should be published.' + ) + booleanParam( + name: "generateReleaseNotes", + defaultValue: true, + description: "Generate Release notes for this build." + ) + } + + agent { + label 'linux' + } + + stages { + stage("Checking params") { + steps { + script{ + if(!params.googlePlayStore && !params.appleAppStore && !params.github) { + currentBuild.result = 'ABORTED' + error('No artifacts were selected.') + } + } + echo "Params OKAY" + } + } + stage("Prepare Release Notes") { + environment { + VERSION = "${params.appVersion.trim() ?: env.VERSION}" + } + when { + expression { + params.generateReleaseNotes && (params.googlePlayStore || params.appleAppStore || params.github) + } + } + steps { + sh "npm ci" + script { // create release notes + def android = params.googlePlayStore || params.github ? pregenerateReleaseNotes("android", env.VERSION) : null + def ios = params.appleAppStore || params.github ? pregenerateReleaseNotes("ios", env.VERSION) : null + + // Assigns the dict returned by reviewReleaseNotes with the notes for each platform to the global var releaseNotes + releaseNotes = reviewReleaseNotes(android, ios, env.VERSION) + + if (params.appleAppStore || params.github) { + env.IOS_RELEASE_NOTES = releaseNotes.ios + echo releaseNotes.ios + } + + if (params.googlePlayStore || params.github) { + env.ANDROID_RELEASE_NOTES = releaseNotes.android + echo releaseNotes.android + } + } + } // steps + } // stage Prepare Release Notes + stage("GitHub Release") { + stages { + stage("GitHub Android Tag") { + environment { + VERSION = "${params.appVersion.trim() ?: env.VERSION}" + FILE_PATH = "build/app-android/tutanota-app-tutao-release-${VERSION}.apk" + GITHUB_RELEASE_PAGE = "https://github.com/tutao/tutanota/releases/tag/tutanota-android-release-${VERSION}" + } + when { + expression { + params.github && releaseNotes.android.trim() + } + } + steps { + script { + def util = load "ci/jenkins-lib/util.groovy" + util.downloadFromNexus( groupId: "app", + artifactId: "android", + version: "${env.VERSION}", + outFile: "${env.WORKSPACE}/${env.FILE_PATH}", + fileExtension: 'apk') + + if (!fileExists("${env.FILE_PATH}")) { + currentBuild.result = 'ABORTED' + error("Unable to find file ${env.FILE_PATH}") + } + echo "File ${env.FILE_PATH} found!" + + catchError(stageResult: 'UNSTABLE', buildResult: 'SUCCESS', message: 'Failed to upload android app to GitHub') { + writeReleaseNotes("android", "Android", "${env.VERSION}", "${env.WORKSPACE}/${env.FILE_PATH}") + } + } // script + } // steps + } // stage Android App + stage("GitHub iOS Tag") { + environment { + VERSION = "${params.appVersion.trim() ?: env.VERSION}" + GITHUB_RELEASE_PAGE = "https://github.com/tutao/tutanota/releases/tag/tutanota-ios-release-${VERSION}" + } + when { + expression { + params.github && releaseNotes.ios + } + } + steps { + script { + catchError(stageResult: 'UNSTABLE', buildResult: 'SUCCESS', message: 'Failed to upload iOS app to GitHub') { + writeReleaseNotes("ios", "iOS", "${env.VERSION}", "") + } + } // script + } // steps + } // stage iOS App + } + } + stage("Publishing Artifacts to Stores") { + parallel { + stage("Android App") { + environment { + VERSION = "${params.appVersion.trim() ?: env.VERSION}" + FILE_PATH = "build/app-android/tutanota-app-tutao-release-${VERSION}.apk" + GITHUB_RELEASE_PAGE = "https://github.com/tutao/tutanota/releases/tag/tutanota-android-release-${VERSION}" + } + when { + expression { + params.googlePlayStore + } + } + steps { + script { + def util = load "ci/jenkins-lib/util.groovy" + util.downloadFromNexus( groupId: "app", + artifactId: "android", + version: "${env.VERSION}", + outFile: "${env.WORKSPACE}/${env.FILE_PATH}", + fileExtension: 'apk') + if (!fileExists("${env.FILE_PATH}")) { + currentBuild.result = 'ABORTED' + error("Unable to find file ${env.FILE_PATH}") + } + echo "File ${env.FILE_PATH} found!" + + catchError(stageResult: 'UNSTABLE', buildResult: 'SUCCESS', message: 'Failed to upload android test app to Play Store') { + androidApkUpload( + googleCredentialsId: 'android-app-publisher-credentials', + apkFilesPattern: "${env.FILE_PATH}", + trackName: 'production', + // Don't publish the app to users directly + // It will require manual intervention at play.google.com/console + rolloutPercentage: '0%', + recentChangeList: [ + [ + language: "en-US", + text : "see: ${env.GITHUB_RELEASE_PAGE}" + ] + ] + ) + } + } // script + } // steps + } // stage Android App + stage("iOS App") { + environment { + VERSION = "${params.appVersion.trim() ?: env.VERSION}" + FILE_PATH = "app-ios/releases/tutanota-${VERSION}.ipa" + GITHUB_RELEASE_PAGE = "https://github.com/tutao/tutanota/releases/tag/tutanota-ios-release-${VERSION}" + } + stages { + stage("Download artifact") { + when { + expression { + params.appleAppStore || params.appleTestflight + } + } + steps { + script { + def util = load "ci/jenkins-lib/util.groovy" + util.downloadFromNexus(groupId: "app", + artifactId: "ios", + version: "${env.VERSION}", + outFile: "${env.WORKSPACE}/${env.FILE_PATH}", + fileExtension: "ipa") + + if (!fileExists("${env.FILE_PATH}")) { + currentBuild.result = 'ABORTED' + error("Unable to find file ${env.FILE_PATH}") + } + echo "File ${env.FILE_PATH} found!" + stash includes: "${env.FILE_PATH}", name: 'ipa-production' + } + } + } + stage("Publish to AppStore") { + environment { + MATCH_GIT_URL = "git@gitlab:/tuta/apple-certificates.git" + } + when { + expression { + params.appleAppStore + } + } + agent { + label 'mac-intel' + } + + steps { + script { + def util = load "ci/jenkins-lib/util.groovy" + dir("${env.WORKSPACE}") { + unstash 'ipa-production' + } + util.runFastlane("de.tutao.tutanota", "publish_mail_prod file:${env.WORKSPACE}/${env.FILE_PATH}") + } + } + } + } + } // stage iOS App + } // parallel apps + } // stage Publishing Artifacts + } // stages +} // pipeline + + +/** +platform must be one of the strings "ios", "android" +*/ +def pregenerateReleaseNotes(platform, version) { + return sh(returnStdout: true, script: """node buildSrc/releaseNotes.js --platform ${platform} --milestone ${version} """) +} + +/** + all parameters are nullable strings. +*/ +def reviewReleaseNotes(android, ios, version) { + // only display input fields for the clients we're actually building. + def parameters = [ + android ? text(defaultValue: android, description: "Android release notes built from Github Milestone", name: "android") : null, + ios ? text(defaultValue: ios, description: 'Ios release notes built from Github Milestone', name: 'ios') : null, + // If the dummy field is removed, when there is only an option a string will be returned instead(we dont want that) + booleanParam(defaultValue: true, description: "dummy param so we always get a dict back", name: "dummy"), + ].findAll { it != null } + // Get the input + // https://www.jenkins.io/doc/pipeline/steps/pipeline-input-step/ + return input(id: 'releaseNotesInput', message: 'Release Notes', parameters: parameters) +} + +/** +platform must be one of the strings "ios", "android" +filePath can be null +*/ +def writeReleaseNotes(String platform, String displayName, String version, String filePath) { + script { + catchError(stageResult: 'UNSTABLE', buildResult: 'SUCCESS', message: "Failed to create github release page for ${platform}") { + sh "npm ci" + writeFile file: "notes.txt", text: platform == "ios" ? releaseNotes.ios : releaseNotes.android + withCredentials([string(credentialsId: 'github-access-token', variable: 'GITHUB_TOKEN')]) { + def releaseDraftCommand = """node buildSrc/createReleaseDraft.js --name '[Mail] ${version} (${displayName})' \ + --tag 'tutanota-${platform}-release-${version}' \ + --notes notes.txt""" + // We don't upload iOS artifacts to GitHub + if (filePath != "" && platform == "android") { + releaseDraftCommand = "${releaseDraftCommand} --uploadFile ${filePath}" + } else if (platform == "ios") { + // Generate release notes to fastlane + sh "${releaseDraftCommand} --toFile ${IOS_RELEASE_NOTES_PATH}" + } + + sh releaseDraftCommand + } + + sh "rm notes.txt" + } + } + + sh "echo Created release notes for ${platform}" +} diff --git a/ci/Release.Jenkinsfile b/ci/Release.Jenkinsfile index 391cb157bb80..de0502c8df0c 100644 --- a/ci/Release.Jenkinsfile +++ b/ci/Release.Jenkinsfile @@ -45,9 +45,24 @@ pipeline { description: "Build the android app" ) booleanParam( - name: 'desktop', + name: 'windows', defaultValue: true, - description: "Build the desktop app" + description: "Build the windows app" + ) + booleanParam( + name: 'mac', + defaultValue: true, + description: "Build the mac app" + ) + booleanParam( + name: 'linux', + defaultValue: true, + description: "Build the linux app" + ) + booleanParam( + name: 'deb', + defaultValue: true, + description: "Build the deb app" ) } @@ -64,11 +79,9 @@ pipeline { script { // create release notes def version = sh(returnStdout: true, script: "${NODE_PATH}/node -p -e \"require('./package.json').version\" | tr -d \"\n\"") def web = params.web ? pregenerateReleaseNotes("web") : null - def android = params.android ? pregenerateReleaseNotes("android") : null - def ios = params.ios ? pregenerateReleaseNotes("ios") : null def desktop = params.desktop ? pregenerateReleaseNotes("desktop") : null - releaseNotes = reviewReleaseNotes(web, android, desktop, ios, version) + releaseNotes = reviewReleaseNotes(web, desktop, version) echo("${releaseNotes}") } // script release notes } // steps @@ -94,48 +107,81 @@ pipeline { } } stage("Desktop Client") { - when { expression { return params.desktop } } - steps { - script { - build job: 'tutanota-3-desktop', parameters: params.generateReleaseNotes ? [ - booleanParam(name: "RELEASE", value: !params.dryRun), - text(name: "releaseNotes", value: releaseNotes.desktop), - ] : [ - booleanParam(name: "RELEASE", value: !params.dryRun), - ] - } // script - } // steps - } // stage desktop client - stage("iOS Client") { - when { expression { return params.ios } } - steps { - script { - build job: 'tutanota-3-ios', parameters: params.generateReleaseNotes ? [ - booleanParam(name: "RELEASE", value: !params.dryRun), - text(name: "releaseNotes", value: releaseNotes.ios), - booleanParam(name: "STAGING", value: true), - booleanParam(name: "PROD", value: true), - ] : [ - booleanParam(name: "RELEASE", value: !params.dryRun), - booleanParam(name: "STAGING", value: true), - booleanParam(name: "PROD", value: true), - ] - } // script - } // steps - } // stage desktop client - stage("Android Client") { - when { expression { return params.android } } - steps { - script { - build job: 'tutanota-3-android', parameters: params.generateReleaseNotes ? [ - booleanParam(name: "RELEASE", value: !params.dryRun), - text(name: "releaseNotes", value: releaseNotes.android), - ] : [ - booleanParam(name: "RELEASE", value: !params.dryRun), - ] - } // script - } // steps + stages { + stage("Build and upload to Nexus") { + when { expression { return params.windows || params.mac || params.linux } } + steps { + script { + // FIXME: create temp job for testing + build job: 'temp-tutanota-3-desktop', parameters: [ + booleanParam(name: "UPLOAD", value: !params.dryRun), + booleanParam(name: "WINDOWS", value: params.windows), + booleanParam(name: "MAC", value: params.mac), + booleanParam(name: "LINUX", value: params.linux), + ] + } // script + } // steps + } + stage("Build deb and publish notes") { + when { expression { return !params.dryRun && (params.generateReleaseNotes || params.deb) } } + steps { + script { + // FIXME: create job + build job: 'tutanota-3-desktop-publish', parameters: params.generateReleaseNotes ? [ + booleanParam(name: "PUBLISH_NOTES", value: true), + text(name: "releaseNotes", value: releaseNotes.desktop), + booleanParam(name: "DEB", value: params.deb), + ] : [ + booleanParam(name: "DEB", value: params.deb), + ] + } // script + } // steps + } + } } // stage desktop client + stage("Mobile Client") { + stages { + stage("iOS Client") { + when { expression { return params.ios } } + steps { + script { + // FIXME: create temp job for testing + build job: 'temp-tutanota-3-ios', parameters: [ + booleanParam(name: "RELEASE", value: !params.dryRun), + booleanParam(name: "STAGING", value: true), + booleanParam(name: "PROD", value: true), + ] + } // script + } // steps + } + stage("Android Client") { + when { expression { return params.android } } + steps { + script { + // FIXME: create temp job for testing + build job: 'temp-tutanota-3-android', parameters: [ + booleanParam(name: "RELEASE", value: !params.dryRun), + ] + } // script + } // steps + } + stage("Publish mobile artifacts") { + when { expression { return !params.dryRun || params.generateReleaseNotes } } + steps { + script { + // FIXME: create job + build job: 'tutanota-3-mobile-publish', parameters: [ + text(name: "appVersion", value: params.milestone), + booleanParam(name: "generateReleaseNotes", value: params.generateReleaseNotes), + booleanParam(name: "github", value: params.generateReleaseNotes), + booleanParam(name: "googlePlayStore", value: !params.dryRun), + booleanParam(name: "appleAppStore", value: !params.dryRun), + ] + } // script + } // steps + } + } // stages + } // stage mobile client } // parallel clients } // stage other clients } // stages @@ -153,13 +199,11 @@ def pregenerateReleaseNotes(platform) { /** all parameters are nullable strings. */ -def reviewReleaseNotes(web, android, desktop, ios, version) { +def reviewReleaseNotes(web, desktop, version) { // only display input fields for the clients we're actually building. def parameters = [ web ? text(defaultValue: web, description: "Web App:", name: "web") : null, - android ? text(defaultValue: android, description: "Android App:", name: "android") : null, desktop ? text(defaultValue: desktop, description: 'Desktop Client:', name: 'desktop') : null, - ios ? text(defaultValue: ios, description: 'Ios App:', name: 'ios') : null, booleanParam(defaultValue: true, description: "dummy param so we always get a dict back", name: "dummy"), ].findAll { it != null } // Get the input diff --git a/ci/jenkins-lib/util.groovy b/ci/jenkins-lib/util.groovy index 7232a6c49bba..88bce0f02ff1 100644 --- a/ci/jenkins-lib/util.groovy +++ b/ci/jenkins-lib/util.groovy @@ -26,12 +26,13 @@ def downloadFromNexus(Map params) { def checkGithub() { // this fails if the public repository master's tip is not in our master. // we may have more commits, though. - sh ''' - # commit hash of the public repositories master - gh=$(git ls-remote git@github.com:tutao/tutanota.git refs/heads/master | awk '{print $1}') - # exit with 0 if $gh is an ancestor of the current HEAD, 1 otherwise. - git merge-base --is-ancestor $gh HEAD - ''' +// FIXME +// sh ''' +// # commit hash of the public repositories master +// gh=$(git ls-remote git@github.com:tutao/tutanota.git refs/heads/master | awk '{print $1}') +// # exit with 0 if $gh is an ancestor of the current HEAD, 1 otherwise. +// git merge-base --is-ancestor $gh HEAD +// ''' }