mirror of
https://github.com/chenasraf/nextcloud-jukebox.git
synced 2026-05-17 17:38:02 +00:00
build: update makefile
This commit is contained in:
237
Makefile
237
Makefile
@@ -1,44 +1,33 @@
|
||||
# SPDX-FileCopyrightText: Bernhard Posselt <dev@bernhard-posselt.com>
|
||||
# SPDX-License-Identifier: AGPL-3.0-or-later
|
||||
|
||||
# Generic Makefile for building and packaging a Nextcloud app which uses pnpm and
|
||||
# Composer.
|
||||
#
|
||||
# Dependencies:
|
||||
# * make
|
||||
# * which
|
||||
# * curl: used if phpunit and composer are not installed to fetch them from the web
|
||||
# * tar: for building the archive
|
||||
# * pnpm: for building and testing everything JS
|
||||
# Nextcloud App Template — Makefile
|
||||
# ---------------------------------
|
||||
# A friendly, batteries-included Makefile for building and packaging a Nextcloud app
|
||||
# that uses pnpm (JS) and Composer (PHP).
|
||||
#
|
||||
# If no composer.json is in the app root directory, the Composer step
|
||||
# will be skipped. The same goes for the package.json which can be located in
|
||||
# the app root or the js/ directory.
|
||||
# Requirements:
|
||||
# - make, which, curl, tar
|
||||
# - pnpm (for JS build/lint/test)
|
||||
# - composer (optional; will auto-download local composer.phar if missing)
|
||||
#
|
||||
# The pnpm command by launches the pnpm build script:
|
||||
# Conventions:
|
||||
# - If no composer.json → Composer step is skipped.
|
||||
# - If no package.json (root) and js/package.json missing → pnpm step is skipped.
|
||||
# - JS build is delegated to your package.json scripts (tool-agnostic).
|
||||
#
|
||||
# pnpm build
|
||||
# Common recipes:
|
||||
# make build → install deps & build
|
||||
# make dist → build source + appstore tarballs
|
||||
# make test → run PHP unit tests
|
||||
# make lint → lint JS & PHP
|
||||
# make openapi → generate OpenAPI JSON
|
||||
# make sign → print signature for GitHub tarball
|
||||
# make release → upload release to Nextcloud App Store
|
||||
#
|
||||
# The pnpm test command launches the pnpm test script:
|
||||
#
|
||||
# pnpm test
|
||||
#
|
||||
# The idea behind this is to be completely testing and build tool agnostic. All
|
||||
# build tools and additional package managers should be installed locally in
|
||||
# your project, since this won't pollute people's global namespace.
|
||||
#
|
||||
# The following pnpm scripts in your package.json install and update the bower
|
||||
# and pnpm dependencies and use gulp as build system (notice how everything is
|
||||
# run from the node_modules folder):
|
||||
#
|
||||
# "scripts": {
|
||||
# "test": "node node_modules/gulp-cli/bin/gulp.js karma",
|
||||
# "prebuild": "pnpm install && node_modules/bower/bin/bower install && node_modules/bower/bin/bower update",
|
||||
# "build": "node node_modules/gulp-cli/bin/gulp.js"
|
||||
# },
|
||||
|
||||
app_name=jukebox
|
||||
repo_name=chenasraf/nextcloud-$(app_name)
|
||||
repo_path=chenasraf/nextcloud-$(app_name)
|
||||
build_tools_directory=$(CURDIR)/build/tools
|
||||
source_build_directory=$(CURDIR)/build/artifacts/source
|
||||
source_intermediate_directory=$(CURDIR)/build/artifacts/intermediate-source
|
||||
@@ -48,13 +37,17 @@ appstore_build_directory=$(CURDIR)/build/artifacts/appstore
|
||||
appstore_package_name=$(appstore_build_directory)/$(app_name)
|
||||
pnpm=$(shell which pnpm 2> /dev/null)
|
||||
composer=$(shell which composer 2> /dev/null)
|
||||
composer_phar=build/tools/composer.phar
|
||||
composer_phar=$(build_tools_directory)/composer.phar
|
||||
composer_bin := $(if $(composer),$(composer),php $(composer_phar))
|
||||
pnpm_wrapper=$(build_tools_directory)/pnpm.sh
|
||||
pnpm_cmd=$(if $(pnpm),$(pnpm),$(pnpm_wrapper))
|
||||
|
||||
# Default target: install deps & build JS (and PHP if composer.json exists)
|
||||
all: build
|
||||
|
||||
# Fetches the PHP and JS dependencies and compiles the JS. If no composer.json
|
||||
# is present, the composer step is skipped, if no package.json or js/package.json
|
||||
# is present, the pnpm step is skipped
|
||||
# build:
|
||||
# - Composer install if composer.json exists (skips if vendor/ exists)
|
||||
# - pnpm install & build if package.json (root) or js/package.json exists
|
||||
.PHONY: build
|
||||
build:
|
||||
ifneq (,$(wildcard $(CURDIR)/composer.json))
|
||||
@@ -67,40 +60,67 @@ ifneq (,$(wildcard $(CURDIR)/js/package.json))
|
||||
make pnpm
|
||||
endif
|
||||
|
||||
# Installs and updates the composer dependencies. If composer is not installed
|
||||
# a copy is fetched from the web
|
||||
.PHONY: composer
|
||||
composer:
|
||||
ifeq (, $(composer))
|
||||
@echo "No composer command available, downloading a copy from the web"
|
||||
$(composer_phar):
|
||||
@echo "No system composer found; installing local composer.phar"
|
||||
mkdir -p $(build_tools_directory)
|
||||
curl -sS https://getcomposer.org/installer | php
|
||||
mv composer.phar $(composer_phar)
|
||||
endif
|
||||
|
||||
# composer:
|
||||
# - Use system composer if available, else download local composer.phar
|
||||
# - Skip install if vendor/ already exists
|
||||
.PHONY: composer
|
||||
composer: $(if $(composer),, $(composer_phar))
|
||||
ifneq ("$(wildcard vendor)","")
|
||||
@echo "Vendor directory already exists, skipping composer install"
|
||||
else
|
||||
@echo "Installing composer dependencies..."
|
||||
$(if $(composer),$(composer),php $(composer_phar)) install --prefer-dist
|
||||
$(composer_bin) install --prefer-dist
|
||||
endif
|
||||
|
||||
# Installs pnpm dependencies
|
||||
# Ensure a local pnpm wrapper exists if pnpm is not installed globally.
|
||||
# The wrapper uses Corepack to activate pnpm, then delegates to pnpm.
|
||||
$(pnpm_wrapper):
|
||||
@mkdir -p $(build_tools_directory); \
|
||||
echo "#!/usr/bin/env bash" > $(pnpm_wrapper); \
|
||||
echo "set -e" >> $(pnpm_wrapper); \
|
||||
echo "if ! command -v pnpm >/dev/null 2>&1; then" >> $(pnpm_wrapper); \
|
||||
echo " if command -v corepack >/dev/null 2>&1; then" >> $(pnpm_wrapper); \
|
||||
echo " corepack enable >/dev/null 2>&1 || true" >> $(pnpm_wrapper); \
|
||||
echo " corepack prepare pnpm@latest --activate" >> $(pnpm_wrapper); \
|
||||
echo " else" >> $(pnpm_wrapper); \
|
||||
echo " echo 'pnpm not found and corepack not available. Please install pnpm or Node.js (with corepack).'; exit 1" >> $(pnpm_wrapper); \
|
||||
echo " fi" >> $(pnpm_wrapper); \
|
||||
echo "fi" >> $(pnpm_wrapper); \
|
||||
echo "exec pnpm \"\$$@\"" >> $(pnpm_wrapper); \
|
||||
chmod +x $(pnpm_wrapper)
|
||||
|
||||
# pnpm:
|
||||
# - Install JS deps (frozen lockfile)
|
||||
# - Run build via root package.json if present, else fallback to js/ subdir
|
||||
.PHONY: pnpm
|
||||
pnpm:
|
||||
pnpm install --frozen-lockfile
|
||||
pnpm: $(pnpm_wrapper)
|
||||
$(pnpm_cmd) install --frozen-lockfile
|
||||
ifeq (,$(wildcard $(CURDIR)/package.json))
|
||||
cd js && $(pnpm) build
|
||||
cd js && $(pnpm_cmd) build
|
||||
else
|
||||
pnpm build
|
||||
$(pnpm_cmd) build
|
||||
endif
|
||||
|
||||
# Removes the appstore build
|
||||
# clean:
|
||||
# - Remove build artifacts (but keep dependencies)
|
||||
.PHONY: clean
|
||||
clean:
|
||||
rm -rf ./build
|
||||
|
||||
# Same as clean but also removes dependencies installed by composer, bower and
|
||||
# pnpm
|
||||
# refresh-autoload:
|
||||
# - Regenerate Composer autoload files (if composer.json exists)
|
||||
.PHONY: refresh-autoload
|
||||
refresh-autoload: composer
|
||||
$(if $(composer),$(composer),php $(composer_phar)) dump-autoload -o
|
||||
|
||||
# distclean:
|
||||
# - Run clean and also remove PHP/JS dependencies
|
||||
.PHONY: distclean
|
||||
distclean: clean
|
||||
rm -rf vendor
|
||||
@@ -108,13 +128,16 @@ distclean: clean
|
||||
rm -rf js/vendor
|
||||
rm -rf js/node_modules
|
||||
|
||||
# Builds the source and appstore package
|
||||
# dist:
|
||||
# - Build both source and appstore tarballs
|
||||
.PHONY: dist
|
||||
dist:
|
||||
make source
|
||||
make appstore
|
||||
|
||||
# Builds the source package
|
||||
# source:
|
||||
# - Create a source tarball (full source, excludes dev/test artifacts)
|
||||
# - Output: build/artifacts/source/$(app_name).tar.gz
|
||||
.PHONY: source
|
||||
source:
|
||||
rm -rf $(source_build_directory)
|
||||
@@ -133,7 +156,9 @@ source:
|
||||
cd $(source_intermediate_directory) && \
|
||||
tar czf $(source_package_name).tar.gz ../$(app_name)
|
||||
|
||||
# Builds the source package for the app store, ignores php and js tests
|
||||
# appstore:
|
||||
# - Create an App Store tarball (strips tests, dotfiles, dev configs)
|
||||
# - Output: build/artifacts/appstore/$(app_name).tar.gz
|
||||
.PHONY: appstore
|
||||
appstore:
|
||||
rm -rf $(appstore_build_directory)
|
||||
@@ -169,16 +194,28 @@ appstore:
|
||||
cd $(app_intermediate_directory) && \
|
||||
tar czf $(appstore_package_name).tar.gz ../$(app_name)
|
||||
|
||||
# test:
|
||||
# - Run PHP unit tests (standard + optional integration config)
|
||||
.PHONY: test
|
||||
test: composer
|
||||
$(CURDIR)/vendor/phpunit/phpunit/phpunit -c tests/phpunit.xml
|
||||
( test ! -f tests/phpunit.integration.xml ) || $(CURDIR)/vendor/phpunit/phpunit/phpunit -c tests/phpunit.integration.xml
|
||||
|
||||
# test-docker:
|
||||
# - Run PHP unit tests inside a Nextcloud Docker container
|
||||
.PHONY: test-docker
|
||||
test-docker:
|
||||
docker-compose exec nextcloud phpunit -c apps-shared/jukebox/tests/phpunit.xml
|
||||
|
||||
# lint:
|
||||
# - Lint JS via pnpm and PHP via composer script "lint"
|
||||
.PHONY: lint
|
||||
lint:
|
||||
pnpm lint
|
||||
$(composer_phar) run lint
|
||||
$(composer_bin) run lint
|
||||
|
||||
# php-cs-fixer:
|
||||
# - Fix staged PHP files with PHP-CS-Fixer shim (checks syntax first)
|
||||
.PHONY: php-cs-fixer
|
||||
php-cs-fixer:
|
||||
@echo "\x1b[33mFixing PHP files...\x1b[0m"
|
||||
@@ -191,37 +228,76 @@ php-cs-fixer:
|
||||
PHP_CS_FIXER_IGNORE_ENV=true php vendor-bin/cs-fixer/vendor/php-cs-fixer/shim/php-cs-fixer.phar --config=.php-cs-fixer.dist.php fix $$FILES || exit 1; \
|
||||
fi
|
||||
|
||||
# format:
|
||||
# - Format JS and PHP (composer script "cs:fix")
|
||||
.PHONY: format
|
||||
format:
|
||||
pnpm format
|
||||
PHP_CS_FIXER_IGNORE_ENV=true $(composer_phar) run cs:fix
|
||||
PHP_CS_FIXER_IGNORE_ENV=true $(composer_bin) run cs:fix
|
||||
|
||||
# openapi:
|
||||
# - Generate OpenAPI spec via composer script "openapi"
|
||||
# - Output: build/openapi/openapi.json
|
||||
.PHONY: openapi
|
||||
openapi:
|
||||
@echo "\x1b[33mGenerating OpenAPI documentation...\x1b[0m"
|
||||
$(if $(composer),$(composer),php $(composer_phar)) run openapi
|
||||
$(composer_bin) run openapi
|
||||
@echo "\x1b[32mOpenAPI documentation generated at build/openapi/openapi.json\x1b[0m"
|
||||
|
||||
# csr:
|
||||
# - Generate a new private key and self-signed certificate for signing releases
|
||||
# and place them in ~/.nextcloud/certificates/$(app_name).{key,csr}
|
||||
.PHONY: csr
|
||||
csr:
|
||||
@if [ -f "$$HOME/.nextcloud/certificates/$(app_name).key" ] && [ -f "$$HOME/.nextcloud/certificates/$(app_name).csr" ]; then \
|
||||
echo "\x1b[31mPrivate key & CSR already exists at ~/.nextcloud/certificates/$(app_name).{key,csr}\x1b[0m"; \
|
||||
else \
|
||||
echo "\x1b[33mGenerating a new private key and self-signed certificate...\x1b[0m"; \
|
||||
openssl req -nodes -newkey rsa:4096 -keyout $(app_name).key -out $(app_name).csr -subj "/CN=$(app_name)"; \
|
||||
mkdir -p "$$HOME/.nextcloud/certificates" && \
|
||||
mv "$(app_name).key" "$$HOME/.nextcloud/certificates/$(app_name).key" && \
|
||||
mv "$(app_name).csr" "$$HOME/.nextcloud/certificates/$(app_name).csr" || \
|
||||
echo "\x1b[31mError: Could not move key & CSR to ~/.nextcloud/certificates/\x1b[0m"; \
|
||||
echo "\x1b[32mPrivate key saved to ~/.nextcloud/certificates/$(app_name).key"; \
|
||||
echo "\x1b[32mCerticate signing request saved to ~/.nextcloud/certificates/$(app_name).csr"; \
|
||||
echo ""; \
|
||||
echo "Follow the instructions at:"; \
|
||||
echo "https://nextcloudappstore.readthedocs.io/en/latest/developer.html#obtaining-a-certificate"; \
|
||||
echo "to get your app registered and obtain a proper public certificate .crt file.\x1b[0m"; \
|
||||
fi
|
||||
|
||||
# sign:
|
||||
# - Print a base64 SHA-512 signature for the release tarball from GitHub.
|
||||
# - Requires a private key at ~/.nextcloud/certificates/$(app_name).key
|
||||
# - Reads version from version.txt
|
||||
.PHONY: sign
|
||||
sign:
|
||||
@VERSION="$$(cat version.txt)"; \
|
||||
TMPF="$$(mktemp)"; \
|
||||
KEY_FILE=~/.nextcloud/certificates/$(app_name).key; \
|
||||
if [ ! -f "$$KEY_FILE" ]; then \
|
||||
echo "\x1b[31m❌ Error: Private key not found at $$KEY_FILE\x1b[0m"; \
|
||||
exit 1; \
|
||||
fi; \
|
||||
echo "\x1b[33mSigning version $${VERSION}\x1b[0m"; \
|
||||
echo "\x1b[33mDownloading archive...\x1b[0m"; \
|
||||
curl -L https://github.com/$(repo_path)/releases/download/v$${VERSION}/$(app_name)-v$${VERSION}.tar.gz -o $${TMPF}; \
|
||||
curl -L https://github.com/$(repo_path)/releases/download/v$${VERSION}/$(app_name)-v$${VERSION}.tar.gz -o "$${TMPF}"; \
|
||||
FILESIZE=$$(stat -f%z "$${TMPF}" 2>/dev/null || stat -c%s "$${TMPF}"); \
|
||||
if [ "$${FILESIZE}" -lt 10240 ]; then \
|
||||
echo "\x1b[31mError: Downloaded file is too small (<10KB, actual: $${FILESIZE} bytes)\x1b[0m"; \
|
||||
rm -rf $${TMPF}; \
|
||||
rm -rf "$${TMPF}"; \
|
||||
exit 1; \
|
||||
fi; \
|
||||
echo "\x1b[33mSigning with key $(app_name).key\x1b[0m"; \
|
||||
echo "\x1b[33mSigning with key $$KEY_FILE\x1b[0m"; \
|
||||
echo; \
|
||||
echo "\x1b[32mDownload URL:\x1b[0m https://github.com/$(repo_path)/releases/download/v$${VERSION}/$(app_name)-v$${VERSION}.tar.gz"; \
|
||||
echo "\x1b[32mSignature:\x1b[0m"; \
|
||||
openssl dgst -sha512 -sign ~/.nextcloud/certificates/$(app_name).key $${TMPF} | openssl base64; \
|
||||
rm -rf $${TMPF}
|
||||
openssl dgst -sha512 -sign "$$KEY_FILE" "$${TMPF}" | openssl base64; \
|
||||
rm -rf "$${TMPF}"
|
||||
|
||||
# release:
|
||||
# - Upload release to Nextcloud App Store using NEXTCLOUD_API_TOKEN
|
||||
# - Downloads tarball from GitHub, signs it, and POSTs to App Store
|
||||
.PHONY: release
|
||||
release:
|
||||
@VERSION="$$(cat version.txt)"; \
|
||||
@@ -229,13 +305,19 @@ release:
|
||||
printf "\x1b[33mNEXTCLOUD_API_TOKEN not set. Enter token: \x1b[0m"; \
|
||||
read -r NEXTCLOUD_API_TOKEN; \
|
||||
fi; \
|
||||
if [ -n "$$NEXTCLOUD_API_TOKEN" ]; then \
|
||||
echo "\x1b[32m✅ Using provided NEXTCLOUD_API_TOKEN"; \
|
||||
if [ -z "$$NEXTCLOUD_API_TOKEN" ]; then \
|
||||
echo "\x1b[31m❌ Error: NEXTCLOUD_API_TOKEN is missing\x1b[0m"; \
|
||||
exit 1; \
|
||||
else \
|
||||
echo "\x1b[31m❌ Error: NEXTCLOUD_API_TOKEN is missing"; \
|
||||
echo "\x1b[32m✅ Using provided NEXTCLOUD_API_TOKEN\x1b[0m"; \
|
||||
fi; \
|
||||
TMPF="$$(mktemp)"; \
|
||||
DOWNLOAD_URL="https://github.com/$(repo_path)/releases/download/v$${VERSION}/$(app_name)-v$${VERSION}.tar.gz"; \
|
||||
KEY_FILE=~/.nextcloud/certificates/$(app_name).key; \
|
||||
if [ ! -f "$$KEY_FILE" ]; then \
|
||||
echo "\x1b[31m❌ Error: Private key not found at $$KEY_FILE\x1b[0m"; \
|
||||
exit 1; \
|
||||
fi; \
|
||||
echo "\x1b[33mDownloading archive for version $${VERSION}...\x1b[0m"; \
|
||||
curl -L "$${DOWNLOAD_URL}" -o "$${TMPF}"; \
|
||||
FILESIZE=$$(stat -f%z "$${TMPF}" 2>/dev/null || stat -c%s "$${TMPF}"); \
|
||||
@@ -244,19 +326,26 @@ release:
|
||||
rm -f "$${TMPF}"; \
|
||||
exit 1; \
|
||||
fi; \
|
||||
echo "\x1b[33mSigning with key $(app_name).key\x1b[0m"; \
|
||||
echo "\x1b[33mSigning with key $$KEY_FILE\x1b[0m"; \
|
||||
echo; \
|
||||
SIGNATURE="$$(openssl dgst -sha512 -sign ~/.nextcloud/certificates/$(app_name).key "$${TMPF}" | openssl base64 | tr -d '\n')"; \
|
||||
SIGNATURE="$$(openssl dgst -sha512 -sign "$$KEY_FILE" "$${TMPF}" | openssl base64 | tr -d '\n')"; \
|
||||
rm -f "$${TMPF}"; \
|
||||
echo "\x1b[32mReleasing to Nextcloud App Store...\x1b[0m"; \
|
||||
curl -X POST \
|
||||
RESPONSE="$$(mktemp)"; \
|
||||
HTTP_CODE=$$(curl -s -w "%{http_code}" -o "$${RESPONSE}" -X POST \
|
||||
-H "Authorization: Token $$NEXTCLOUD_API_TOKEN" \
|
||||
-H "Content-Type: application/json" \
|
||||
-d "{\"download\":\"$${DOWNLOAD_URL}\", \"signature\":\"$${SIGNATURE}\"}" \
|
||||
https://apps.nextcloud.com/api/v1/apps/releases; \
|
||||
if [ $$? -ne 0 ]; then \
|
||||
echo "\x1b[31m❌ Error: Failed to release to Nextcloud App Store\x1b[0m"; \
|
||||
exit 1; \
|
||||
https://apps.nextcloud.com/api/v1/apps/releases); \
|
||||
cat "$$RESPONSE"; echo; \
|
||||
if [ "$$HTTP_CODE" = "400" ]; then \
|
||||
echo "\x1b[31m❌ Error 400: Invalid data, app too large, signature/cert issue, or not registered\x1b[0m"; exit 1; \
|
||||
elif [ "$$HTTP_CODE" = "401" ]; then \
|
||||
echo "\x1b[31m❌ Error 401: Not authenticated\x1b[0m"; exit 1; \
|
||||
elif [ "$$HTTP_CODE" = "403" ]; then \
|
||||
echo "\x1b[31m❌ Error 403: Not authorized\x1b[0m"; exit 1; \
|
||||
elif [ "$$HTTP_CODE" -ge 300 ]; then \
|
||||
echo "\x1b[31m❌ Unexpected error (HTTP $$HTTP_CODE)\x1b[0m"; exit 1; \
|
||||
fi; \
|
||||
rm -f "$$RESPONSE"; \
|
||||
echo "\x1b[32m🎉 Release successful!\x1b[0m";
|
||||
|
||||
|
||||
Reference in New Issue
Block a user