OUT_DIR?="bin" $(shell mkdir -p $(OUT_DIR)) .PHONY: default default: streamplace VERSION?=$(shell go run ./pkg/config/git/git.go -v) VERSION_NO_V=$(subst v,,$(VERSION)) VERSION_ELECTRON=$(subst -,-z,$(subst v,,$(VERSION))) UUID?=$(shell go run ./pkg/config/uuid/uuid.go) BRANCH?=$(shell go run ./pkg/config/git/git.go --branch) BUILDOS ?= $(shell uname -s | tr '[:upper:]' '[:lower:]') BUILDARCH ?= $(shell uname -m | tr '[:upper:]' '[:lower:]') ifeq ($(BUILDARCH),aarch64) BUILDARCH=arm64 endif ifeq ($(BUILDARCH),x86_64) BUILDARCH=amd64 endif MACOS_VERSION_FLAG= ifeq ($(BUILDOS),darwin) MACOS_VERSION_FLAG=-mmacosx-version-min=$(shell sw_vers -productVersion) endif BUILDDIR?=build-$(BUILDOS)-$(BUILDARCH) PKG_CONFIG_PATH=$(shell pwd)/$(BUILDDIR)/lib/pkgconfig:$(shell pwd)/$(BUILDDIR)/lib/gstreamer-1.0/pkgconfig:$(shell pwd)/$(BUILDDIR)/meson-uninstalled .PHONY: version version: @go run ./pkg/config/git/git.go -v \ && go run ./pkg/config/git/git.go --env -o .ci/build.env # _____ _____ # /\ | __ \| __ \ # / \ | |__) | |__) | # / /\ \ | ___/| ___/ # / ____ \| | | | # /_/ \_\_| |_| # front-end stuff innit .PHONY: install install: pnpm install .PHONY: app app: install pnpm run build .PHONY: app-cached app-cached: if [ ! -f js/app/dist/index.html ]; then $(MAKE) app; else echo "js/app/dist/index.html exists, not rebuilding, run make app to rebuild"; fi .PHONY: ci-ios ci-ios: version install app $(MAKE) ios $(MAKE) ci-upload-ios .PHONY: ci-android ci-android: version install android ci-upload-android .PHONY: ci-android-debug ci-android-debug: version install pnpm run app prebuild $(MAKE) android-debug $(MAKE) ci-upload-android-debug .PHONY: ci-android-release ci-android-release: version install pnpm run app prebuild $(MAKE) android-release $(MAKE) ci-upload-android-release ANDROID_KEYSTORE_PASSWORD?=streamplace ANDROID_KEYSTORE_ALIAS?=alias_name ANDROID_KEYSTORE_BASE64?= .PHONY: android-keystore android-keystore: if [ -n "$$ANDROID_KEYSTORE_BASE64" ]; then \ echo "$$ANDROID_KEYSTORE_BASE64" | base64 -d > my-release-key.keystore; \ fi; \ if [ ! -f my-release-key.keystore ]; then \ keytool -genkey -v -keystore my-release-key.keystore -alias alias_name -keyalg RSA -keysize 2048 -validity 10000 -storepass $(ANDROID_KEYSTORE_PASSWORD) -keypass $(ANDROID_KEYSTORE_PASSWORD) -dname "CN=Streamplace, OU=Streamplace, O=Streamplace, L=Streamplace, S=Streamplace, C=US"; \ fi .PHONY: android android: app .build/bundletool.jar $(MAKE) android-release $(MAKE) android-debug .PHONY: android-release android-release: .build/bundletool.jar android-keystore export NODE_ENV=production \ && cd ./js/app/android \ && ./gradlew :app:bundleRelease \ && cd - \ && mv ./js/app/android/app/build/outputs/bundle/release/app-release.aab ./bin/streamplace-$(VERSION)-android-release.aab \ && cd bin \ && java -jar ../.build/bundletool.jar build-apks --ks ../my-release-key.keystore --ks-key-alias alias_name --ks-pass pass:$(ANDROID_KEYSTORE_PASSWORD) --bundle=streamplace-$(VERSION)-android-release.aab --output=streamplace-$(VERSION)-android-release.apks --mode=universal \ && unzip streamplace-$(VERSION)-android-release.apks && mv universal.apk streamplace-$(VERSION)-android-release.apk && rm toc.pb .PHONY: android-debug android-debug: .build/bundletool.jar android-keystore export NODE_ENV=production \ && cd ./js/app/android \ && ./gradlew :app:bundleDebug \ && cd - \ && mv ./js/app/android/app/build/outputs/bundle/debug/app-debug.aab ./bin/streamplace-$(VERSION)-android-debug.aab \ && cd bin \ && java -jar ../.build/bundletool.jar build-apks --ks ../my-release-key.keystore --ks-key-alias alias_name --ks-pass pass:$(ANDROID_KEYSTORE_PASSWORD) --bundle=streamplace-$(VERSION)-android-debug.aab --output=streamplace-$(VERSION)-android-debug.apks --mode=universal \ && unzip streamplace-$(VERSION)-android-debug.apks && mv universal.apk streamplace-$(VERSION)-android-debug.apk && rm toc.pb .PHONY: ios ios: app xcodebuild \ -workspace ./js/app/ios/Streamplace.xcworkspace \ -sdk iphoneos \ -configuration Release \ -scheme Streamplace \ -archivePath ./bin/streamplace-$(VERSION)-ios-release.xcarchive \ CODE_SIGN_IDENTITY=- \ AD_HOC_CODE_SIGNING_ALLOWED=YES \ CODE_SIGN_STYLE=Automatic \ DEVELOPMENT_TEAM=ZZZZZZZZZZ \ clean archive | xcpretty \ && cd bin \ && tar -czvf streamplace-$(VERSION)-ios-release.xcarchive.tar.gz streamplace-$(VERSION)-ios-release.xcarchive # _____ ____ # / ____|/ __ \ # | | __| | | | # | | |_ | | | | # | |__| | |__| | # \_____|\____/ .PHONY: rtcrec rtcrec: go build -o $(BUILDDIR)/rtcrec ./pkg/rtcrec/cmd/... # Go takes care of this automatically but we can speed up the build by downloading go deps while the build is running .PHONY: godeps godeps: go get ./pkg/... # __ __ ______ _____ ____ _ _ # | \/ | ____|/ ____|/ __ \| \ | | # | \ / | |__ | (___ | | | | \| | # | |\/| | __| \___ \| | | | . ` | # | | | | |____ ____) | |__| | |\ | # |_| |_|______|_____/ \____/|_| \_| BASE_OPTS = \ --buildtype=debugoptimized \ -D "gst-plugins-base:audioresample=enabled" \ -D "gst-plugins-base:playback=enabled" \ -D "gst-plugins-base:opus=enabled" \ -D "gst-plugins-base:gio-typefinder=enabled" \ -D "gst-plugins-base:videotestsrc=enabled" \ -D "gst-plugins-base:videoconvertscale=enabled" \ -D "gst-plugins-base:typefind=enabled" \ -D "gst-plugins-base:compositor=enabled" \ -D "gst-plugins-base:videorate=enabled" \ -D "gst-plugins-base:app=enabled" \ -D "gst-plugins-base:audiorate=enabled" \ -D "gst-plugins-base:audiotestsrc=enabled" \ -D "gst-plugins-base:audioconvert=enabled" \ -D "gst-plugins-good:matroska=enabled" \ -D "gst-plugins-good:multifile=enabled" \ -D "gst-plugins-good:rtp=enabled" \ -D "gst-plugins-bad:fdkaac=enabled" \ -D "gst-plugins-bad:rtmp2=enabled" \ -D "gst-plugins-good:audioparsers=enabled" \ -D "gst-plugins-good:isomp4=enabled" \ -D "gst-plugins-good:png=enabled" \ -D "gst-plugins-good:videobox=enabled" \ -D "gst-plugins-good:jpeg=enabled" \ -D "gst-plugins-good:audioparsers=enabled" \ -D "gst-plugins-bad:videoparsers=enabled" \ -D "gst-plugins-bad:mpegtsmux=enabled" \ -D "gst-plugins-bad:mpegtsdemux=enabled" \ -D "gst-plugins-bad:codectimestamper=enabled" \ -D "gst-plugins-bad:opus=enabled" \ -D "gst-plugins-bad:rtmp2=enabled" \ -D "gst-plugins-good:flv=enabled" \ -D "gst-plugins-ugly:x264=enabled" \ -D "gst-plugins-ugly:gpl=enabled" \ -D "x264:asm=enabled" \ -D "gstreamer-full:gst-full=enabled" \ -D "gstreamer-full:gst-full-plugins=libgstflv.a;libgstrtmp2.a;libgstopusparse.a;libgstcodectimestamper.a;libgstrtp.a;libgstaudioresample.a;libgstlibav.a;libgstmatroska.a;libgstmultifile.a;libgstjpeg.a;libgstaudiotestsrc.a;libgstaudioconvert.a;libgstaudioparsers.a;libgstfdkaac.a;libgstisomp4.a;libgstapp.a;libgstvideoconvertscale.a;libgstvideobox.a;libgstvideorate.a;libgstpng.a;libgstcompositor.a;libgstaudiorate.a;libgstx264.a;libgstopus.a;libgstvideotestsrc.a;libgstvideoparsersbad.a;libgstaudioparsers.a;libgstmpegtsmux.a;libgstmpegtsdemux.a;libgstplayback.a;libgsttypefindfunctions.a;libgstcoretracers.a;libgstcodec2json.a" \ -D "gstreamer-full:gst-full-libraries=gstreamer-controller-1.0,gstreamer-plugins-base-1.0,gstreamer-pbutils-1.0" \ -D "gstreamer-full:gst-full-elements=coreelements:concat,filesrc,filesink,queue,queue2,multiqueue,typefind,tee,capsfilter,fakesink,identity" \ -D "gstreamer-full:bad=enabled" \ -D "gstreamer-full:tls=disabled" \ -D "gstreamer-full:libav=enabled" \ -D "gstreamer-full:ugly=enabled" \ -D "gstreamer-full:gpl=enabled" \ -D "gstreamer-full:gst-full-typefind-functions=" \ -D "gstreamer:coretracers=enabled" \ -D "gstreamer-full:glib_assert=false" \ -D "gstreamer:glib_assert=false" \ -D "gst-plugins-good:glib_assert=false" \ -D "gst-plugins-bad:glib_assert=false" \ -D "gst-plugins-base:glib_assert=false" \ -D "gst-plugins-ugly:glib_assert=false" \ -D "glib:glib_assert=false" \ -D "glib:glib_assert=false" \ -D "gst-libav:glib_assert=false" \ -D "gst-plugins-good:adaptivedemux2=disabled" \ -D "gst-plugins-bad:codec2json=enabled" STATIC_OPTS = \ $(BASE_OPTS) \ -D "gstreamer-full:gst-full-target-type=static_library" \ -D "gstreamer:registry=false" SHARED_OPTS = \ $(BASE_OPTS) \ -D "FFmpeg:default_library=shared" \ -D "gstreamer:registry=true" .PHONY: meson meson: $(MAKE) meson-setup $(MAKE) meson-compile MESON_SETUP_OPTS?= .PHONY: meson-setup meson-setup: @meson setup $(BUILDDIR) $(MESON_SETUP_OPTS) @meson configure $(BUILDDIR) $(OPTS) .PHONY: meson-setup-static meson-setup-static: @meson setup $(BUILDDIR) $(MESON_SETUP_OPTS) $(STATIC_OPTS) @meson configure $(BUILDDIR) $(STATIC_OPTS) .PHONY: meson-compile meson-compile: meson install -C $(BUILDDIR) # _____ _______ _______ _____ _____ # / ____|__ __|/\|__ __|_ _/ ____| # | (___ | | / \ | | | || | # \___ \ | | / /\ \ | | | || | # ____) | | |/ ____ \| | _| || |____ # |_____/ |_/_/ \_\_| |_____\_____| # Production distributions of Streamplace with bundled node and app. static: $(MAKE) meson-setup meson-compile app OPTS=$(STATIC_OPTS) -j16 $(MAKE) streamplace .PHONY: static-test static-test: go install github.com/jstemmer/go-junit-report/v2@latest \ && PKG_CONFIG_PATH=$(shell realpath $(BUILDDIR)/meson-uninstalled) \ bash -euo pipefail -c "go test -p 1 -timeout 300s ./pkg/... -v | tee /dev/stderr | go-junit-report -out test.xml" # _____ ________ __ # | __ \| ____\ \ / / # | | | | |__ \ \ / / # | | | | __| \ \/ / # | |__| | |____ \ / # |_____/|______| \/ # Local builds of Streamplace with node and app. .PHONY: dev-setup dev-setup: $(MAKE) -j16 app-cached dev-setup-meson .PHONY: dev dev: app-cached if [ ! -d $(BUILDDIR) ]; then $(MAKE) dev-setup; fi cp ./util/streamplace-dev.sh $(BUILDDIR)/streamplace $(MAKE) dev-rust PKG_CONFIG_PATH=$(PKG_CONFIG_PATH) \ CGO_LDFLAGS="$(MACOS_VERSION_FLAG)" \ LD_LIBRARY_PATH=$(BUILDDIR)/lib go build -tags mainnet -o $(BUILDDIR)/libstreamplace ./cmd/libstreamplace/... .PHONY: dev-setup-meson dev-setup-meson: $(MAKE) dev-setup-meson-configure $(MAKE) meson-compile .PHONY: dev-setup-meson-configure dev-setup-meson-configure: meson setup --default-library=shared $(BUILDDIR) $(SHARED_OPTS) meson configure --default-library=shared $(BUILDDIR) $(SHARED_OPTS) .PHONY: dev-rust dev-rust: .build/bin/uniffi-bindgen-go-forked cargo build EXT=so; \ if [ "$(BUILDOS)" = "darwin" ]; then EXT=dylib; fi; \ .build/bin/uniffi-bindgen-go-forked --out-dir pkg/iroh/generated --library ./target/debug/libiroh_streamplace.$$EXT \ && mkdir -p $(BUILDDIR)/rust/iroh-streamplace/ \ && mkdir -p $(BUILDDIR)/lib/ \ && cp ./target/debug/libiroh_streamplace.$$EXT $(BUILDDIR)/rust/iroh-streamplace/libiroh_streamplace.$$EXT.tmp \ && mv $(BUILDDIR)/rust/iroh-streamplace/libiroh_streamplace.$$EXT.tmp $(BUILDDIR)/rust/iroh-streamplace/libiroh_streamplace.$$EXT \ && cp ./target/debug/libiroh_streamplace.$$EXT $(BUILDDIR)/lib/libiroh_streamplace.$$EXT.tmp \ && mv $(BUILDDIR)/lib/libiroh_streamplace.$$EXT.tmp $(BUILDDIR)/lib/libiroh_streamplace.$$EXT .PHONY: dev-test dev-test: go install github.com/jstemmer/go-junit-report/v2@latest \ && PKG_CONFIG_PATH=$(PKG_CONFIG_PATH) \ LD_LIBRARY_PATH=$(shell realpath $(BUILDDIR))/lib \ CGO_LDFLAGS="-lm" \ bash -euo pipefail -c "go test -p 1 -timeout 30m ./pkg/... -v | tee /dev/stderr | go-junit-report -out test.xml" .PHONY: iroh-test iroh-test: $(MAKE) dev-rust PKG_CONFIG_PATH=$(PKG_CONFIG_PATH) \ LD_LIBRARY_PATH=$(shell realpath $(BUILDDIR))/lib \ go build -o $(BUILDDIR)/iroh-test ./pkg/iroh/iroh_test/... # _ _____ _ _ _______ _____ _ _ _____ # | | |_ _| \ | |__ __|_ _| \ | |/ ____| # | | | | | \| | | | | | | \| | | __ # | | | | | . ` | | | | | | . ` | | |_ | # | |____ _| |_| |\ | | | _| |_| |\ | |__| | # |______|_____|_| \_| |_| |_____|_| \_|\_____| .PHONY: check check: install $(MAKE) golangci-lint pnpm run check if [ "`gofmt -l . | wc -l`" -gt 0 ]; then echo 'gofmt failed, run make fix'; exit 1; fi RUSTFLAGS="-D warnings" cargo check .PHONY: fix fix: pnpm run fix gofmt -w . cargo fix --allow-dirty .PHONY: golangci-lint golangci-lint: @PKG_CONFIG_PATH=$(PKG_CONFIG_PATH) \ go run github.com/golangci/golangci-lint/v2/cmd/golangci-lint run -c ./.golangci.yaml # _ ________ _______ _____ ____ _ _ _____ # | | | ____\ \ / /_ _/ ____/ __ \| \ | |/ ____| # | | | |__ \ V / | || | | | | | \| | (___ # | | | __| > < | || | | | | | . ` |\___ \ # | |____| |____ / . \ _| || |___| |__| | |\ |____) | # |______|______/_/ \_\_____\_____\____/|_| \_|_____/ # Mostly codegen for lexicons. .PHONY: lexicons lexicons: $(MAKE) go-lexicons \ && $(MAKE) js-lexicons \ && $(MAKE) md-lexicons \ && make fix .PHONY: go-lexicons go-lexicons: rm -rf ./pkg/streamplace \ && mkdir -p ./pkg/streamplace \ && rm -rf ./pkg/streamplace/cbor_gen.go \ && $(MAKE) lexgen \ && sed -i.bak 's/\tlexutil\.RegisterType/\/\/\tlexutil.RegisterType/' $$(find ./pkg/streamplace -type f) \ && go run golang.org/x/tools/cmd/goimports@latest -w $$(find ./pkg/streamplace -type f) \ && go run ./pkg/gen/gen.go \ && $(MAKE) lexgen \ && find . | grep bak$$ | xargs rm \ && rm -rf api .PHONY: js-lexicons js-lexicons: node_modules/.bin/lex gen-api ./js/streamplace/src/lexicons $$(find ./lexicons -type f -name '*.json') --yes \ && rm -rf ./js/streamplace/src/lexicons/types/com ./js/streamplace/src/lexicons/types/app \ && sed -i.bak "s/^..port.*app\/bsky.*//g" $$(find ./js/streamplace/src/lexicons -type f) \ && sed -i.bak "s/^..port.*com\/atproto.*//g" $$(find ./js/streamplace/src/lexicons -type f) \ && sed -i.bak "s/\(..port .*\)\.js\(.*\)/\1\2/g" $$(find ./js/streamplace/src/lexicons -type f) \ && sed -i.bak 's/AppBskyGraphBlock\.Main/AppBskyGraphBlock\.Record/' $$(find ./js/streamplace/src/lexicons/types/place/stream -type f) \ && sed -i.bak 's/PlaceStreamMultistreamTarget\.Main/PlaceStreamMultistreamTarget\.Record/' $$(find ./js/streamplace/src/lexicons/types/place/stream -type f) \ && sed -i.bak 's/PlaceStreamChatProfile\.Main/PlaceStreamChatProfile\.Record/' $$(find ./js/streamplace/src/lexicons/types/place/stream -type f) \ && for x in $$(find ./js/streamplace/src/lexicons -type f -name '*.ts'); do \ echo 'import { ComAtprotoSyncGetRepo, AppBskyRichtextFacet, AppBskyGraphBlock, ComAtprotoRepoStrongRef, AppBskyActorDefs, ComAtprotoSyncListRepos, AppBskyActorGetProfile, AppBskyFeedGetFeedSkeleton, ComAtprotoIdentityResolveHandle, ComAtprotoModerationCreateReport, ComAtprotoRepoCreateRecord, ComAtprotoRepoDeleteRecord, ComAtprotoRepoDescribeRepo, ComAtprotoRepoGetRecord, ComAtprotoRepoListRecords, ComAtprotoRepoPutRecord, ComAtprotoRepoUploadBlob, ComAtprotoServerDescribeServer, ComAtprotoSyncGetRecord, ComAtprotoSyncListReposComAtprotoRepoCreateRecord, ComAtprotoRepoDeleteRecord, ComAtprotoRepoGetRecord, ComAtprotoRepoListRecords, ComAtprotoIdentityRefreshIdentity } from "@atproto/api"' >> $$x; \ done \ && npx prettier --ignore-unknown --write $$(find ./js/streamplace/src/lexicons -type f -name '*.ts') \ && find . | grep bak$$ | xargs rm .PHONY: md-lexicons md-lexicons: find "js/docs/src/content/docs/lex-reference" -type f -name '*.md' -delete \ && pnpm exec lexmd \ ./lexicons \ .build/temp \ subprojects/atproto/lexicons \ js/docs/src/content/docs/lex-reference/openapi.json \ && ls -R .build/temp \ && cp -rf .build/temp/place/stream/* js/docs/src/content/docs/lex-reference/ \ && rm -rf .build/temp \ && $(MAKE) fix .PHONY: lexgen lexgen: $(MAKE) lexgen-types $(MAKE) lexgen-server .PHONY: lexgen-types lexgen-types: go run github.com/bluesky-social/indigo/cmd/lexgen \ -outdir ./pkg/spxrpc \ --build-file util/lexgen-types.json \ --external-lexicons subprojects/atproto/lexicons \ lexicons/place/stream \ ./subprojects/atproto/lexicons .PHONY: lexgen-server lexgen-server: mkdir -p ./pkg/spxrpc \ && go run github.com/bluesky-social/indigo/cmd/lexgen \ --gen-server \ --types-import place.stream:stream.place/streamplace/pkg/streamplace \ --types-import app.bsky:github.com/bluesky-social/indigo/api/bsky \ --types-import com.atproto:github.com/bluesky-social/indigo/api/atproto \ --types-import chat.bsky:github.com/bluesky-social/indigo/api/chat \ --types-import tools.ozone:github.com/bluesky-social/indigo/api/ozone \ -outdir ./pkg/spxrpc \ --build-file util/lexgen-types.json \ --external-lexicons subprojects/atproto/lexicons \ --package spxrpc \ lexicons/place/stream \ lexicons/app/bsky \ lexicons/com/atproto .PHONY: ci-lexicons ci-lexicons: $(MAKE) lexicons \ && if ! git diff --exit-code >/dev/null; then echo "lexicons are out of date, run 'make lexicons' to fix"; git diff; exit 1; fi # _______ ______ _____ _______ _____ _ _ _____ # |__ __| ____|/ ____|__ __|_ _| \ | |/ ____| # | | | |__ | (___ | | | | | \| | | __ # | | | __| \___ \ | | | | | . ` | | |_ | # | | | |____ ____) | | | _| |_| |\ | |__| | # |_| |______|_____/ |_| |_____|_| \_|\_____| # test to make sure we haven't added any more dynamic dependencies LINUX_LINK_COUNT=5 .PHONY: link-test-linux link-test-linux: count=$(shell ldd ./build-linux-amd64/streamplace | wc -l) \ && echo $$count \ && if [ "$$count" != "$(LINUX_LINK_COUNT)" ]; then echo "ldd reports new libaries linked! want $(LINUX_LINK_COUNT) got $$count" \ && ldd ./build-linux-amd64/streamplace \ && exit 1; \ fi MACOS_LINK_COUNT=10 .PHONY: link-test-macos link-test-macos: count=$(shell otool -L ./build-darwin-arm64/streamplace | wc -l | xargs) \ && echo $$count \ && if [ "$$count" != "$(MACOS_LINK_COUNT)" ]; then echo "otool -L reports new libaries linked! want $(MACOS_LINK_COUNT) got $$count" \ && otool -L ./build-darwin-arm64/streamplace \ && exit 1; \ fi WINDOWS_LINK_COUNT=19 .PHONY: link-test-windows link-test-windows: count=$(shell x86_64-w64-mingw32-objdump -p ./build-windows-amd64/streamplace.exe | grep "DLL Name" | tr '[:upper:]' '[:lower:]' | sort | uniq | wc -l | xargs) \ && echo $$count \ && if [ "$$count" != "$(WINDOWS_LINK_COUNT)" ]; then echo "x86_64-w64-mingw32-objdump -p reports new libaries linked! want $(WINDOWS_LINK_COUNT) got $$count" \ && x86_64-w64-mingw32-objdump -p ./build-windows-amd64/streamplace.exe | grep "DLL Name" \ && exit 1; \ fi .PHONY: ci ci: version install app node-all-platforms ci-upload-node .PHONY: ci-desktop-darwin ci-desktop-darwin: version install ./util/mac-codesign.sh \ && for arch in amd64 arm64; do \ curl -v --fail-with-body -O "$$CI_API_V4_URL/projects/$$CI_PROJECT_ID/packages/generic/$(BRANCH)/$(VERSION)/streamplace-$(VERSION)-darwin-$$arch.tar.gz" || exit 1 \ && tar -xzvf streamplace-$(VERSION)-darwin-$$arch.tar.gz \ && ./streamplace --version \ && ./streamplace self-test \ && mkdir -p build-darwin-$$arch \ && mv ./streamplace ./build-darwin-$$arch/streamplace; \ done \ && $(MAKE) desktop-darwin \ && for arch in amd64 arm64; do \ export file=streamplace-desktop-$(VERSION)-darwin-$$arch.zip \ && $(MAKE) ci-upload-file upload_file=$$file \ && export file=streamplace-desktop-$(VERSION)-darwin-$$arch.dmg \ && $(MAKE) ci-upload-file upload_file=$$file; \ done .PHONY: ci-test ci-test: app meson setup $(BUILDDIR) $(OPTS) meson test -C $(BUILDDIR) go-tests .PHONY: ci-npm-release ci-npm-release: install echo //registry.npmjs.org/:_authToken=$$NPM_TOKEN > ~/.npmrc \ && npx lerna publish from-package --yes .build/bin/uniffi-bindgen-go-forked: mkdir -p .build \ && cargo install uniffi-bindgen-go --git https://github.com/kegsay/uniffi-bindgen-go --rev f1f1064871faf05377c75e098d525d530d402d38 --root .build \ && mv .build/bin/uniffi-bindgen-go .build/bin/uniffi-bindgen-go-forked .build/bundletool.jar: mkdir -p .build \ && curl -L -o ./.build/bundletool.jar https://github.com/google/bundletool/releases/download/1.17.0/bundletool-all-1.17.0.jar .PHONY: selftest-macos selftest-macos: js/desktop/out/Streamplace-darwin-arm64/Streamplace.app/Contents/MacOS/Streamplace -- --self-test # _____ _____ ____ _____ _____ # / ____| __ \ / __ \ / ____/ ____| # | | | |__) | | | | (___| (___ # | | | _ /| | | |\___ \\___ \ # | |____| | \ \| |__| |____) |___) | # \_____|_| \_\\____/|_____/_____/ # _____ ____ __ __ _____ _____ _ _____ _ _ _____ # / ____/ __ \| \/ | __ \_ _| | |_ _| \ | |/ ____| # | | | | | | \ / | |__) || | | | | | | \| | | __ # | | | | | | |\/| | ___/ | | | | | | | . ` | | |_ | # | |___| |__| | | | | | _| |_| |____ _| |_| |\ | |__| | # \_____\____/|_| |_|_| |_____|______|_____|_| \_|\_____| # Cross-compiling is exclusively right now from linux-amd64 to: # linux-amd64 (native) # linux-arm64 # windows-amd64 # darwin-amd64 # darwin-arm64 .PHONY: node-all-platforms node-all-platforms: app meson setup build-linux-amd64 $(OPTS) --buildtype debugoptimized meson compile -C build-linux-amd64 archive $(MAKE) link-test-linux $(MAKE) linux-arm64 $(MAKE) windows-amd64 $(MAKE) windows-amd64-startup-test $(MAKE) desktop-linux $(MAKE) desktop-windows-amd64 .PHONY: desktop-linux desktop-linux: $(MAKE) desktop-linux-amd64 $(MAKE) desktop-linux-arm64 .PHONY: desktop-linux-amd64 desktop-linux-amd64: cd js/desktop \ && pnpm run make --platform linux --arch x64 \ && cd - \ && mv "js/desktop/out/make/AppImage/x64/Streamplace-$(VERSION_ELECTRON)-x64.AppImage" ./bin/streamplace-desktop-$(VERSION)-linux-amd64.AppImage .PHONY: desktop-linux-arm64 desktop-linux-arm64: cd js/desktop \ && pnpm run make --platform linux --arch arm64 \ && cd - \ && mv "js/desktop/out/make/AppImage/arm64/Streamplace-$(VERSION_ELECTRON)-arm64.AppImage" ./bin/streamplace-desktop-$(VERSION)-linux-arm64.AppImage .PHONY: desktop-windows-amd64 desktop-windows-amd64: cd js/desktop \ && pnpm run make --platform win32 --arch x64 \ && cd - \ && export SUM=$$(cat ./js/desktop/out/make/squirrel.windows/x64/streamplace_desktop-$(VERSION_ELECTRON)-full.nupkg | openssl sha1 | awk '{ print $$2 }') \ && echo $$SUM > ./bin/streamplace-desktop-$(VERSION)-windows-amd64.nupkg.sha1 \ && mv "js/desktop/out/make/squirrel.windows/x64/streamplace_desktop-$(VERSION_ELECTRON)-full.nupkg" ./bin/streamplace-desktop-$(VERSION)-windows-amd64.$$SUM.nupkg \ && mv "js/desktop/out/make/squirrel.windows/x64/Streamplace-$(VERSION_ELECTRON) Setup.exe" ./bin/streamplace-desktop-$(VERSION)-windows-amd64.exe .PHONY: streamplace streamplace: app-cached meson-setup-static meson compile -C $(BUILDDIR) streamplace | grep -v drectve .PHONY: archive archive: app-cached meson-setup-static godeps meson compile -C $(BUILDDIR) archive | grep -v drectve .PHONY: linux-amd64 linux-amd64: $(MAKE) -j $(shell nproc) archive $(MAKE) link-test-linux .PHONY: linux-arm64 linux-arm64: rustup target add aarch64-unknown-linux-gnu \ && CC=aarch64-linux-gnu-gcc \ LD=aarch64-linux-gnu-ld \ CROSS_COMPILE=1 \ MESON_SETUP_OPTS="--cross-file util/linux-arm64-gnu.ini" \ BUILDDIR=build-linux-arm64 \ CARGO_TARGET_AARCH64_UNKNOWN_LINUX_GNU_LINKER=aarch64-linux-gnu-gcc \ $(MAKE) -j $(shell nproc) archive .PHONY: windows-amd64 windows-amd64: rustup target add x86_64-pc-windows-gnu \ && CC=x86_64-w64-mingw32-gcc \ LD=x86_64-w64-mingw32-ld \ CROSS_COMPILE=1 \ MESON_SETUP_OPTS="--cross-file util/windows-amd64-gnu.ini" \ BUILDDIR=build-windows-amd64 \ $(MAKE) -j $(shell nproc) archive .PHONY: darwin-amd64 darwin-amd64: export CC=x86_64-apple-darwin24.4-clang \ && export CC_X86_64_APPLE_DARWIN=x86_64-apple-darwin24.4-clang \ && export CXX_X86_64_APPLE_DARWIN=x86_64-apple-darwin24.4-clang++ \ && export AR_X86_64_APPLE_DARWIN=x86_64-apple-darwin24.4-ar \ && export CARGO_TARGET_X86_64_APPLE_DARWIN_LINKER=x86_64-apple-darwin24.4-clang \ && export LD=x86_64-apple-darwin24.4-ld \ && export CROSS_COMPILE=1 \ && export MESON_SETUP_OPTS="--cross-file util/osxcross-darwin-amd64.ini" \ && export BUILDDIR=build-darwin-amd64 \ && $(MAKE) -j $(shell nproc) streamplace \ && ./util/osxcross-codesign.sh ./build-darwin-amd64/streamplace \ && mkdir -p bin \ && cd build-darwin-amd64 \ && tar -czvf ../bin/streamplace-$(VERSION)-darwin-amd64.tar.gz ./streamplace \ && cd - .PHONY: darwin-arm64gofmt -w . darwin-arm64: export CC=aarch64-apple-darwin24.4-clang \ && export CC_AARCH64_APPLE_DARWIN=aarch64-apple-darwin24.4-clang \ && export CXX_AARCH64_APPLE_DARWIN=aarch64-apple-darwin24.4-clang++ \ && export AR_AARCH64_APPLE_DARWIN=aarch64-apple-darwin24.4-ar \ && export CARGO_TARGET_AARCH64_APPLE_DARWIN_LINKER=aarch64-apple-darwin24.4-clang \ && export LD=aarch64-apple-darwin24.4-ld \ && export CROSS_COMPILE=1 \ && export MESON_SETUP_OPTS="--cross-file util/osxcross-darwin-arm64.ini" \ && export BUILDDIR=build-darwin-arm64 \ && $(MAKE) -j $(shell nproc) streamplace \ && ./util/osxcross-codesign.sh ./build-darwin-arm64/streamplace \ && mkdir -p bin \ && cd build-darwin-arm64 \ && tar -czvf ../bin/streamplace-$(VERSION)-darwin-arm64.tar.gz ./streamplace \ && cd - # unbuffer here is a workaround for wine trying to pop up a terminal window and failing .PHONY: windows-amd64-startup-test windows-amd64-startup-test: bash -c 'set -euo pipefail && unbuffer wine64 ./build-windows-amd64/streamplace.exe self-test | cat' .PHONY: desktop-darwin-amd64 desktop-darwin-amd64: echo "TODO" .PHONY: desktop-darwin-arm64 desktop-darwin-arm64: echo "TODO" .PHONY: desktop-darwin desktop-darwin: export DEBUG="electron-osx-sign*" \ && cd js/desktop \ && pnpm run make --platform darwin --arch arm64 \ && pnpm run make --platform darwin --arch x64 \ && cd - \ && mv js/desktop/out/make/Streamplace-$(VERSION_ELECTRON)-x64.dmg ./bin/streamplace-desktop-$(VERSION)-darwin-amd64.dmg \ && mv js/desktop/out/make/Streamplace-$(VERSION_ELECTRON)-arm64.dmg ./bin/streamplace-desktop-$(VERSION)-darwin-arm64.dmg \ && mv js/desktop/out/make/zip/darwin/x64/Streamplace-darwin-x64-$(VERSION_ELECTRON).zip ./bin/streamplace-desktop-$(VERSION)-darwin-amd64.zip \ && mv js/desktop/out/make/zip/darwin/arm64/Streamplace-darwin-arm64-$(VERSION_ELECTRON).zip ./bin/streamplace-desktop-$(VERSION)-darwin-arm64.zip # link your local version of mist for dev .PHONY: link-mist link-mist: rm -rf subprojects/mistserver ln -s $$(realpath ../mistserver) ./subprojects/mistserver # link your local version of gstreamer .PHONY: link-gstreamer link-gstreamer: rm -rf subprojects/gstreamer-full ln -s $$(realpath ../gstreamer) ./subprojects/gstreamer-full # link your local version of ffmpeg for dev .PHONY: link-ffmpeg link-ffmpeg: rm -rf subprojects/FFmpeg ln -s $$(realpath ../ffmpeg) ./subprojects/FFmpeg # _____ ____ _____ _ ________ _____ # | __ \ / __ \ / ____| |/ / ____| __ \ # | | | | | | | | | ' /| |__ | |__) | # | | | | | | | | | < | __| | _ / # | |__| | |__| | |____| . \| |____| | \ \ # |_____/ \____/ \_____|_|\_\______|_| \_\ # Actually, Podman. I'm not sure if these commands will actually run on Docker, untested. .PHONY: docker docker: docker build -f docker/local.Dockerfile -t dist.stream.place/streamplace/streamplace:local . .PHONY: docker-build docker-build: docker-build-builder docker-build-in-container .PHONY: docker-test docker-test: docker-build-builder docker-test-in-container DOCKER_BUILD_OPTS?= BUILDER_TARGET?=builder .PHONY: docker-build-builder docker-build-builder: podman build --target=$(BUILDER_TARGET) --os=linux --arch=amd64 -f docker/build.Dockerfile -t dist.stream.place/streamplace/streamplace:$(BUILDER_TARGET) $(DOCKER_BUILD_OPTS) . .PHONY: precommit precommit: dockerfile-hash-precommit .PHONY: dockefile-hash-precommit dockerfile-hash-precommit: @bash -c 'printf "variables:\n DOCKERFILE_HASH: `git hash-object docker/build.Dockerfile`" > .ci/dockerfile-hash.yaml' \ && git add .ci/dockerfile-hash.yaml # tricks the github action for golangci-lint to run inside a container .PHONY: golangci-lint-container golangci-lint-container: docker-build-builder podman run \ -v $$(pwd):$$(pwd) \ -w $$(pwd) \ -e PKG_CONFIG_PATH=$$(pwd)/build-linux-amd64/lib/pkgconfig \ -d \ --name golangci-lint \ dist.stream.place/streamplace/streamplace:$(BUILDER_TARGET) \ tail -f /dev/null podman exec golangci-lint mkdir -p js/app/dist podman exec golangci-lint touch js/app/dist/index.html podman exec golangci-lint make dev # runs a command in the build container, building if necessary IN_CONTAINER_CMD?=echo 'usage: make in-container IN_CONTAINER_CMD=\"\"' DOCKER_BIN?=podman DOCKER_REF?=dist.stream.place/streamplace/streamplace:$(BUILDER_TARGET) DOCKER_OPTS?= .PHONY: in-container in-container: docker-build-builder $(DOCKER_BIN) run $(DOCKER_OPTS) -v $$(pwd):$$(pwd) -w $$(pwd) --rm $(DOCKER_REF) bash -c "$(IN_CONTAINER_CMD)" .PHONY: docker-shell docker-shell: $(MAKE) in-container IN_CONTAINER_CMD="bash" DOCKER_OPTS="-it -v $$(pwd):$$(pwd) -w $$(pwd)" STREAMPLACE_URL?=https://git.stream.place/streamplace/streamplace/-/package_files/10122/download .PHONY: docker-release docker-release: cd docker \ && docker build -f release.Dockerfile \ --build-arg TARGETARCH=$(BUILDARCH) \ --build-arg STREAMPLACE_URL=$(STREAMPLACE_URL) \ -t dist.stream.place/streamplace/streamplace \ . .PHONY: docker-mistserver docker-mistserver: cd docker \ && docker build -f mistserver.Dockerfile \ --build-arg TARGETARCH=$(BUILDARCH) \ --build-arg STREAMPLACE_URL=$(STREAMPLACE_URL) \ -t dist.stream.place/streamplace/streamplace:mistserver \ # _____ _____ _____ _______ _____ ____ # | __ \_ _|/ ____|__ __| __ \ / __ \ # | | | || | | (___ | | | |__) | | | | # | | | || | \___ \ | | | _ /| | | | # | |__| || |_ ____) | | | | | \ \| |__| | # |_____/_____|_____/ |_| |_| \_\\____/ # distro: cut new versions, build apt and brew packages, upload to S3/gitlab .PHONY: release release: install $(MAKE) lexicons pnpm run release .PHONY: ci-release ci-release: go install gitlab.com/gitlab-org/release-cli/cmd/release-cli go run ./pkg/config/git/git.go -release -o release.yml release-cli create-from-file --file release.yml .PHONY: homebrew homebrew: $(MAKE) ci-download-file download_file=streamplace-$(VERSION)-linux-amd64.tar.gz $(MAKE) ci-download-file download_file=streamplace-$(VERSION)-linux-arm64.tar.gz $(MAKE) ci-download-file download_file=streamplace-$(VERSION)-darwin-amd64.tar.gz $(MAKE) ci-download-file download_file=streamplace-$(VERSION)-darwin-arm64.tar.gz git clone git@github.com:streamplace/homebrew-streamplace.git /tmp/homebrew-streamplace go run ./pkg/config/git/git.go -homebrew -o /tmp/homebrew-streamplace/Formula/streamplace.rb .PHONY: ci-homebrew ci-homebrew: git config --global user.name "Streamplace Homebrew Robot" git config --global user.email support@stream.place mkdir -p ~/.ssh echo "Host * \n\ StrictHostKeyChecking no \n\ UserKnownHostsFile=/dev/null" > ~/.ssh/config echo "$$CI_HOMEBREW_KEY_BASE64" | base64 -d > ~/.ssh/id_ed25519 chmod 600 ~/.ssh/id_ed25519 chmod 600 ~/.ssh/config $(MAKE) homebrew cd /tmp/homebrew-streamplace \ && git add . \ && git commit -m "Update streamplace $(VERSION)" \ && git push FPM_BASE_OPTS= \ -s dir \ -t deb \ -v $(VERSION_NO_V) \ --force \ --license=GPL-3.0-or-later \ --maintainer="Streamplace " \ --vendor="Streamplace" \ --url="https://stream.place" SP_ARCH_NAME?=amd64 .PHONY: deb-pkg deb-pkg: fpm $(FPM_BASE_OPTS) \ -n streamplace \ -a $(SP_ARCH_NAME) \ -p bin/streamplace-$(VERSION)-linux-$(SP_ARCH_NAME).deb \ --deb-systemd=util/systemd/streamplace.service \ --deb-systemd-restart-after-upgrade \ --after-install=util/systemd/after-install.sh \ --description="Live video for the AT Protocol. Solving video for everybody forever." \ build-linux-$(SP_ARCH_NAME)/streamplace=/usr/bin/streamplace .PHONY: pkg-linux-amd64 pkg-linux-amd64: $(MAKE) deb-pkg SP_ARCH_NAME=amd64 .PHONY: pkg-linux-arm64 pkg-linux-arm64: $(MAKE) deb-pkg SP_ARCH_NAME=arm64 .PHONY: pkg-darwin-amd64 pkg-darwin-amd64: echo todo .PHONY: pkg-darwin-arm64 pkg-darwin-arm64: echo todo .PHONY: pkg-windows-amd64 pkg-windows-amd64: echo todo .PHONY: deb-release deb-release: aptly repo create -distribution=all -component=main streamplace-releases aptly mirror create old-version $$S3_PUBLIC_URL/debian all aptly mirror update old-version aptly repo import old-version streamplace-releases streamplace aptly repo add streamplace-releases \ bin/streamplace-$(VERSION)-linux-arm64.deb \ bin/streamplace-$(VERSION)-linux-amd64.deb aptly snapshot create streamplace-$(VERSION) from repo streamplace-releases aptly publish snapshot -distribution=all streamplace-$(VERSION) s3:streamplace-releases: .PHONY: ci-deb-release ci-deb-release: $(MAKE) ci-download-file download_file=streamplace-$(VERSION)-linux-amd64.deb $(MAKE) ci-download-file download_file=streamplace-$(VERSION)-linux-arm64.deb echo $$CI_SIGNING_KEY_BASE64 | base64 -d | gpg --import gpg --armor --export | gpg --no-default-keyring --keyring trustedkeys.gpg --import echo '{"S3PublishEndpoints":{"streamplace-releases":{"region":"'$$S3_REGION'","bucket":"'$$S3_BUCKET_NAME'","endpoint":"'$$S3_ENDPOINT'","acl":"public-read","prefix":"debian"}}}' > ~/.aptly.conf $(MAKE) deb-release # _____ _____ # / ____|_ _| # | | | | # | | | | # | |____ _| |_ # \_____|_____| # mostly "do a thing and then gitlab upload the artifact" .PHONY: ci-upload ci-upload: ci-upload-node ci-upload-android .PHONY: ci-upload-node-linux-amd64 ci-upload-node-linux-amd64: $(MAKE) ci-upload-file upload_file=streamplace-$(VERSION)-linux-amd64.tar.gz \ && $(MAKE) ci-upload-file upload_file=streamplace-desktop-$(VERSION)-linux-amd64.AppImage \ && $(MAKE) ci-upload-file upload_file=streamplace-$(VERSION)-linux-amd64.deb .PHONY: ci-upload-node-linux-arm64 ci-upload-node-linux-arm64: $(MAKE) ci-upload-file upload_file=streamplace-$(VERSION)-linux-arm64.tar.gz \ && $(MAKE) ci-upload-file upload_file=streamplace-desktop-$(VERSION)-linux-arm64.AppImage \ && $(MAKE) ci-upload-file upload_file=streamplace-$(VERSION)-linux-arm64.deb .PHONY: ci-upload-node-darwin-arm64 ci-upload-node-darwin-arm64: export file=streamplace-$(VERSION)-darwin-arm64.tar.gz \ && $(MAKE) ci-upload-file upload_file=$$file; .PHONY: ci-upload-node-darwin-amd64 ci-upload-node-darwin-amd64: export file=streamplace-$(VERSION)-darwin-amd64.tar.gz \ && $(MAKE) ci-upload-file upload_file=$$file; .PHONY: ci-upload-node-windows-amd64 ci-upload-node-windows-amd64: export file=streamplace-$(VERSION)-windows-amd64.zip \ && $(MAKE) ci-upload-file upload_file=$$file; \ export file=streamplace-desktop-$(VERSION)-windows-amd64.exe \ && $(MAKE) ci-upload-file upload_file=$$file; \ export SUM=$$(cat bin/streamplace-desktop-$(VERSION)-windows-amd64.nupkg.sha1) \ && export file=streamplace-desktop-$(VERSION)-windows-amd64.$$SUM.nupkg \ && $(MAKE) ci-upload-file upload_file=$$file; .PHONY: ci-upload-node-macos ci-upload-node-macos: node-all-platforms-macos for GOOS in darwin; do \ for GOARCH in amd64 arm64; do \ export file=streamplace-$(VERSION)-$$GOOS-$$GOARCH.tar.gz \ && $(MAKE) ci-upload-file upload_file=$$file; \ export file=streamplace-desktop-$(VERSION)-$$GOOS-$$GOARCH.dmg \ && $(MAKE) ci-upload-file upload_file=$$file; \ export file=streamplace-desktop-$(VERSION)-$$GOOS-$$GOARCH.zip \ && $(MAKE) ci-upload-file upload_file=$$file; \ done \ done; .PHONY: ci-upload-android ci-upload-android: android $(MAKE) ci-upload-android-debug \ && $(MAKE) ci-upload-android-release .PHONY: ci-upload-android-debug ci-upload-android-debug: $(MAKE) ci-upload-file upload_file=streamplace-$(VERSION)-android-debug.apk \ && $(MAKE) ci-upload-file upload_file=streamplace-$(VERSION)-android-debug.aab .PHONY: ci-upload-android-release ci-upload-android-release: $(MAKE) ci-upload-file upload_file=streamplace-$(VERSION)-android-release.apk \ && $(MAKE) ci-upload-file upload_file=streamplace-$(VERSION)-android-release.aab .PHONY: ci-upload-ios ci-upload-ios: ios $(MAKE) ci-upload-file upload_file=streamplace-$(VERSION)-ios-release.xcarchive.tar.gz upload_file?="" .PHONY: ci-upload-file ci-upload-file: curl \ --fail-with-body \ --header "JOB-TOKEN: $$CI_JOB_TOKEN" \ --upload-file bin/$(upload_file) \ "$$CI_API_V4_URL/projects/$$CI_PROJECT_ID/packages/generic/$(BRANCH)/$(VERSION)/$(upload_file)"; download_file?="" .PHONY: ci-download-file ci-download-file: curl \ --fail-with-body \ --header "JOB-TOKEN: $$CI_JOB_TOKEN" \ -o bin/$(download_file) \ "$$CI_API_V4_URL/projects/$$CI_PROJECT_ID/packages/generic/$(BRANCH)/$(VERSION)/$(download_file)"; .PHONY: c2pa-types c2pa-types: $(MAKE) dev-rust ./target/debug/export_c2pa_schema npx quicktype --lang go --src-lang schema --package c2patypes --out pkg/c2patypes/c2patypes.go ./target/schema/C2PA.schema.json gofmt -w pkg/c2patypes/c2patypes.go