diff --git a/.circleci/config.yml b/.circleci/config.yml new file mode 100644 index 000000000..4d6257a33 --- /dev/null +++ b/.circleci/config.yml @@ -0,0 +1,174 @@ +version: 2.1 + +############################################################################ + +references: + checkout_job_steps: &checkout_job_steps + steps: + - checkout + - run: + name: Checking out dependencies + command: | + git clone -b develop --single-branch https://github.com/Zimbra/zm-pkg-tool.git ~/zm-pkg-tool + echo "BUILD_NO=$CIRCLE_BUILD_NUM" > config.build + - persist_to_workspace: + root: .. + paths: + - zm-onlyoffice + - zm-pkg-tool + + build_job_steps: &build_job_steps + steps: + - attach_workspace: + at: .. + - run: + name: Creating build + command: make -f Makefile.build + no_output_timeout: 30m + - store_artifacts: + path: build/dist + - persist_to_workspace: + root: .. + paths: zm-onlyoffice/build/dist/* + + deploy_s3_job_steps: &deploy_s3_job_steps + docker: + - image: 'cimg/python:3.10' + resource_class: small + steps: + - attach_workspace: + at: .. + - run: + name: Create a zip file for artifacts and upload them to the OCI bucket + command: | + cd build/dist/ + zip -r $CIRCLE_PROJECT_REPONAME.zip . + ZIP_FILE_PATH="$CIRCLE_PROJECT_USERNAME/$CIRCLE_PROJECT_REPONAME/$CIRCLE_BRANCH-$CIRCLE_BUILD_NUM/archives/$CIRCLE_PROJECT_REPONAME.zip" + curl -X PUT "$OCI_PAR_URL$ZIP_FILE_PATH" -T $CIRCLE_PROJECT_REPONAME.zip + echo "artifacts uploaded to $ZIP_FILE_PATH" + + std_filters: &std_filters + filters: + branches: + only: + - master + - develop + +############################################################################ + +jobs: + checkout: + working_directory: ~/zm-onlyoffice + shell: /bin/bash -eo pipefail + docker: + - image: zimbra/zm-base-os:core-ubuntu + resource_class: small + <<: *checkout_job_steps + + build_u22: + working_directory: ~/zm-onlyoffice + shell: /bin/bash -eo pipefail + docker: + - image: $DOCKER_REGISTRY/zm-base-os:devcore-ubuntu-22.04 + auth: + username: $DOCKER_USER + password: $DOCKER_PASS + resource_class: small + <<: *build_job_steps + + build_u20: + working_directory: ~/zm-onlyoffice + shell: /bin/bash -eo pipefail + docker: + - image: zimbra/zm-base-os:devcore-ubuntu-20.04 + resource_class: small + <<: *build_job_steps + + build_u18: + working_directory: ~/zm-onlyoffice + shell: /bin/bash -eo pipefail + docker: + - image: zimbra/zm-base-os:devcore-ubuntu-18.04 + resource_class: small + <<: *build_job_steps + + build_c9: + working_directory: ~/zm-onlyoffice + shell: /bin/bash -eo pipefail + docker: + - image: $DOCKER_REGISTRY/zm-base-os:devcore-centos-9 + auth: + username: $DOCKER_USER + password: $DOCKER_PASS + resource_class: small + <<: *build_job_steps + + build_c8: + working_directory: ~/zm-onlyoffice + shell: /bin/bash -eo pipefail + docker: + - image: zimbra/zm-base-os:devcore-centos-8 + resource_class: small + <<: *build_job_steps + + build_c7: + working_directory: ~/zm-onlyoffice + shell: /bin/bash -eo pipefail + docker: + - image: zimbra/zm-base-os:devcore-centos-7 + resource_class: small + <<: *build_job_steps + + deploy_s3: + working_directory: ~/zm-onlyoffice + shell: /bin/bash -eo pipefail + <<: *deploy_s3_job_steps + +############################################################################ + +workflows: + version: 2 + main: + jobs: + - build: + type: approval + - checkout: + requires: + - build + - build_u22: + requires: + - checkout + context: + - docker-dev-registry + - build_u20: + requires: + - checkout + - build_u18: + requires: + - checkout + - build_c9: + requires: + - checkout + context: + - docker-dev-registry + - build_c8: + requires: + - checkout + - build_c7: + requires: + - checkout + - deploy_s3_hold: + type: approval + requires: + - build_u22 + - build_u20 + - build_u18 + - build_c9 + - build_c8 + - build_c7 + + - deploy_s3: + context: + - oracle + requires: + - deploy_s3_hold diff --git a/.gitignore b/.gitignore index e7a33fb04..774b3a23f 100644 --- a/.gitignore +++ b/.gitignore @@ -10,4 +10,5 @@ node_modules /Gruntfile.js.out local-development-*.json *.pyc -run-develop-local.py \ No newline at end of file +run-develop-local.py +.DS_Store diff --git a/Common/config/default.json b/Common/config/default.json index 06ac3c0dc..d1d5b7dd6 100644 --- a/Common/config/default.json +++ b/Common/config/default.json @@ -229,8 +229,12 @@ }, "services": { "CoAuthoring": { + "ssl" :{ + "key" : "", + "cert" : "" + }, "server": { - "port": 8000, + "port": 7084, "workerpercpu": 1, "mode": "development", "limits_tempfile_upload": 104857600, @@ -278,11 +282,11 @@ "limits_image_types_upload": "jpg;jpeg;jpe;png;gif;bmp;svg;tiff;tif" }, "sql": { - "type": "postgres", + "type": "mysql", "tableChanges": "doc_changes", "tableResult": "task_result", "dbHost": "localhost", - "dbPort": 5432, + "dbPort": 7306, "dbName": "onlyoffice", "dbUser": "onlyoffice", "dbPass": "onlyoffice", @@ -504,5 +508,21 @@ } ] } + }, + "FileStorage": { + "host": "", + "port": 4567, + "directory": "", + "silent": true + }, + "SpellChecker": { + "ssl" :{ + "key" : "", + "cert" : "" + }, + "server": { + "port": 7085, + "mode": "development" + } } -} +} \ No newline at end of file diff --git a/Common/config/log4js/logconfig.json b/Common/config/log4js/logconfig.json new file mode 100644 index 000000000..a6f117193 --- /dev/null +++ b/Common/config/log4js/logconfig.json @@ -0,0 +1,15 @@ +{ + "appenders": { + "default": { + "type": "file", + "filename": "/opt/zimbra/log/onlyoffice.log", + "layout": { + "type": "pattern", + "pattern": "[%d] [%p] %c - %.10000m" + } + } + }, + "categories": { + "default": { "appenders": [ "default" ], "level": "WARN" } + } +} \ No newline at end of file diff --git a/Common/config/onlyofficeconfig.json b/Common/config/onlyofficeconfig.json new file mode 100644 index 000000000..3f4cc7ecd --- /dev/null +++ b/Common/config/onlyofficeconfig.json @@ -0,0 +1,70 @@ +{ + "log": { + "filePath": "/opt/zimbra/onlyoffice/documentserver/server/Common/config/log4js/logconfig.json" + }, + "storage": { + "fs": { + "folderPath": "/opt/zimbra/onlyoffice/documentserver/App_Data/cache/files" + } + }, + "services": { + "CoAuthoring": { + "server": { + "static_content": { + "/fonts": { + "path": "/opt/zimbra/onlyoffice/documentserver/fonts", + "options": {"maxAge": "7d"} + }, + "/sdkjs": { + "path": "/opt/zimbra/onlyoffice/documentserver/sdkjs", + "options": {"maxAge": "7d"} + }, + "/web-apps": { + "path": "/opt/zimbra/onlyoffice/documentserver/web-apps", + "options": {"maxAge": "7d"} + }, + "/sdkjs-plugins": { + "path": "/opt/zimbra/onlyoffice/documentserver/sdkjs-plugins", + "options": {"maxAge": "7d"} + }, + "/App_Data": { + "path": "/opt/zimbra/onlyoffice/documentserver/App_Data", + "options": {"maxAge": "7d"} + } + } + }, + "utils": { + "utils_common_fontdir": "/usr/share/fonts" + }, + "request-filtering-agent" : { + "allowPrivateIPAddress": true, + "allowMetaIPAddress": true + }, + "sockjs": { + "sockjs_url": "/opt/zimbra/onlyoffice/documentserver/web-apps/vendor/sockjs/sockjs.min.js" + } + } + }, + "license": { + "license_file": "/opt/zimbra/onlyoffice/documentserver/license.lic", + "warning_limit_percents": 70, + "packageType": 0 + }, + "FileConverter": { + "converter": { + "fontDir": "/usr/share/fonts", + "presentationThemesDir": "/opt/zimbra/onlyoffice/documentserver/sdkjs/slide/themes", + "x2tPath": "/opt/zimbra/onlyoffice/documentserver/server/FileConverter/bin/x2t", + "docbuilderPath": "/opt/zimbra/onlyoffice/documentserver/server/FileConverter/bin/docbuilder", + "docbuilderAllFontsPath": "/opt/zimbra/onlyoffice/documentserver/App_Data/docbuilder/AllFonts.js" + } + }, + "FileStorage": { + "directory": "/opt/zimbra/onlyoffice/documentserver/App_Data" + }, + "SpellChecker": { + "server": { + "dictDir": "/opt/zimbra/onlyoffice/documentserver/dictionaries" + } + } +} \ No newline at end of file diff --git a/DocService/sources/server.js b/DocService/sources/server.js index a46c4d839..b73bcf1e3 100644 --- a/DocService/sources/server.js +++ b/DocService/sources/server.js @@ -41,6 +41,7 @@ const fs = require('fs'); const express = require('express'); const http = require('http'); +const https = require('https'); const urlModule = require('url'); const path = require('path'); const bodyParser = require("body-parser"); @@ -87,13 +88,29 @@ if (false) { return; } } +const configStorage = configCommon.get('storage'); +const cfgWopiEnable = configCommon.get('wopi.enable'); +const cfgHtmlTemplate = configCommon.get('wopi.htmlTemplate'); +const cfgTokenEnableBrowser = configCommon.get('services.CoAuthoring.token.enable.browser'); +const cfgTokenEnableRequestInbox = configCommon.get('services.CoAuthoring.token.enable.request.inbox'); +const cfgTokenEnableRequestOutbox = configCommon.get('services.CoAuthoring.token.enable.request.outbox'); +const cfgLicenseFile = configCommon.get('license.license_file'); +const cfgDownloadMaxBytes = configCommon.get('FileConverter.converter.maxDownloadBytes'); const app = express(); -app.disable('x-powered-by'); -//path.resolve uses __dirname by default(unexpected path in pkg) -app.set("views", path.resolve(process.cwd(), cfgHtmlTemplate)); -app.set("view engine", "ejs"); -const server = http.createServer(app); + +let server = null; + +if (config.has('ssl')) { + const privateKey = fs.readFileSync(config.get('ssl.key')).toString(); + const certificateKey = fs.readFileSync(config.get('ssl.cert')).toString(); + //See detailed options format here: http://nodejs.org/api/tls.html#tls_tls_createserver_options_secureconnectionlistener + const options = {key: privateKey, cert: certificateKey}; + + server = https.createServer(options, app); +} else { + server = http.createServer(app); +} let licenseInfo, licenseOriginal, updatePluginsTime, userPlugins; const updatePluginsCacheExpire = ms("5m"); diff --git a/Makefile.build b/Makefile.build new file mode 100644 index 000000000..f48c87af6 --- /dev/null +++ b/Makefile.build @@ -0,0 +1,55 @@ +######################################################################################################## + +SHELL = bash + +SHA512 = sha512sum + +OS = $(shell cat /etc/os-release | grep '^NAME' | cut -d'=' -f2 | cut -d'"' -f2) + +.PHONY: clean all + +######################################################################################################## + + +all: clean zimbra-onlyoffice-pkg + +######################################################################################################## + +ONLY_OFFICE_VERSION = 1.0 +ONLY_OFFICE_URL="https://files.zimbra.com/downloads/onlyoffice/u14/zimbra-onlyoffice-v7.2.2.56.tar.gz" + +stage-zimbra-onlyoffice-pkg: downloads + mkdir -p build/stage/zimbra-onlyoffice/opt/zimbra + cp -r downloads/onlyoffice build/stage/zimbra-onlyoffice/opt/zimbra/ + chmod +x build/stage/zimbra-onlyoffice/opt/zimbra/onlyoffice/bin/jq + cp -af bin/zmonlyofficeconfig build/stage/zimbra-onlyoffice/opt/zimbra/onlyoffice/bin/zmonlyofficeconfig + rm -rf build/stage/zimbra-onlyoffice/opt/zimbra/onlyoffice/.DS_Store + rm -rf build/stage/zimbra-onlyoffice/opt/zimbra/onlyoffice/._.DS_Store + +zimbra-onlyoffice-pkg: stage-zimbra-onlyoffice-pkg + ./pkg-build.pl \ + --pkg-version=$(ONLY_OFFICE_VERSION).$(shell git log --format=%at -1) \ + --pkg-release=1 \ + --pkg-name=zimbra-onlyoffice \ + --out-type=binary \ + --pkg-summary="Zimbra onlyoffice package" \ + --pkg-pre-install-script='scripts/preinst.sh' \ + --pkg-post-install-script='scripts/postinst.sh' \ + --pkg-installs='/opt/zimbra/onlyoffice/*' + + +downloads: + mkdir -p downloads + wget -O downloads/zimbra-onlyoffice.tar.gz $(ONLY_OFFICE_URL) + cd downloads/; tar -xf zimbra-onlyoffice.tar.gz + mv downloads/zimbra-onlyoffice downloads/onlyoffice + +######################################################################################################## + +######################################################################################################## + +clean: + rm -rf build + rm -rf downloads + +######################################################################################################## diff --git a/SpellChecker/sources/server.js b/SpellChecker/sources/server.js index a6e3a947c..df3635462 100644 --- a/SpellChecker/sources/server.js +++ b/SpellChecker/sources/server.js @@ -93,9 +93,9 @@ if (cluster.isMaster) { if (config.has('ssl')) { const privateKey = fs.readFileSync(config.get('ssl.key')).toString(); const certificateKey = fs.readFileSync(config.get('ssl.cert')).toString(); - const trustedCertificate = fs.readFileSync(config.get('ssl.ca')).toString(); + //See detailed options format here: http://nodejs.org/api/tls.html#tls_tls_createserver_options_secureconnectionlistener - const options = {key: privateKey, cert: certificateKey, ca: [trustedCertificate]}; + const options = {key: privateKey, cert: certificateKey}; server = https.createServer(options, app); } else { diff --git a/bin/createdb.sql b/bin/createdb.sql new file mode 100644 index 000000000..68de88c20 --- /dev/null +++ b/bin/createdb.sql @@ -0,0 +1,81 @@ +-- +-- Create schema onlyoffice +-- +CREATE DATABASE IF NOT EXISTS onlyoffice DEFAULT CHARACTER SET utf8 DEFAULT COLLATE utf8_general_ci; +--CREATE USER IF NOT EXISTS 'onlyoffice' IDENTIFIED BY 'onlyoffice'; +--grant all on onlyoffice.* to 'onlyoffice' identified by 'onlyoffice'; +GRANT ALL PRIVILEGES ON onlyoffice.* TO 'zimbra'; +FLUSH PRIVILEGES; + + +/*!40101 SET @OLD_CHARACTER_SET_CLIENT=@@CHARACTER_SET_CLIENT */; +/*!40101 SET @OLD_CHARACTER_SET_RESULTS=@@CHARACTER_SET_RESULTS */; +/*!40101 SET @OLD_COLLATION_CONNECTION=@@COLLATION_CONNECTION */; +/*!40101 SET NAMES utf8 */; + +/*!40014 SET @OLD_UNIQUE_CHECKS=@@UNIQUE_CHECKS, UNIQUE_CHECKS=0 */; +/*!40014 SET @OLD_FOREIGN_KEY_CHECKS=@@FOREIGN_KEY_CHECKS, FOREIGN_KEY_CHECKS=0 */; +/*!40101 SET @OLD_SQL_MODE=@@SQL_MODE, SQL_MODE='NO_AUTO_VALUE_ON_ZERO' */; + + + +USE onlyoffice; + +-- +-- Definition of table `doc_changes` +-- + +CREATE TABLE IF NOT EXISTS `doc_changes` ( + `tenant` varchar(255) NOT NULL, + `id` varchar(255) NOT NULL, + `change_id` int(10) unsigned NOT NULL, + `user_id` varchar(255) NOT NULL, + `user_id_original` varchar(255) NOT NULL, + `user_name` varchar(255) NOT NULL, + `change_data` longtext NOT NULL, + `change_date` datetime NOT NULL, + PRIMARY KEY (`tenant`, `id`,`change_id`) +) ENGINE=InnoDB DEFAULT CHARSET=utf8; + +-- +-- Dumping data for table `doc_changes` +-- + +/*!40000 ALTER TABLE `doc_changes` DISABLE KEYS */; +/*!40000 ALTER TABLE `doc_changes` ENABLE KEYS */; + +-- +-- Definition of table `task_result` +-- + +CREATE TABLE IF NOT EXISTS `task_result` ( + `tenant` varchar(255) NOT NULL, + `id` varchar(255) NOT NULL, + `status` tinyint(3) NOT NULL, + `status_info` int(10) NOT NULL, + `created_at` timestamp NOT NULL DEFAULT CURRENT_TIMESTAMP, + `last_open_date` datetime NOT NULL, + `user_index` int(10) unsigned NOT NULL DEFAULT 1, + `change_id` int(10) unsigned NOT NULL DEFAULT 0, + `callback` longtext NOT NULL, + `baseurl` text NOT NULL, + `password` longtext NULL, + `additional` longtext NULL, + PRIMARY KEY (`tenant`, `id`) +) ENGINE=InnoDB DEFAULT CHARSET=utf8; + +-- +-- Dumping data for table `task_result` +-- + +/*!40000 ALTER TABLE `task_result` DISABLE KEYS */; +/*!40000 ALTER TABLE `task_result` ENABLE KEYS */; + + +/*!40101 SET SQL_MODE=@OLD_SQL_MODE */; +/*!40014 SET FOREIGN_KEY_CHECKS=@OLD_FOREIGN_KEY_CHECKS */; +/*!40014 SET UNIQUE_CHECKS=@OLD_UNIQUE_CHECKS */; +/*!40101 SET CHARACTER_SET_CLIENT=@OLD_CHARACTER_SET_CLIENT */; +/*!40101 SET CHARACTER_SET_RESULTS=@OLD_CHARACTER_SET_RESULTS */; +/*!40101 SET COLLATION_CONNECTION=@OLD_COLLATION_CONNECTION */; +/*!40101 SET CHARACTER_SET_CLIENT=@OLD_CHARACTER_SET_CLIENT */; diff --git a/bin/jq b/bin/jq new file mode 100644 index 000000000..f48b0ca92 Binary files /dev/null and b/bin/jq differ diff --git a/bin/process_id.json b/bin/process_id.json new file mode 100644 index 000000000..9bdeba80c --- /dev/null +++ b/bin/process_id.json @@ -0,0 +1,4 @@ +{ + "docservice": -1, + "converter" : -1 +} diff --git a/bin/zmonlyofficeconfig b/bin/zmonlyofficeconfig new file mode 100644 index 000000000..510bc6e54 --- /dev/null +++ b/bin/zmonlyofficeconfig @@ -0,0 +1,260 @@ +#!/bin/bash +# +# ***** BEGIN LICENSE BLOCK ***** +# Zimbra Collaboration Suite Server +# Copyright (C) 2005, 2006, 2007, 2008, 2009, 2010, 2011, 2012, 2013, 2014, 2016, 2021 Synacor, Inc. +# +# This program is free software: you can redistribute it and/or modify it under +# the terms of the GNU General Public License as published by the Free Software Foundation, +# version 2 of the License. +# +# This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; +# without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. +# See the GNU General Public License for more details. +# You should have received a copy of the GNU General Public License along with this program. +# If not, see . +# ***** END LICENSE BLOCK ***** +# + +source /opt/zimbra/bin/zmshutil || exit 1 +zmsetvars + +zmjq="/opt/zimbra/onlyoffice/bin/jq" + +defaultjson_path="/opt/zimbra/onlyoffice/documentserver/server/Common/config/default.json" +mysql_port=7306 +zimbra_mysql_user="zimbra" +zimbra_mysql_password="" + +# Check for commercial certificates +if [ -f /opt/zimbra/ssl/zimbra/commercial/commercial.crt ] && [ -f /opt/zimbra/ssl/zimbra/commercial/commercial.key ]; then + server_crt="/opt/zimbra/ssl/zimbra/commercial/commercial.crt" + server_key="/opt/zimbra/ssl/zimbra/commercial/commercial.key" +else + server_crt="/opt/zimbra/ssl/zimbra/server/server.crt" + server_key="/opt/zimbra/ssl/zimbra/server/server.key" +fi + +STATUS=0 + +document_editing_JWT_secret="" + +usage() { + echo "" + echo "Invalid argument. Allowed :" + echo "new Configure Onlyoffice on the server" + echo "update-jwt-secret arg0 Update Onlyoffice server's jwt secret with the passed value" + echo "update-db Update Onlyoffice server's DB connection details from config" + echo "update-certificate Update Onlyoffice server's certificate paths" + echo "" + exit +} + +generate_random() +{ + length=8 + digits=({2..9}) + lower=({a..k} {m..n} {p..z}) + upper=({A..N} {P..Z}) + CharArray=(${digits[*]} ${lower[*]} ${upper[*]}) + ArrayLength=${#CharArray[*]} + password="" + while [ $length -gt 0 ] + do + index=$(($RANDOM%$ArrayLength)) + char=${CharArray[$index]} + password=${password}${char} + ((length--)) + done + echo "$password" +} + +update_db_config() +{ + #port + config_mysql_port=$(su - zimbra -c "zmlocalconfig -x -s mysql_port") + if [ -n "$config_mysql_port" ]; then + IFS='=' + read -ra arr <<< "$config_mysql_port" + mysql_port=$(echo "${arr[1]}" | xargs) + fi + + content=$(${zmjq} --argjson mysql_port "$mysql_port" '.services.CoAuthoring.sql.dbPort=$mysql_port' ${defaultjson_path}) + echo "${content}" > ${defaultjson_path} + #user + config_mysql_user=$(su - zimbra -c "zmlocalconfig -x -s zimbra_mysql_user") + if [ -n "$config_mysql_user" ]; then + IFS='=' + read -ra arr <<< "$config_mysql_user" + zimbra_mysql_user=$(echo "${arr[1]}" | xargs) + fi + + content=$(${zmjq} --arg zimbra_mysql_user "$zimbra_mysql_user" '.services.CoAuthoring.sql.dbUser=$zimbra_mysql_user' ${defaultjson_path}) + echo "${content}" > ${defaultjson_path} + + #password + config_mysql_pass=$(su - zimbra -c "zmlocalconfig -x -s zimbra_mysql_password") + if [ -n "$config_mysql_pass" ]; then + IFS='=' + read -ra arr <<< "$config_mysql_pass" + zimbra_mysql_password=$(echo "${arr[1]}" | xargs) + fi + + content=$(${zmjq} --arg zimbra_mysql_password "$zimbra_mysql_password" '.services.CoAuthoring.sql.dbPass=$zimbra_mysql_password' ${defaultjson_path}) + echo "${content}" > ${defaultjson_path} + +} + +update_certificate_path() +{ + content=$(${zmjq} --arg server_crt "$server_crt" '.services.CoAuthoring.ssl.cert=$server_crt' ${defaultjson_path}) + echo "${content}" > ${defaultjson_path} + + content=$(${zmjq} --arg server_crt "$server_crt" '.SpellChecker.ssl.cert=$server_crt' ${defaultjson_path}) + echo "${content}" > ${defaultjson_path} + + content=$(${zmjq} --arg server_key "$server_key" '.services.CoAuthoring.ssl.key=$server_key' ${defaultjson_path}) + echo "${content}" > ${defaultjson_path} + + content=$(${zmjq} --arg server_key "$server_key" '.SpellChecker.ssl.key=$server_key' ${defaultjson_path}) + echo "${content}" > ${defaultjson_path} +} + +update_token_config() +{ + + content=$(${zmjq} '.services.CoAuthoring.token.enable.browser=true' ${defaultjson_path}) + echo "${content}" > ${defaultjson_path} + + content=$(${zmjq} '.services.CoAuthoring.token.enable.request.inbox=true' ${defaultjson_path}) + echo "${content}" > ${defaultjson_path} + + content=$(${zmjq} '.services.CoAuthoring.token.enable.request.outbox=true' ${defaultjson_path}) + echo "${content}" > ${defaultjson_path} + + content=$(${zmjq} '.services.CoAuthoring.requestDefaults.rejectUnauthorized=false' ${defaultjson_path}) + echo "${content}" > ${defaultjson_path} + +} + +update_jwt_config() +{ + jwtsecret=$1 + cmd="su - zimbra -c \"zmprov -m -l mcf zimbraDocumentEditingJwtSecret ${jwtsecret}\"" + eval "$cmd" + RC=$? + if [ $RC -ne "0" ]; then + STATUS=$R + echo "Updating zimbraDocumentEditingJwtSecret faced problem." + fi + + #update the default.json + content=$(${zmjq} --arg jwtsecret "$jwtsecret" '.services.CoAuthoring.secret.browser.string=$jwtsecret' ${defaultjson_path}) + echo "${content}" > ${defaultjson_path} + + content=$(${zmjq} --arg jwtsecret "$jwtsecret" '.services.CoAuthoring.secret.inbox.string=$jwtsecret' ${defaultjson_path}) + echo "${content}" > ${defaultjson_path} + + content=$(${zmjq} --arg jwtsecret "$jwtsecret" '.services.CoAuthoring.secret.outbox.string=$jwtsecret' ${defaultjson_path}) + echo "${content}" > ${defaultjson_path} + + content=$(${zmjq} --arg jwtsecret "$jwtsecret" '.services.CoAuthoring.secret.session.string=$jwtsecret' ${defaultjson_path}) + echo "${content}" > ${defaultjson_path} +} + +set_jwt_secret() +{ + echo "Setting Random Secret..." + jwtsecret=$(generate_random) + update_jwt_config "$jwtsecret" +} + +generate_fonts_data() +{ + mkdir -p /opt/zimbra/onlyoffice/documentserver/fonts + + fonts_gen=$(LD_LIBRARY_PATH=/opt/zimbra/onlyoffice/documentserver/server/FileConverter/bin /opt/zimbra/onlyoffice/documentserver/server/tools/allfontsgen \ + --input="/opt/zimbra/onlyoffice/documentserver/core-fonts" \ + --allfonts-web="/opt/zimbra/onlyoffice/documentserver/sdkjs/common/AllFonts.js" \ + --allfonts="/opt/zimbra/onlyoffice/documentserver/server/FileConverter/bin/AllFonts.js" \ + --images="/opt/zimbra/onlyoffice/documentserver/sdkjs/common/Images" \ + --selection="/opt/zimbra/onlyoffice/documentserver/server/FileConverter/bin/font_selection.bin" \ + --output-web="/opt/zimbra/onlyoffice/documentserver/fonts" \ + --use-system="true") + R=$? + if [ $R -ne "0" ]; then + STATUS=$R + echo "Error while generating fonts. ${fonts_gen}" + fi + +} + +generate_presentation_themes() +{ + presentation_themes_gen=$(LD_LIBRARY_PATH=/opt/zimbra/onlyoffice/documentserver/server/FileConverter/bin /opt/zimbra/onlyoffice/documentserver/server/tools/allthemesgen \ + --converter-dir="/opt/zimbra/onlyoffice/documentserver/server/FileConverter/bin"\ + --src="/opt/zimbra/onlyoffice/documentserver/sdkjs/slide/themes"\ + --output="/opt/zimbra/onlyoffice/documentserver/sdkjs/common/Images") + R=$? + if [ $R -ne "0" ]; then + STATUS=$R + echo "Error while generating presentation themes. ${presentation_themes_gen}" + fi +} + +type="$1" +if [ "$#" -eq 0 ]; then + type="new" +fi + + +case "$type" in + new) + + jwtsecret=$(su - zimbra -c "zmprov -m -l getConfig zimbraDocumentEditingJwtSecret") + if [ -z "$jwtsecret" ]; then + echo "JWT Secret not set." + set_jwt_secret + else + IFS=':' + read -ra arr <<< "$jwtsecret" + document_editing_JWT_secret=$(echo "${arr[1]}" | xargs) + update_jwt_config "$document_editing_JWT_secret" + fi + + update_certificate_path + update_token_config + update_db_config + generate_fonts_data + generate_presentation_themes + echo "Onlyoffice configuration done." + exit $STATUS + ;; + + update-jwt-secret) + + if [ "$#" -ne 2 ]; then + usage + exit 1 + fi + jwtsecret="$2" + + update_jwt_config "$jwtsecret" + exit $STATUS + ;; + + update-db ) + update_db_config + exit $STATUS + ;; + + update-certificate ) + update_certificate_path + exit $STATUS + ;; + + *) + usage + exit 1 + ;; +esac diff --git a/pkg-build.pl b/pkg-build.pl new file mode 100755 index 000000000..c1bb67ac8 --- /dev/null +++ b/pkg-build.pl @@ -0,0 +1,714 @@ +#!/usr/bin/perl + +use strict; +use warnings; + +use Config; +use Cwd; +use Data::Dumper; +use File::Basename; +use File::Copy; +use File::Find; +use File::Path qw/make_path/; +use Getopt::Long; +use IPC::Cmd qw/run can_run/; +use Term::ANSIColor; + +my $GLOBAL_PATH_TO_SCRIPT_FILE; +my $GLOBAL_PATH_TO_SCRIPT_DIR; +my $GLOBAL_PATH_TO_TOP; +my $CWD; +my $ZM_PKG_TOOL_DIR; +my $PKG_CREATION_DATE = `date -R`; + +my %CFG = (); + +BEGIN +{ + $ENV{ANSI_COLORS_DISABLED} = 1 if ( ! -t STDOUT ); + $GLOBAL_PATH_TO_SCRIPT_FILE = Cwd::abs_path(__FILE__); + $GLOBAL_PATH_TO_SCRIPT_DIR = dirname($GLOBAL_PATH_TO_SCRIPT_FILE); + $GLOBAL_PATH_TO_TOP = dirname($GLOBAL_PATH_TO_SCRIPT_DIR); + $ZM_PKG_TOOL_DIR = $GLOBAL_PATH_TO_TOP."/zm-pkg-tool"; + $GLOBAL_PATH_TO_SCRIPT_DIR = $ZM_PKG_TOOL_DIR; + $CWD = getcwd(); +} + +sub LoadConfiguration($) +{ + my $args = shift; + + my $cfg_name = $args->{name}; + my $cmd_hash = $args->{hash_src}; + my $default_sub = $args->{default_sub}; + my $validate_sub = $args->{validate_sub}; + + my $val; + my $src; + + if ( !defined $val ) + { + y/A-Z_/a-z-/ foreach ( my $cmd_name = $cfg_name ); + + if ( $cmd_hash && exists $cmd_hash->{$cmd_name} ) + { + $val = $cmd_hash->{$cmd_name}; + $src = "cmdline"; + } + } + + if ( !defined $val ) + { + if ( $CFG{CFG_DIR} ) + { + my $file = "$CFG{CFG_DIR}/config.pl"; + my $hash = LoadProperties($file) + if ( -f $file ); + + if ( $hash && exists $hash->{$cfg_name} ) + { + $val = $hash->{$cfg_name}; + $src = "config" + } + } + } + + my $valid = 1; + + if ( defined $val ) + { + $valid = &$validate_sub($val) + if ($validate_sub); + } + + if ( !defined $val || !$valid ) + { + if ($default_sub) + { + $val = &$default_sub($cfg_name); + $src = "default" . ( $valid ? "" : "($src was rejected)" ); + } + } + + if ( defined $val ) + { + $valid = &$validate_sub($val) + if ($validate_sub); + + if ( ref($val) eq "HASH" ) + { + foreach my $k ( keys %{$val} ) + { + $CFG{$cfg_name}{$k} = ${$val}{$k}; + + printf( " %-25s: %-17s : %s\n", $cfg_name, $cmd_hash ? $src : "detected", "{" . $k . " => " . ${$val}{$k} . "}" ); + } + } + elsif ( ref($val) eq "ARRAY" ) + { + $CFG{$cfg_name} = $val; + + printf( " %-25s: %-17s : %s\n", $cfg_name, $cmd_hash ? $src : "detected", "[" . join( ", ", @{ $CFG{$cfg_name} } ) . "]" ); + } + else + { + $CFG{$cfg_name} = $val; + + printf( " %-25s: %-17s : %s\n", $cfg_name, $cmd_hash ? $src : "detected", $val ); + } + } +} + + +sub Die($;$) +{ + my $msg = shift; + my $info = shift || ""; + my $err = "$!"; + + print "\n"; + print "\n"; + print "=========================================================================================================\n"; + print color('red') . "FAILURE MSG" . color('reset') . " : $msg\n"; + print color('red') . "SYSTEM ERR " . color('reset') . " : $err\n" if ($err); + print color('red') . "EXTRA INFO " . color('reset') . " : $info\n" if ($info); + print "\n"; + print "=========================================================================================================\n"; + print color('red'); + print "--Stack Trace--\n"; + my $i = 1; + + while ( ( my @call_details = ( caller( $i++ ) ) ) ) + { + print $call_details[1] . ":" . $call_details[2] . " called from " . $call_details[3] . "\n"; + } + print color('reset'); + print "\n"; + print "=========================================================================================================\n"; + + die "END"; +} + + +sub assert($$;$) +{ + my $l = shift || ""; + my $r = shift || ""; + my $m = shift || ""; + + Die( $m, "Got:[$l] != Exp:[$r]" ) + if ( $l ne $r ); +} + +sub LoadProperties($) +{ + my $f = shift; + + my $x = SlurpFile($f); + + my @cfg_kvs = + map { $_ =~ s/^\s+|\s+$//g; $_ } # trim + map { split( /=/, $_, 2 ) } # split around = + map { $_ =~ s/#.*$//g; $_ } # strip comments + grep { $_ !~ /^\s*#/ } # ignore comments + grep { $_ !~ /^\s*$/ } # ignore empty lines + @$x; + + my %ret_hash = (); + for ( my $e = 0 ; $e < scalar @cfg_kvs ; $e += 2 ) + { + my $probe_key = $cfg_kvs[$e]; + my $probe_val = $cfg_kvs[ $e + 1 ]; + + if ( $probe_key =~ /^%(.*)/ ) + { + my @val_kv_pair = split( /=/, $probe_val, 2 ); + + $ret_hash{$1}{ $val_kv_pair[0] } = $val_kv_pair[1]; + } + else + { + $ret_hash{$probe_key} = $probe_val; + } + } + + return \%ret_hash; +} + + +sub SlurpFile($) +{ + my $f = shift; + + open( FD, "<", "$f" ) || Die( "In open for read", "file='$f'" ); + + chomp( my @x = ); + close(FD); + + return \@x; +} + +sub System(@) +{ + my $cmd_str = "@_"; + + print color('green') . "#: pwd=@{[Cwd::getcwd()]}" . color('reset') . "\n"; + print color('green') . "#: $cmd_str" . color('reset') . "\n"; + + $! = 0; + my ( $success, $error_message, $full_buf, $stdout_buf, $stderr_buf ) = run( command => \@_, verbose => 1 ); + + Die( "cmd='$cmd_str'", $error_message ) + if ( !$success ); + + return { msg => $error_message, out => $stdout_buf, err => $stderr_buf }; +} + +sub GetPkgFormat() +{ + if ( -f "/etc/redhat-release" ) + { + return "rpm"; + } + elsif ( -f "/etc/lsb-release" ) + { + return "deb"; + } + else + { + Die("Unknown OS"); + } +} + +sub GetOsTag() +{ + our $gOSTAG; + + if ( !defined $gOSTAG ) + { + if ( -f "/etc/redhat-release" ) + { + chomp( $gOSTAG = `sed -n -e '1{s/[^0-9]*/r/; s/[.].*//; p;}' /etc/redhat-release` ); + } + elsif ( -f "/etc/lsb-release" ) + { + chomp( $gOSTAG = `sed -n -e '/DISTRIB_RELEASE/{s/.*=/u/; s/[.].*//; p;}' /etc/lsb-release` ); + } + else + { + Die("Unknown OS"); + } + } + + return $gOSTAG +} + +sub _ValidateOutType($) +{ + my $ty = shift; + + if ( defined $ty ) + { + return 1 if ( $ty eq "all" ); + return 1 if ( $ty eq "source" ); + return 1 if ( $ty eq "binary" ); + } + + return undef; +} + +sub Init() +{ + my %cmd_hash = (); + + my @cmd_args = ( + { + name => "CFG_DIR", + type => "=s", + hash_src => \%cmd_hash, + validate_sub => undef, + default_sub => sub { return "pkg-spec"; }, + }, + { + name => "OUT_TYPE", + type => "=s", + hash_src => \%cmd_hash, + validate_sub => &_ValidateOutType, + default_sub => sub { return "all"; }, + }, + { + name => "OUT_BASE_DIR", + type => "=s", + hash_src => \%cmd_hash, + validate_sub => undef, + default_sub => sub { return "build"; }, + }, + { + name => "OUT_TEMP_DIR", + type => "=s", + hash_src => \%cmd_hash, + validate_sub => undef, + default_sub => sub { return "$CFG{OUT_BASE_DIR}/tmp"; }, + }, + { + name => "OUT_STAGE_DIR", + type => "=s", + hash_src => \%cmd_hash, + validate_sub => undef, + default_sub => sub { return "$CFG{OUT_BASE_DIR}/stage"; }, + }, + { + name => "OUT_DIST_DIR", + type => "=s", + hash_src => \%cmd_hash, + validate_sub => undef, + default_sub => sub { return "$CFG{OUT_BASE_DIR}/dist"; }, + }, + { + name => "PKG_NAME", + type => "=s", + hash_src => \%cmd_hash, + validate_sub => undef, + default_sub => sub { return Die("@_ not unspecfied"); }, + }, + { + name => "PKG_RELEASE", + type => "=s", + hash_src => \%cmd_hash, + validate_sub => undef, + default_sub => sub { return Die("@_ not specified"); }, + }, + { + name => "PKG_VERSION", + type => "=s", + hash_src => \%cmd_hash, + validate_sub => undef, + default_sub => sub { return Die("@_ not specified"); }, + }, + { + name => "PKG_SUMMARY", + type => "=s", + hash_src => \%cmd_hash, + validate_sub => undef, + default_sub => sub { return Die("@_ not specified"); }, + }, + { + name => "PKG_INSTALLS", + type => "=s@", + hash_src => \%cmd_hash, + validate_sub => undef, + default_sub => sub { return []; }, + }, + { + name => "PKG_DEPENDS", + type => "=s@", + hash_src => \%cmd_hash, + validate_sub => undef, + default_sub => sub { return []; }, + }, + { + name => "PKG_PRE_DEPENDS", + type => "=s@", + hash_src => \%cmd_hash, + validate_sub => undef, + default_sub => sub { return []; }, + }, + { + name => "PKG_PROVIDES", + type => "=s@", + hash_src => \%cmd_hash, + validate_sub => undef, + default_sub => sub { return []; }, + }, + { + name => "PKG_CONFLICTS", + type => "=s@", + hash_src => \%cmd_hash, + validate_sub => undef, + default_sub => sub { return []; }, + }, + { + name => "PKG_OBSOLETES", + type => "=s@", + hash_src => \%cmd_hash, + validate_sub => undef, + default_sub => sub { return []; }, + }, + { + name => "PKG_REPLACES", + type => "=s@", + hash_src => \%cmd_hash, + validate_sub => undef, + default_sub => sub { return []; }, + }, + { + name => "PKG_PRE_INSTALL_SCRIPT", + type => "=s", + hash_src => \%cmd_hash, + validate_sub => undef, + default_sub => sub { return undef; }, + }, + { + name => "PKG_POST_INSTALL_SCRIPT", + type => "=s", + hash_src => \%cmd_hash, + validate_sub => undef, + default_sub => sub { return undef; }, + }, + { + name => "PKG_FORMAT", + type => "", + hash_src => \%cmd_hash, + validate_sub => undef, + default_sub => sub { return GetPkgFormat(); }, + }, + { + name => "PKG_OS_TAG", + type => "", + hash_src => \%cmd_hash, + validate_sub => undef, + default_sub => sub { return GetOsTag(); }, + }, + ); + + { + my @cmd_opts = + map { $_->{opt} =~ y/A-Z_/a-z-/; $_; } # convert the opt named to lowercase to make command line options + map { { opt => $_->{name}, opt_s => $_->{type} } } # create a new hash with keys opt, opt_s + grep { $_->{type} } # get only names which have a valid type + @cmd_args; + + my $help_func = sub { + print "Usage: $0 \n"; + print "Supported options: \n"; + print " --" . "$_->{opt}$_->{opt_s}\n" foreach (@cmd_opts); + exit(0); + }; + + if ( !GetOptions( \%cmd_hash, ( map { $_->{opt} . $_->{opt_s} } @cmd_opts ), help => $help_func ) ) + { + print Die("wrong commandline options, use --help"); + } + } + + print "=========================================================================================================\n"; + LoadConfiguration($_) foreach (@cmd_args); + print "=========================================================================================================\n"; +} + + +sub _SanitizePkgList($) +{ + my $list = shift; + + my $san_list = ""; + foreach my $entry ( @{$list} ) + { + $entry =~ s/\s//g; + $entry =~ s/\/|/g; + $entry =~ s/[|][|]*/|/g; + + my @comp = split( /[|]/, $entry ); + + my $san_entry = ""; + foreach my $comp_entry (@comp) + { + # abc-2.5 -> $pkn + # abc-2.5>=2.1.0-1 -> $pkn$cmp$ver + # abc-2.5(>=2.1.0-1) -> $pkn($cmp$ver) + if ( $comp_entry =~ m/^([^>=<]+)([>=<]*)(.*)$/ ) + { + my $pkn = $1; + my $cmp = $2; + my $ver = $3; + + $pkn =~ s/[(]$//; + $ver =~ s/[)]$//; + + my $no_add_os_tag = 1 + if ( $ver =~ s/[!]$// ); + + my $san_comp = ""; + + if ($pkn) + { + $san_comp = $pkn; + + if ( $cmp && $ver ) + { + my $tag = ( !$no_add_os_tag && $ver =~ m/[-][^-]*$/ ) ? ".$CFG{PKG_OS_TAG}" : ""; + + if ( $CFG{PKG_FORMAT} eq "deb" ) + { + $cmp = ">>" if ( $cmp eq ">" ); + $cmp = "<<" if ( $cmp eq "<" ); + $cmp = "=" if ( $cmp eq "==" ); + + $san_comp .= " (" . $cmp . " " . $ver . "$tag)"; + } + if ( $CFG{PKG_FORMAT} eq "rpm" ) + { + $cmp = ">" if ( $cmp eq ">>" ); + $cmp = "<" if ( $cmp eq "<<" ); + $cmp = "=" if ( $cmp eq "==" ); + + $san_comp .= " " . $cmp . " " . $ver . "$tag"; + } + } + } + + $san_entry .= ( $san_entry && $san_comp ? " | " : "" ) . $san_comp + if ( $CFG{PKG_FORMAT} eq "deb" ); + $san_entry .= ( $san_entry && $san_comp ? " or " : "" ) . $san_comp + if ( $CFG{PKG_FORMAT} eq "rpm" ); + } + } + + $san_list .= ( $san_list && $san_entry ? ", " : "" ) . $san_entry; + } + + return $san_list; +} + + +sub Build() +{ + System( "rm", "-f", $_ ) foreach glob("$CFG{OUT_DIST_DIR}/$CFG{PKG_OS_TAG}/$CFG{PKG_NAME}_*"); + System( "rm", "-f", $_ ) foreach glob("$CFG{OUT_DIST_DIR}/$CFG{PKG_OS_TAG}/$CFG{PKG_NAME}-*.rpm"); + + System( "rm", "-rf", "$CFG{OUT_TEMP_DIR}/$CFG{PKG_NAME}/" ); + System( "mkdir", "-p", "$CFG{OUT_TEMP_DIR}/$CFG{PKG_NAME}/" ); + System( "mkdir", "-p", "$CFG{OUT_DIST_DIR}/$CFG{PKG_OS_TAG}/" ); + + my $pkg_pre_install_list = ""; + my $pkg_post_install_list = ""; + if ( defined $CFG{PKG_POST_INSTALL_SCRIPT}) { + open FILE, "$CFG{PKG_POST_INSTALL_SCRIPT}" or die "Couldn't open postinst file: $!"; + $pkg_post_install_list = join("", ); + close FILE; + } + if ( defined $CFG{PKG_PRE_INSTALL_SCRIPT}) { + open FILE, "$CFG{PKG_PRE_INSTALL_SCRIPT}" or die "Couldn't open preinst file: $!"; + $pkg_pre_install_list = join("", ); + close FILE; + } + + if ( $CFG{PKG_FORMAT} eq "rpm" ) + { + System( "cp", "-a", "$GLOBAL_PATH_TO_SCRIPT_DIR/default-template/rpm", "$CFG{OUT_TEMP_DIR}/$CFG{PKG_NAME}/" ); + System( "cp", "-a", "$CFG{CFG_DIR}/rpm", "$CFG{OUT_TEMP_DIR}/$CFG{PKG_NAME}/SPECS" ) if ( -d "$CFG{CFG_DIR}/rpm" ); + } + elsif ( $CFG{PKG_FORMAT} eq "deb" ) + { + System( "cp", "-a", "$GLOBAL_PATH_TO_SCRIPT_DIR/default-template/debian", "$CFG{OUT_TEMP_DIR}/$CFG{PKG_NAME}/" ); + System( "cp", "-a", "$CFG{CFG_DIR}/debian", "$CFG{OUT_TEMP_DIR}/$CFG{PKG_NAME}/" ) if ( -d "$CFG{CFG_DIR}/debian" ); + } + else + { + Die("Unknown PACKAGING format"); + } + + find( + { + wanted => sub { + my $tpl_file = $_; + if ( -f $tpl_file && $tpl_file =~ /[.]in$/ ) + { + s/[.]in$// for ( my $new_file = $tpl_file ); + + open( FDr, "<", "$tpl_file" ); + open( FDw, ">", "$new_file" ); + + while ( my $line = ) + { + { + my $pkg_install_list = + ( $CFG{PKG_FORMAT} eq "deb" && $tpl_file =~ /install[.]in/ ) + ? join( "\n", map { $_ =~ s,^/,,; $_ } @{ $CFG{PKG_INSTALLS} } ) + : join( "\n", @{ $CFG{PKG_INSTALLS} } ); + + $line =~ s/[@][@]PKG_INSTALLS[@][@]/$pkg_install_list/g; + } + $line =~ s/[@][@]PKG_POST_INSTALL[@][@]/$pkg_post_install_list/g; + $line =~ s/[@][@]PKG_PRE_INSTALL[@][@]/$pkg_pre_install_list/g; + $line =~ s/[@][@]PKG_NAME[@][@]/$CFG{PKG_NAME}/g; + $line =~ s/[@][@]PKG_RELEASE[@][@]/$CFG{PKG_RELEASE}/g; + $line =~ s/[@][@]PKG_OS_TAG[@][@]/$CFG{PKG_OS_TAG}/g; + $line =~ s/[@][@]PKG_VERSION[@][@]/$CFG{PKG_VERSION}/g; + $line =~ s/[@][@]PKG_SUMMARY[@][@]/$CFG{PKG_SUMMARY}/g; + $line =~ s/[@][@]PKG_CREATION_DATE[@][@]/$PKG_CREATION_DATE/g; + $line =~ s/[@][@]PKG_DEPENDS[@][@]/@{[_SanitizePkgList($CFG{PKG_DEPENDS})]}/g; + $line =~ s/[@][@]PKG_PRE_DEPENDS[@][@]/@{[_SanitizePkgList($CFG{PKG_PRE_DEPENDS})]}/g; + $line =~ s/[@][@]PKG_PROVIDES[@][@]/@{[_SanitizePkgList($CFG{PKG_PROVIDES})]}/g; + $line =~ s/[@][@]PKG_CONFLICTS[@][@]/@{[_SanitizePkgList($CFG{PKG_CONFLICTS})]}/g; + $line =~ s/[@][@]PKG_OBSOLETES[@][@]/@{[_SanitizePkgList($CFG{PKG_OBSOLETES})]}/g; + $line =~ s/[@][@]PKG_REPLACES[@][@]/@{[_SanitizePkgList($CFG{PKG_REPLACES})]}/g; + + if ( $line =~ m/^\s*[A-Za-z][A-Za-z_0-9-]*\s*[:](\s*,*\s*)*$/ ) # drop lines with empty headers + { + } + else + { + print FDw $line; + } + } + + close(FDr); + close(FDw); + + unlink($tpl_file); + } + }, + }, + "$CFG{OUT_TEMP_DIR}/$CFG{PKG_NAME}" + ); + + System( "make", "-f", "$CFG{CFG_DIR}/$CFG{PKG_NAME}/Makefile", "PKG_NAME=$CFG{PKG_NAME}", "PKG_STAGE_DIR=$CFG{OUT_STAGE_DIR}/$CFG{PKG_NAME}", "stage" ) + if ( -f "$CFG{CFG_DIR}/$CFG{PKG_NAME}/Makefile" ); + + System( "ant", "-f", "$CFG{CFG_DIR}/$CFG{PKG_NAME}/build.xml", "-DPKG_NAME=$CFG{PKG_NAME}", "-DPKG_STAGE_DIR=$CFG{OUT_STAGE_DIR}/$CFG{PKG_NAME}", "stage" ) + if ( -f "$CFG{CFG_DIR}/$CFG{PKG_NAME}/build.xml" ); + + if ( $CFG{PKG_FORMAT} eq "rpm" ) + { + my @pkg_type_opts = ("-ba"); + @pkg_type_opts = ("-bb") if ( $CFG{OUT_TYPE} eq "binary" ); + @pkg_type_opts = ("-bs") if ( $CFG{OUT_TYPE} eq "source" ); + + System( "mv", "$CFG{OUT_TEMP_DIR}/$CFG{PKG_NAME}/rpm/1.spec", "$CFG{OUT_TEMP_DIR}/$CFG{PKG_NAME}/rpm/$CFG{PKG_NAME}.spec" ); + System( "mkdir", "-p", "$CFG{OUT_TEMP_DIR}/$CFG{PKG_NAME}/BUILDROOT/" ); + + System( "cp", "-a", $_, "$CFG{OUT_TEMP_DIR}/$CFG{PKG_NAME}/BUILDROOT/@{[basename $_]}" ) foreach glob("$CFG{OUT_STAGE_DIR}/$CFG{PKG_NAME}/*"); + + { + System( "rpmbuild", "-v", "--define", "_topdir $CWD/$CFG{OUT_TEMP_DIR}/$CFG{PKG_NAME}", "--buildroot=$CWD/$CFG{OUT_TEMP_DIR}/$CFG{PKG_NAME}/BUILDROOT/", @pkg_type_opts, "$CFG{OUT_TEMP_DIR}/$CFG{PKG_NAME}/rpm/$CFG{PKG_NAME}.spec" ); + } + + print "\n\n"; + print "=========================================================================================================\n"; + System( "mv", "-v", $_, "$CFG{OUT_DIST_DIR}/$CFG{PKG_OS_TAG}/" ) foreach glob("$CFG{OUT_TEMP_DIR}/$CFG{PKG_NAME}/SRPMS/*.rpm"); + System( "mv", "-v", $_, "$CFG{OUT_DIST_DIR}/$CFG{PKG_OS_TAG}/" ) foreach glob("$CFG{OUT_TEMP_DIR}/$CFG{PKG_NAME}/RPMS/*/*.rpm"); + print "=========================================================================================================\n"; + } + elsif ( $CFG{PKG_FORMAT} eq "deb" ) + { + my @pkg_type_opts = ( "-uc", "-us" ); + push( @pkg_type_opts, "-b" ) if ( $CFG{OUT_TYPE} eq "binary" ); + push( @pkg_type_opts, "-S" ) if ( $CFG{OUT_TYPE} eq "source" ); + open(my $fd, ">>$CFG{OUT_TEMP_DIR}/$CFG{PKG_NAME}/debian/rules"); + print $fd "override_dh_strip: \n"; + print $fd " cp -r \$\$(pwd)/opt/zimbra/onlyoffice/documentserver/server/DocService/docservice \$\$(pwd)/debian/zimbra-onlyoffice/opt/zimbra/onlyoffice/documentserver/server/DocService/docservice \n"; + print $fd "override_dh_strip_nondeterminism: \n"; + print $fd "override_dh_shlibdeps: \n"; + print $fd "override_dh_builddeb:\n"; + print $fd " dh_builddeb -- -Zgzip\n"; + close($fd); + + System( "cp", "-a", $_, "$CFG{OUT_TEMP_DIR}/$CFG{PKG_NAME}/@{[basename $_]}" ) foreach glob("$CFG{OUT_STAGE_DIR}/$CFG{PKG_NAME}/*"); + + { + chdir("$CFG{OUT_TEMP_DIR}/$CFG{PKG_NAME}"); + print "$CWD \n"; + System( "dpkg-buildpackage", @pkg_type_opts ); + chdir($CWD); + } + + print "\n\n"; + print "=========================================================================================================\n"; + System( "mv", "-v", $_, "$CFG{OUT_DIST_DIR}/$CFG{PKG_OS_TAG}/" ) foreach glob("$CFG{OUT_TEMP_DIR}/$CFG{PKG_NAME}_*.*"); + print "=========================================================================================================\n"; + } + else + { + Die("Unknown PACKAGING format"); + } +} + +sub SelfTest() +{ + if ( $CFG{PKG_FORMAT} eq "DEB" ) + { + assert( + _SanitizePkgList( [ "abc-3.4(>=4.4)", "def-6.7(>6.6-1)", "def-6.7(>6.6-1!)", "ghi(=7.0)", "jkl", "aaa | bbb | perl(Carp) >= 3.2" ] ), + "abc-3.4 (>= 4.4), def-6.7 (>> 6.6-1.$CFG{PKG_OS_TAG}), def-6.7 (>> 6.6-1), ghi (= 7.0), jkl, aaa | bbb | perl(Carp) (>= 3.2)" + ); + } + + if ( $CFG{PKG_FORMAT} eq "RPM" ) + { + assert( + _SanitizePkgList( [ "abc-3.4(>=4.4)", "def-6.7(>6.6-1)", "def-6.7(>6.6-1!)", "ghi(=7.0)", "jkl", "aaa | bbb | perl(Carp) >= 3.2" ] ), + "abc-3.4 >= 4.4, def-6.7 > 6.6-1.$CFG{PKG_OS_TAG}, def-6.7 > 6.6-1, ghi = 7.0, jkl, aaa or bbb or perl(Carp) >= 3.2" + ); + } +} + + +sub main() +{ + Init(); + SelfTest(); + Build(); +} + +main(); diff --git a/scripts/postinst.sh b/scripts/postinst.sh new file mode 100755 index 000000000..053b0f07f --- /dev/null +++ b/scripts/postinst.sh @@ -0,0 +1,21 @@ +if [ -f /opt/zimbra/onlyoffice/bin/jq ]; then + chmod +x /opt/zimbra/onlyoffice/bin/jq +fi + +if [ -f /opt/zimbra/onlyoffice/bin/zmonlyofficeconfig ]; then + chmod +x /opt/zimbra/onlyoffice/bin/zmonlyofficeconfig +fi + +if [ -f /opt/zimbra/onlyoffice/bin/process_id.json.bak ]; then + mv -f /opt/zimbra/onlyoffice/bin/process_id.json.bak /opt/zimbra/onlyoffice/bin/process_id.json + /opt/zimbra/onlyoffice/bin/zmonlyofficeconfig +fi + +if [ -f /opt/zimbra/onlyoffice/bin/process_id.json ]; then + chmod 775 /opt/zimbra/onlyoffice/bin/process_id.json + chown zimbra:zimbra /opt/zimbra/onlyoffice/bin/process_id.json +fi + +if [ -d /opt/zimbra/onlyoffice ]; then + chown -R zimbra:zimbra /opt/zimbra/onlyoffice/ +fi diff --git a/scripts/preinst.sh b/scripts/preinst.sh new file mode 100755 index 000000000..8d5fcb257 --- /dev/null +++ b/scripts/preinst.sh @@ -0,0 +1,5 @@ +if [ -f /opt/zimbra/onlyoffice/bin/process_id.json ]; then + cp -af /opt/zimbra/onlyoffice/bin/process_id.json /opt/zimbra/onlyoffice/bin/process_id.json.bak + chmod 775 /opt/zimbra/onlyoffice/bin/process_id.json.bak + chown zimbra:zimbra /opt/zimbra/onlyoffice/bin/process_id.json.bak +fi