Compare commits
	
		
			126 Commits
		
	
	
		
			v0.0.27
			...
			tasiaiso-n
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| d67e47ae4b | |||
| b43b8da9ab | |||
| 8534e16469 | |||
| f6cc6f2eae | |||
| 0904425221 | |||
| a729886522 | |||
| e5dfedc7d1 | |||
| f02423d084 | |||
| 8f4b6e83eb | |||
| 3029919553 | |||
| ac67db0591 | |||
| 1e08838f5b | |||
| d814f7ee77 | |||
| 7edfb9d386 | |||
| 6928d6caba | |||
| 1a626875cf | |||
| 6eb3b64334 | |||
| 09f3595e93 | |||
| 11e89622d4 | |||
| 0fa8acc264 | |||
| afc7c64ed8 | |||
| c794c1b885 | |||
| 6247529799 | |||
| ad3eedc1fb | |||
| 1cfac3cae6 | |||
| 478bcd5d13 | |||
| 6932da69b3 | |||
| 857f47bf55 | |||
| 373d742751 | |||
| 575622c522 | |||
| a3e86ccb1d | |||
| e491798ff1 | |||
| 15df4ac236 | |||
| 017a74c4e4 | |||
| 6e5b1127f3 | |||
| 95f4f88949 | |||
| 92858882d7 | |||
| 4cba95ec8c | |||
| bbb2f89d85 | |||
| 88213038b2 | |||
| cc18b41e76 | |||
| f16127a238 | |||
| 2a6ecfaede | |||
| cf4a09bf03 | |||
| 20e60262fd | |||
| 9e3928c976 | |||
| 95d8768545 | |||
| 8679d09040 | |||
| 4cc5c6acb3 | |||
| 6f9b548b1a | |||
| fbff3386a9 | |||
| e5899fca58 | |||
| dddec489b9 | |||
| b4049eaeaa | |||
| b3fd724b5a | |||
| aca25be86a | |||
| 0f8cbdac57 | |||
| 630219d667 | |||
| fef268e434 | |||
| e18dcf3a48 | |||
| e019320146 | |||
| 65b31e14f9 | |||
| 9cb5cbcde9 | |||
| 25914ff5a7 | |||
| c4af799279 | |||
| 3f8f0e14f4 | |||
| 5414b30e7f | |||
| 7aee897c1b | |||
| 4060f9cc11 | |||
| 5b526cbf5b | |||
| f5065ff42b | |||
| 86b5546f5f | |||
| 43ae2a293b | |||
| b8ddbd4255 | |||
| 87cdba1db8 | |||
| df83187e33 | |||
| eccdbf29ab | |||
| 28d181f8bc | |||
| d386daf2ff | |||
| 0c1e116c1e | |||
| bb0ed67827 | |||
| b111d06851 | |||
| 79388845ea | |||
| 99faef2e77 | |||
| 21fffbfe10 | |||
| 64fb9f0c2a | |||
| a42e0bef2c | |||
| 45a09006e1 | |||
| 240484be4c | |||
| 22f4d115e3 | |||
| 32920e0e5d | |||
| f03a5918d1 | |||
| dd1870b52a | |||
| f0c1a8f98f | |||
| 0c181d7e7c | |||
| e3dc0e833a | |||
| 6de875edea | |||
| 986a55173f | |||
| 59c8cabf02 | |||
| a33b9fab07 | |||
| f8a725e1e7 | |||
| b3ab3af01b | |||
| 79f9463e56 | |||
| 4257b2ed51 | |||
| 6488ab60ec | |||
| 18bd3dfcf9 | |||
| ec114e160d | |||
| de033af18f | |||
| e971c6fcb7 | |||
| d3c465391c | |||
| f1fa19593d | |||
| 60b6f9c73e | |||
| 55b95ddecb | |||
| 0800a251b2 | |||
| 82cf7a80eb | |||
| 379f3d12eb | |||
| e52972d4d4 | |||
| 1a0ca4dec2 | |||
| 0bac9d8d5a | |||
| a9608363c5 | |||
| f1a2c5ae8e | |||
| 192a81ede7 | |||
| 916aa5abbd | |||
| d4a5cc6eee | |||
| d19605cc8d | |||
| 8d529327a4 | 
| @@ -1,4 +1,3 @@ | |||||||
| .svn | .git | ||||||
| db.sqlite | db.sqlite* | ||||||
| out/**/*.o | out/ | ||||||
| out/**/*.d |  | ||||||
|   | |||||||
| @@ -6,15 +6,51 @@ jobs: | |||||||
|   Build-All: |   Build-All: | ||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|     container: |     container: | ||||||
|       valid_volumes: ['/opt/keys'] |       image: node:23-bookworm-slim | ||||||
|  |       valid_volumes: | ||||||
|  |         - '/opt/keys' | ||||||
|  |         - '/opt/deps' | ||||||
|       volumes: |       volumes: | ||||||
|         - /opt/keys:/opt/keys |         - /opt/keys:/opt/keys | ||||||
|  |         - /opt/deps:/opt/deps | ||||||
|     steps: |     steps: | ||||||
|       - name: check out code |       - name: Install build dependencies | ||||||
|  |         run: > | ||||||
|  |           apt update && apt install -y \ | ||||||
|  |             build-essential \ | ||||||
|  |             clang-19 \ | ||||||
|  |             cmake \ | ||||||
|  |             curl \ | ||||||
|  |             docker.io \ | ||||||
|  |             doxygen \ | ||||||
|  |             file \ | ||||||
|  |             gcc-aarch64-linux-gnu \ | ||||||
|  |             git \ | ||||||
|  |             graphviz \ | ||||||
|  |             libgpgme11 \ | ||||||
|  |             libssl-dev \ | ||||||
|  |             mingw-w64 \ | ||||||
|  |             rsync \ | ||||||
|  |             unzip \ | ||||||
|  |             zip \ | ||||||
|  |             zlib1g-dev | ||||||
|  |       - name: Get code | ||||||
|         uses: actions/checkout@v4 |         uses: actions/checkout@v4 | ||||||
|         with: |         with: | ||||||
|           submodules: true |           submodules: true | ||||||
|       - run: ln -s /opt/keys .keys |       - name: Setup environment | ||||||
|  |         run: | | ||||||
|  |           update-alternatives --install /usr/bin/clang clang /usr/bin/clang-19 100 | ||||||
|  |           update-alternatives --install /usr/bin/clang++ clang++ /usr/bin/clang++-19 100 | ||||||
|  |           ln -s /opt/keys .keys | ||||||
|  |           ln -sf /opt/deps/ios_toolchain deps/ | ||||||
|  |           ln -sf /opt/deps/macos_toolchain deps/ | ||||||
|  |       - name: Build documentation | ||||||
|  |         run: | | ||||||
|  |           mkdir -p out/html/ ~/.ssh/ | ||||||
|  |           make docs | ||||||
|  |           echo 'pildefriends ssh-ed25519 AAAAC3NzaC1lZDI1NTE5AAAAIKD3Kde5vDO0TrMBDK0IGGeNGe/XinWAZkSQ/rXxwUjt' >> ~/.ssh/known_hosts | ||||||
|  |           rsync -avP --delete -e "ssh -i /opt/keys/ssh.ed25519" out/html/ tfdocs@pildefriends:docs/html/ | ||||||
|       - name: Setup JDK |       - name: Setup JDK | ||||||
|         uses: actions/setup-java@v3 |         uses: actions/setup-java@v3 | ||||||
|         with: |         with: | ||||||
| @@ -24,14 +60,12 @@ jobs: | |||||||
|         uses: android-actions/setup-android@v3 |         uses: android-actions/setup-android@v3 | ||||||
|         with: |         with: | ||||||
|           packages: 'tools platform-tools build-tools;34.0.0 platforms;android-34 ndk;26.3.11579264' |           packages: 'tools platform-tools build-tools;34.0.0 platforms;android-34 ndk;26.3.11579264' | ||||||
|       - run: sudo apt update && sudo apt install -y doxygen graphviz mingw-w64 libgpgme11 gcc-aarch64-linux-gnu |       - name: Docker build | ||||||
|       - run: ANDROID_SDK=$HOME/.android/sdk make -j`nproc` all docs |         run: DOCKER_BUILDKIT=1 docker build . | ||||||
|       - run: docker build . |       - name: Build | ||||||
|       - uses: actions/upload-artifact@v3 |         run: ANDROID_SDK=$HOME/.android/sdk make -j`nproc` all dist docs | ||||||
|  |       - name: Upload artifacts | ||||||
|  |         uses: actions/upload-artifact@v3 | ||||||
|         with: |         with: | ||||||
|           path: | |           name: dist | ||||||
|             out/TildeFriends-release.fdroid.apk |           path: dist/* | ||||||
|             out/winrelease/tildefriends.standalone.exe |  | ||||||
|             out/tildefriends-x86_64.AppImage |  | ||||||
|             out/release/tildefriends.standalone |  | ||||||
|             out/armrelease/tildefriends.standalone |  | ||||||
|   | |||||||
							
								
								
									
										4
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -1,11 +1,13 @@ | |||||||
| build/ | build/ | ||||||
| *.core | *.core | ||||||
| db.* | db.* | ||||||
| deps/ios_toolchain/ | deps/ios_toolchain | ||||||
|  | deps/macos_toolchain | ||||||
| deps/openssl/ | deps/openssl/ | ||||||
| dist/ | dist/ | ||||||
| .flatpak-builder | .flatpak-builder | ||||||
| .keys | .keys | ||||||
|  | **/.DS_Store | ||||||
| logs/ | logs/ | ||||||
| **/node_modules | **/node_modules | ||||||
| out | out | ||||||
|   | |||||||
							
								
								
									
										6
									
								
								.gitmodules
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -26,6 +26,6 @@ | |||||||
| [submodule "deps/c-ares"] | [submodule "deps/c-ares"] | ||||||
| 	path = deps/c-ares | 	path = deps/c-ares | ||||||
| 	url = https://github.com/c-ares/c-ares.git | 	url = https://github.com/c-ares/c-ares.git | ||||||
| [submodule "docs"] | [submodule "deps/zsign"] | ||||||
| 	path = docs | 	path = deps/zsign | ||||||
| 	url = https://dev.tildefriends.net/cory/tildefriends.wiki.git | 	url = https://github.com/zhlynn/zsign.git | ||||||
|   | |||||||
| @@ -1,19 +1,16 @@ | |||||||
| FROM bitnami/minideb:bullseye AS build | FROM bitnami/minideb:bookworm AS build | ||||||
|  |  | ||||||
| RUN apt-get update && \ | RUN apt-get update && \ | ||||||
| 	apt-get install -y --no-install-recommends \ | 	apt-get install -y --no-install-recommends \ | ||||||
| 		gcc \ | 		gcc \ | ||||||
| 		libc6-dev \ | 		libc6-dev \ | ||||||
| 		libssl-dev \ | 		perl \ | ||||||
| 		make | 		make | ||||||
|  |  | ||||||
| COPY . /app | COPY . /app | ||||||
| RUN make -C /app -j $(nproc) release | RUN make -C /app -j $(nproc) release | ||||||
|  |  | ||||||
| FROM bitnami/minideb:bullseye | FROM bitnami/minideb:bookworm | ||||||
| RUN apt-get update && \ |  | ||||||
| 	apt-get install -y --no-install-recommends \ |  | ||||||
| 		libssl1.1 |  | ||||||
|  |  | ||||||
| COPY --from=build /app/out/release/tildefriends /app/out/release/tildefriends | COPY --from=build /app/out/release/tildefriends /app/out/release/tildefriends | ||||||
| COPY --from=build /app/apps /app/apps | COPY --from=build /app/apps /app/apps | ||||||
|   | |||||||
							
								
								
									
										6
									
								
								Doxyfile
									
									
									
									
									
								
							
							
						
						| @@ -943,7 +943,7 @@ WARN_LOGFILE           = | |||||||
| # spaces. See also FILE_PATTERNS and EXTENSION_MAPPING | # spaces. See also FILE_PATTERNS and EXTENSION_MAPPING | ||||||
| # Note: If this tag is empty the current directory is searched. | # Note: If this tag is empty the current directory is searched. | ||||||
|  |  | ||||||
| INPUT                  = src/ | INPUT                  = README.md docs/ src/ | ||||||
|  |  | ||||||
| # This tag can be used to specify the character encoding of the source files | # This tag can be used to specify the character encoding of the source files | ||||||
| # that doxygen parses. Internally doxygen uses the UTF-8 encoding. Doxygen uses | # that doxygen parses. Internally doxygen uses the UTF-8 encoding. Doxygen uses | ||||||
| @@ -1110,7 +1110,7 @@ FILTER_SOURCE_PATTERNS = | |||||||
| # (index.html). This can be useful if you have a project on for instance GitHub | # (index.html). This can be useful if you have a project on for instance GitHub | ||||||
| # and want to reuse the introduction page also for the doxygen output. | # and want to reuse the introduction page also for the doxygen output. | ||||||
|  |  | ||||||
| USE_MDFILE_AS_MAINPAGE = | USE_MDFILE_AS_MAINPAGE = README.md | ||||||
|  |  | ||||||
| # The Fortran standard specifies that for fixed formatted Fortran code all | # The Fortran standard specifies that for fixed formatted Fortran code all | ||||||
| # characters from position 72 are to be considered as comment. A common | # characters from position 72 are to be considered as comment. A common | ||||||
| @@ -1680,7 +1680,7 @@ DISABLE_INDEX          = NO | |||||||
| # The default value is: NO. | # The default value is: NO. | ||||||
| # This tag requires that the tag GENERATE_HTML is set to YES. | # This tag requires that the tag GENERATE_HTML is set to YES. | ||||||
|  |  | ||||||
| GENERATE_TREEVIEW      = NO | GENERATE_TREEVIEW      = YES | ||||||
|  |  | ||||||
| # When both GENERATE_TREEVIEW and DISABLE_INDEX are set to YES, then the | # When both GENERATE_TREEVIEW and DISABLE_INDEX are set to YES, then the | ||||||
| # FULL_SIDEBAR option determines if the side bar is limited to only the treeview | # FULL_SIDEBAR option determines if the side bar is limited to only the treeview | ||||||
|   | |||||||
							
								
								
									
										337
									
								
								GNUmakefile
									
									
									
									
									
								
							
							
						
						| @@ -16,11 +16,14 @@ MAKEFLAGS += --no-builtin-rules | |||||||
| ## LD := Linker. | ## LD := Linker. | ||||||
| ## ANDROID_SDK := Path to the Android SDK. | ## ANDROID_SDK := Path to the Android SDK. | ||||||
|  |  | ||||||
| VERSION_CODE := 32 | VERSION_CODE := 33 | ||||||
| VERSION_NUMBER := 0.0.27 | VERSION_CODE_IOS := 8 | ||||||
|  | VERSION_NUMBER := 0.0.28-wip | ||||||
| VERSION_NAME := This program kills fascists. | VERSION_NAME := This program kills fascists. | ||||||
|  |  | ||||||
| SQLITE_URL := https://www.sqlite.org/2025/sqlite-amalgamation-3480000.zip | IPHONEOS_VERSION_MIN=14.0 | ||||||
|  |  | ||||||
|  | SQLITE_URL := https://www.sqlite.org/2025/sqlite-amalgamation-3490100.zip | ||||||
| BUNDLETOOL_URL := https://github.com/google/bundletool/releases/download/1.17.0/bundletool-all-1.17.0.jar | BUNDLETOOL_URL := https://github.com/google/bundletool/releases/download/1.17.0/bundletool-all-1.17.0.jar | ||||||
| APPIMAGETOOL_URL := https://github.com/AppImage/AppImageKit/releases/download/continuous/appimagetool-x86_64.AppImage | APPIMAGETOOL_URL := https://github.com/AppImage/AppImageKit/releases/download/continuous/appimagetool-x86_64.AppImage | ||||||
| APPIMAGETOOL_MD5 := e989fadfc4d685fd3d6aeeb9b525d74d  out/appimagetool | APPIMAGETOOL_MD5 := e989fadfc4d685fd3d6aeeb9b525d74d  out/appimagetool | ||||||
| @@ -33,21 +36,23 @@ UNAME_M := $(shell uname -m) | |||||||
| ANDROID_SDK ?= ~/Android/Sdk | ANDROID_SDK ?= ~/Android/Sdk | ||||||
| BUNDLETOOL = out/bundletool.jar | BUNDLETOOL = out/bundletool.jar | ||||||
|  |  | ||||||
| HAVE_WIN := 0 | HAVE_WIN := | ||||||
| HAVE_CROSS_AARCH64 := 0 | HAVE_CROSS_AARCH64 := | ||||||
|  | USE_SYSTEM_SSL := | ||||||
|  |  | ||||||
| export SOURCE_DATE_EPOCH=1 | export SOURCE_DATE_EPOCH=1 | ||||||
| export TZ=UTC | export TZ=UTC | ||||||
|  |  | ||||||
| ifeq ($(UNAME_S),Darwin) | ifeq ($(UNAME_S),Darwin) | ||||||
| BUILD_TYPES := macosdebug macosrelease iosdebug iosrelease iossimdebug iossimrelease | BUILD_TYPES := debug release iosdebug iosrelease iossimdebug iossimrelease | ||||||
| else ifeq ($(UNAME_S),Linux) | else ifeq ($(UNAME_S),Linux) | ||||||
| BUILD_TYPES := debug release | BUILD_TYPES := debug release | ||||||
| HAVE_ANDROID = $(if $(shell which $(ANDROID_SDK)/platform-tools/adb),1,0) | HAVE_ANDROID = $(if $(shell which $(ANDROID_SDK)/platform-tools/adb),1) | ||||||
| HAVE_LINUX_IOS = $(if $(shell which deps/ios_toolchain/target/bin deps/ios_toolchain/target/bin/arm-apple-darwin11-clang),1,0) | HAVE_LINUX_IOS = $(if $(shell which deps/ios_toolchain/target/bin deps/ios_toolchain/target/bin/arm-apple-darwin11-clang),1) | ||||||
| HAVE_WIN = $(if $(shell which x86_64-w64-mingw32-gcc-win32),1,0) | HAVE_LINUX_MACOS = $(if $(shell which deps/macos_toolchain/bin/oa64-clang),1) | ||||||
|  | HAVE_WIN = $(if $(shell which x86_64-w64-mingw32-gcc-win32),1) | ||||||
| ifneq ($(UNAME_M),aarch64) | ifneq ($(UNAME_M),aarch64) | ||||||
| HAVE_CROSS_AARCH64 = $(if $(shell which aarch64-linux-gnu-gcc),1,0) | HAVE_CROSS_AARCH64 = $(if $(shell which aarch64-linux-gnu-gcc),1) | ||||||
| endif | endif | ||||||
| else ifeq ($(UNAME_S),Haiku) | else ifeq ($(UNAME_S),Haiku) | ||||||
| BUILD_TYPES := debug release | BUILD_TYPES := debug release | ||||||
| @@ -56,6 +61,7 @@ LDFLAGS += \ | |||||||
| 	-lbsd \ | 	-lbsd \ | ||||||
| 	-lnetwork \ | 	-lnetwork \ | ||||||
| 	-Wno-stringop-overflow | 	-Wno-stringop-overflow | ||||||
|  | USE_SYSTEM_SSL := 1 | ||||||
| else ifeq ($(UNAME_S),OpenBSD) | else ifeq ($(UNAME_S),OpenBSD) | ||||||
| BUILD_TYPES := debug release | BUILD_TYPES := debug release | ||||||
| CFLAGS += \ | CFLAGS += \ | ||||||
| @@ -63,18 +69,24 @@ CFLAGS += \ | |||||||
| LDFLAGS += \ | LDFLAGS += \ | ||||||
| 	-lexecinfo \ | 	-lexecinfo \ | ||||||
| 	-lc++abi | 	-lc++abi | ||||||
| HAVE_ANDROID := 0 | HAVE_ANDROID := | ||||||
| HAVE_LINUX_IOS := 0 | HAVE_LINUX_IOS := | ||||||
|  | HAVE_LINUX_MACOS := | ||||||
|  | USE_SYSTEM_SSL := 1 | ||||||
| else | else | ||||||
| $(error Unexpected host platform $(UNAME_S).) | $(error Unexpected host platform $(UNAME_S).) | ||||||
| endif | endif | ||||||
|  |  | ||||||
|  | # Everything is set above. | ||||||
|  | $(info Building Tilde Friends $(VERSION_NUMBER) android=$(if $(HAVE_ANDROID),1,0) win=$(if $(HAVE_WIN),1,0) cross_aarch64=$(if $(HAVE_CROSS_AARCH64),1,0) cross_ios=$(if $(HAVE_LINUX_IOS),1,0) cross_macos=$(if $(HAVE_LINUX_MACOS),1,0) system_ssl=$(if $(USE_SYSTEM_SSL),1,0)) | ||||||
|  |  | ||||||
| CFLAGS += \ | CFLAGS += \ | ||||||
| 	-std=gnu11 \ | 	-std=gnu11 \ | ||||||
| 	-Wall \ | 	-Wall \ | ||||||
| 	-Wextra \ | 	-Wextra \ | ||||||
| 	-Wno-unused-parameter \ | 	-Wno-cast-function-type-mismatch \ | ||||||
| 	-Wno-unknown-warning-option \ | 	-Wno-unknown-warning-option \ | ||||||
|  | 	-Wno-unused-parameter \ | ||||||
| 	-MMD \ | 	-MMD \ | ||||||
| 	-MP \ | 	-MP \ | ||||||
| 	-ffunction-sections \ | 	-ffunction-sections \ | ||||||
| @@ -136,12 +148,28 @@ ifeq ($(HAVE_CROSS_AARCH64),1) | |||||||
| BUILD_TYPES += armdebug armrelease | BUILD_TYPES += armdebug armrelease | ||||||
| endif | endif | ||||||
|  |  | ||||||
| LINUX_TARGETS := \ | HOST_TARGETS := \ | ||||||
| 	out/debug/tildefriends \ | 	out/debug/tildefriends \ | ||||||
| 	out/release/tildefriends | 	out/release/tildefriends | ||||||
|  | ifeq ($(UNAME_S),Darwin) | ||||||
| MACOS_TARGETS := \ | MACOS_TARGETS := \ | ||||||
| 	out/macosdebug/tildefriends \ | 	out/debug/tildefriends \ | ||||||
| 	out/macosrelease/tildefriends | 	out/release/tildefriends | ||||||
|  | else ifeq ($(UNAME_S),Linux) | ||||||
|  | ifeq ($(HAVE_LINUX_MACOS),1) | ||||||
|  | MACOS_TARGETS := \ | ||||||
|  | 	out/macosdebug-arm/tildefriends \ | ||||||
|  | 	out/macosrelease-arm/tildefriends \ | ||||||
|  | 	out/macosdebug-x86_64/tildefriends \ | ||||||
|  | 	out/macosrelease-x86_64/tildefriends | ||||||
|  | all: out/macosdebug/tildefriends.standalone | ||||||
|  | all: out/macosrelease/tildefriends.standalone | ||||||
|  | else | ||||||
|  | MACOS_TARGETS := | ||||||
|  | endif | ||||||
|  | else | ||||||
|  | MACOS_TARGETS := | ||||||
|  | endif | ||||||
| IOS_TARGETS := \ | IOS_TARGETS := \ | ||||||
| 	out/iosdebug/tildefriends \ | 	out/iosdebug/tildefriends \ | ||||||
| 	out/iosrelease/tildefriends | 	out/iosrelease/tildefriends | ||||||
| @@ -155,6 +183,14 @@ ifeq ($(HAVE_LINUX_IOS),1) | |||||||
| BUILD_TYPES += iosdebug iosrelease | BUILD_TYPES += iosdebug iosrelease | ||||||
| all: $(IOS_APPS) | all: $(IOS_APPS) | ||||||
| endif | endif | ||||||
|  | ifeq ($(HAVE_LINUX_MACOS),1) | ||||||
|  | BUILD_TYPES += \ | ||||||
|  | 	macosdebug-arm \ | ||||||
|  | 	macosrelease-arm \ | ||||||
|  | 	macosdebug-x86_64 \ | ||||||
|  | 	macosrelease-x86_64 | ||||||
|  | all: $(IOS_APPS) | ||||||
|  | endif | ||||||
| ifeq ($(UNAME_S),Darwin) | ifeq ($(UNAME_S),Darwin) | ||||||
| all: $(IOS_APPS) \ | all: $(IOS_APPS) \ | ||||||
| 	out/tildefriends-iossimdebug.app/tildefriends \ | 	out/tildefriends-iossimdebug.app/tildefriends \ | ||||||
| @@ -169,35 +205,38 @@ DEBUG_TARGETS := \ | |||||||
| 	out/windebug/tildefriends.exe \ | 	out/windebug/tildefriends.exe \ | ||||||
| 	out/iosdebug/tildefriends \ | 	out/iosdebug/tildefriends \ | ||||||
| 	out/iossimdebug/tildefriends \ | 	out/iossimdebug/tildefriends \ | ||||||
| 	out/macosdebug/tildefriends \ |  | ||||||
| 	out/androiddebug/tildefriends \ | 	out/androiddebug/tildefriends \ | ||||||
| 	out/androiddebug-armv7a/tildefriends \ | 	out/androiddebug-armv7a/tildefriends \ | ||||||
| 	out/androiddebug-x86_64/tildefriends \ | 	out/androiddebug-x86_64/tildefriends \ | ||||||
| 	out/androiddebug-x86/tildefriends \ | 	out/androiddebug-x86/tildefriends \ | ||||||
| 	out/armdebug/tildefriends | 	out/armdebug/tildefriends \ | ||||||
|  | 	out/macosdebug-arm/tildefriends \ | ||||||
|  | 	out/macosdebug-x86_64/tildefriends | ||||||
| RELEASE_TARGETS := \ | RELEASE_TARGETS := \ | ||||||
| 	out/release/tildefriends \ | 	out/release/tildefriends \ | ||||||
| 	out/winrelease/tildefriends.exe \ | 	out/winrelease/tildefriends.exe \ | ||||||
| 	out/iosrelease/tildefriends \ | 	out/iosrelease/tildefriends \ | ||||||
| 	out/iossimrelease/tildefriends \ | 	out/iossimrelease/tildefriends \ | ||||||
| 	out/macosrelease/tildefriends \ |  | ||||||
| 	out/androidrelease/tildefriends \ | 	out/androidrelease/tildefriends \ | ||||||
| 	out/androidrelease-armv7a/tildefriends \ | 	out/androidrelease-armv7a/tildefriends \ | ||||||
| 	out/androidrelease-x86_64/tildefriends \ | 	out/androidrelease-x86_64/tildefriends \ | ||||||
| 	out/androidrelease-x86/tildefriends \ | 	out/androidrelease-x86/tildefriends \ | ||||||
| 	out/armrelease/tildefriends | 	out/armrelease/tildefriends \ | ||||||
|  | 	out/macosrelease-arm/tildefriends \ | ||||||
|  | 	out/macosrelease-x86_64/tildefriends | ||||||
| ALL_TARGETS = $(DEBUG_TARGETS) $(RELEASE_TARGETS) | ALL_TARGETS = $(DEBUG_TARGETS) $(RELEASE_TARGETS) | ||||||
| ANDROID_RELEASE_TARGETS := $(filter-out $(DEBUG_TARGETS),$(ANDROID_TARGETS)) | ANDROID_RELEASE_TARGETS := $(filter-out $(DEBUG_TARGETS),$(ANDROID_TARGETS)) | ||||||
| NONANDROID_RELEASE_TARGETS := $(filter-out $(ANDROID_ARM64_TARGETS),$(RELEASE_TARGETS)) | NONANDROID_RELEASE_TARGETS := $(filter-out $(ANDROID_ARM64_TARGETS),$(RELEASE_TARGETS)) | ||||||
| NONANDROID_TARGETS := $(filter-out $(ANDROID_TARGETS),$(ALL_TARGETS)) | NONANDROID_TARGETS := $(filter-out $(ANDROID_TARGETS),$(ALL_TARGETS)) | ||||||
| NONMACOS_TARGETS := $(filter-out $(MACOS_TARGETS) $(IOS_TARGETS) $(IOSSIM_TARGETS),$(ALL_TARGETS)) | NONMACOS_TARGETS := $(filter-out $(MACOS_TARGETS) $(IOS_TARGETS) $(IOSSIM_TARGETS),$(ALL_TARGETS)) | ||||||
| DEADSTRIP_TARGETS := $(filter-out $(ANDROID_TARGETS),$(NONMACOS_TARGETS)) |  | ||||||
| ifneq ($(UNAME_S),OpenBSD) | ifneq ($(UNAME_S),OpenBSD) | ||||||
| $(NONMACOS_TARGETS): LDFLAGS += -static-libgcc | $(NONMACOS_TARGETS): LDFLAGS += -static-libgcc | ||||||
| endif | endif | ||||||
|  |  | ||||||
| $(NONANDROID_TARGETS): CFLAGS += -fno-omit-frame-pointer | $(NONANDROID_TARGETS): CFLAGS += -fno-omit-frame-pointer | ||||||
| $(filter-out $(WINDOWS_TARGETS),$(ALL_TARGETS)): LDFLAGS += -rdynamic | $(filter-out $(WINDOWS_TARGETS),$(ALL_TARGETS)): LDFLAGS += \ | ||||||
|  | 	-rdynamic \ | ||||||
|  | 	-gz=zlib | ||||||
| $(ANDROID_TARGETS): CFLAGS += \ | $(ANDROID_TARGETS): CFLAGS += \ | ||||||
| 	--sysroot $(ANDROID_NDK)/toolchains/llvm/prebuilt/linux-x86_64/sysroot \ | 	--sysroot $(ANDROID_NDK)/toolchains/llvm/prebuilt/linux-x86_64/sysroot \ | ||||||
| 	-fPIC \ | 	-fPIC \ | ||||||
| @@ -221,25 +260,29 @@ $(WINDOWS_TARGETS): CFLAGS += \ | |||||||
| 	-D_WIN32_WINNT=0x0A00 \ | 	-D_WIN32_WINNT=0x0A00 \ | ||||||
| 	-DWINVER=0x0A00 \ | 	-DWINVER=0x0A00 \ | ||||||
| 	-DNTDDI_VERSION=NTDDI_WIN10 \ | 	-DNTDDI_VERSION=NTDDI_WIN10 \ | ||||||
| 	-Ideps/openssl/mingw64/usr/local/include | 	-Iout/openssl/$(UNAME_S)/mingw64/usr/local/include | ||||||
| $(WINDOWS_TARGETS): LDFLAGS += \ | $(WINDOWS_TARGETS): LDFLAGS += \ | ||||||
| 	-static \ | 	-static \ | ||||||
| 	-lm \ | 	-lm \ | ||||||
| 	-Ldeps/openssl/mingw64/usr/local/lib | 	-Lout/openssl/$(UNAME_S)/mingw64/usr/local/lib | ||||||
| $(AARCH64_TARGETS): CC = aarch64-linux-gnu-gcc | $(AARCH64_TARGETS): CC = aarch64-linux-gnu-gcc | ||||||
| $(AARCH64_TARGETS): AS = $(CC) | $(AARCH64_TARGETS): AS = $(CC) | ||||||
| $(AARCH64_TARGETS): CFLAGS += -Ideps/openssl/Linux/aarch64/usr/local/include | $(AARCH64_TARGETS): CFLAGS += -Iout/openssl/Linux/aarch64/usr/local/include | ||||||
| $(AARCH64_TARGETS): LDFLAGS += -Ldeps/openssl/Linux/aarch64/usr/local/lib | $(AARCH64_TARGETS): LDFLAGS += -Lout/openssl/Linux/aarch64/usr/local/lib | ||||||
| ifeq ($(UNAME_S),Darwin) | ifeq ($(UNAME_S),Darwin) | ||||||
| $(MACOS_TARGETS): CC = xcrun clang | $(HOST_TARGETS): CC = xcrun clang | ||||||
| $(IOS_TARGETS): IOS_SYSROOT := $(shell xcrun --sdk iphoneos --show-sdk-path) | $(IOS_TARGETS): IOS_SYSROOT := $(shell xcrun --sdk iphoneos --show-sdk-path) | ||||||
| $(IOS_TARGETS): CC = xcrun --sdk iphoneos clang -isysroot $(IOS_SYSROOT) -arch arm64 | $(IOS_TARGETS): CC = xcrun --sdk iphoneos clang -isysroot $(IOS_SYSROOT) -arch arm64 | ||||||
| $(IOSSIM_TARGETS): IOSSIM_SYSROOT := $(shell xcrun --sdk iphonesimulator --show-sdk-path) | $(IOSSIM_TARGETS): IOSSIM_SYSROOT := $(shell xcrun --sdk iphonesimulator --show-sdk-path) | ||||||
| $(IOSSIM_TARGETS): CC = xcrun --sdk iphonesimulator clang -isysroot $(IOSSIM_SYSROOT) -arch x86_64 | $(IOSSIM_TARGETS): CC = xcrun --sdk iphonesimulator clang -isysroot $(IOSSIM_SYSROOT) -arch x86_64 | ||||||
| else ifeq ($(UNAME_S),Linux) | else ifeq ($(UNAME_S),Linux) | ||||||
| $(IOS_TARGETS): CFLAGS += -isysroot deps/ios_toolchain/target/SDKs/iPhoneOS18.2.sdk -arch arm64 | $(IOS_TARGETS): CFLAGS += -isysroot deps/ios_toolchain/target/SDKs/iPhoneOS18.2.sdk -arch arm64 -DTARGET_OS_IPHONE=1 | ||||||
| $(IOS_TARGETS): LDFLAGS += -isysroot deps/ios_toolchain/target/SDKs/iPhoneOS18.2.sdk | $(IOS_TARGETS): LDFLAGS += -isysroot deps/ios_toolchain/target/SDKs/iPhoneOS18.2.sdk | ||||||
| $(IOS_TARGETS): CC = PATH=$$PATH:deps/ios_toolchain/target/bin deps/ios_toolchain/target/bin/arm-apple-darwin11-clang | $(IOS_TARGETS): CC = PATH=$$PATH:deps/ios_toolchain/target/bin LD_LIBRARY_PATH=$$LD_LIBRARY_PATH:deps/ios_toolchain/target/lib deps/ios_toolchain/target/bin/arm-apple-darwin11-clang | ||||||
|  | $(filter $(BUILD_DIR)/macosdebug-x86_64/%,$(ALL_TARGETS)): CC = PATH=deps/macos_toolchain/bin:$$PATH LD_LIBRARY_PATH=$$LD_LIBRARY_PATH:deps/macos_toolchain/lib deps/macos_toolchain/bin/o64-clang | ||||||
|  | $(filter $(BUILD_DIR)/macosdebug-arm/%,$(ALL_TARGETS)): CC = PATH=deps/macos_toolchain/bin:$$PATH LD_LIBRARY_PATH=$$LD_LIBRARY_PATH:deps/macos_toolchain/lib deps/macos_toolchain/bin/oa64-clang | ||||||
|  | $(filter $(BUILD_DIR)/macosrelease-x86_64/%,$(ALL_TARGETS)): CC = PATH=deps/macos_toolchain/bin:$$PATH LD_LIBRARY_PATH=$$LD_LIBRARY_PATH:deps/macos_toolchain/lib deps/macos_toolchain/bin/o64-clang | ||||||
|  | $(filter $(BUILD_DIR)/macosrelease-arm/%,$(ALL_TARGETS)): CC = PATH=deps/macos_toolchain/bin:$$PATH LD_LIBRARY_PATH=$$LD_LIBRARY_PATH:deps/macos_toolchain/lib deps/macos_toolchain/bin/oa64-clang | ||||||
| endif | endif | ||||||
| $(ANDROID_X86_64_TARGETS): ANDROID_NDK_TARGET_TRIPLE := x86_64-linux-android | $(ANDROID_X86_64_TARGETS): ANDROID_NDK_TARGET_TRIPLE := x86_64-linux-android | ||||||
| $(ANDROID_X86_TARGETS): ANDROID_NDK_TARGET_TRIPLE := i686-linux-android | $(ANDROID_X86_TARGETS): ANDROID_NDK_TARGET_TRIPLE := i686-linux-android | ||||||
| @@ -250,30 +293,39 @@ $(ANDROID_TARGETS): AS = $(CC) | |||||||
| $(ANDROID_TARGETS): CFLAGS += \ | $(ANDROID_TARGETS): CFLAGS += \ | ||||||
| 	-target $(ANDROID_NDK_TARGET_TRIPLE)$(ANDROID_MIN_SDK_VERSION) \ | 	-target $(ANDROID_NDK_TARGET_TRIPLE)$(ANDROID_MIN_SDK_VERSION) \ | ||||||
| 	-Wno-unknown-warning-option | 	-Wno-unknown-warning-option | ||||||
| $(ANDROID_ARMV7A_TARGETS): CFLAGS += -Ideps/openssl/android/armeabi-v7a/usr/local/include | $(ANDROID_ARMV7A_TARGETS): CFLAGS += -Iout/openssl/android/armeabi-v7a/usr/local/include | ||||||
| $(ANDROID_ARMV7A_TARGETS): LDFLAGS += -Ldeps/openssl/android/armeabi-v7a/usr/local/lib | $(ANDROID_ARMV7A_TARGETS): LDFLAGS += -Lout/openssl/android/armeabi-v7a/usr/local/lib | ||||||
| $(ANDROID_ARM64_TARGETS): CFLAGS += -Ideps/openssl/android/arm64-v8a/usr/local/include | $(ANDROID_ARM64_TARGETS): CFLAGS += -Iout/openssl/android/arm64-v8a/usr/local/include | ||||||
| $(ANDROID_ARM64_TARGETS): LDFLAGS += -Ldeps/openssl/android/arm64-v8a/usr/local/lib | $(ANDROID_ARM64_TARGETS): LDFLAGS += -Lout/openssl/android/arm64-v8a/usr/local/lib | ||||||
| $(ANDROID_X86_TARGETS): CFLAGS += -Ideps/openssl/android/x86/usr/local/include | $(ANDROID_X86_TARGETS): CFLAGS += -Iout/openssl/android/x86/usr/local/include | ||||||
| $(ANDROID_X86_TARGETS): CFLAGS += -Wno-atomic-alignment | $(ANDROID_X86_TARGETS): CFLAGS += -Wno-atomic-alignment | ||||||
| $(ANDROID_X86_TARGETS): LDFLAGS += -Ldeps/openssl/android/x86/usr/local/lib | $(ANDROID_X86_TARGETS): LDFLAGS += -Lout/openssl/android/x86/usr/local/lib | ||||||
| $(ANDROID_X86_64_TARGETS): CFLAGS += -Ideps/openssl/android/x86_64/usr/local/include | $(ANDROID_X86_64_TARGETS): CFLAGS += -Iout/openssl/android/x86_64/usr/local/include | ||||||
| $(ANDROID_X86_64_TARGETS): LDFLAGS += -Ldeps/openssl/android/x86_64/usr/local/lib | $(ANDROID_X86_64_TARGETS): LDFLAGS += -Lout/openssl/android/x86_64/usr/local/lib | ||||||
| $(NONMACOS_TARGETS): CFLAGS += -Wno-cast-function-type | $(NONMACOS_TARGETS): CFLAGS += -Wno-cast-function-type | ||||||
| $(DEADSTRIP_TARGETS): LDFLAGS += -Wl,--gc-sections | $(MACOS_TARGETS): LDFLAGS += -Wl,-dead_strip | ||||||
| $(IOS_TARGETS): CFLAGS += -miphoneos-version-min=9.0 | $(NONMACOS_TARGETS): LDFLAGS += -Wl,--gc-sections -Wl,--as-needed | ||||||
| $(IOS_TARGETS): LDFLAGS += -miphoneos-version-min=9.0 | $(IOS_TARGETS): CFLAGS += -miphoneos-version-min=$(IPHONEOS_VERSION_MIN) | ||||||
|  | $(IOS_TARGETS): LDFLAGS += -miphoneos-version-min=$(IPHONEOS_VERSION_MIN) | ||||||
| ifeq ($(UNAME_S),Darwin) | ifeq ($(UNAME_S),Darwin) | ||||||
| $(IOS_TARGETS): CFLAGS += -Ideps/openssl/ios/ios64-xcrun/usr/local/include | $(IOS_TARGETS): CFLAGS += -Iout/openssl/ios/ios64-xcrun/usr/local/include | ||||||
| $(IOS_TARGETS): LDFLAGS += -Ldeps/openssl/ios/ios64-xcrun/usr/local/lib | $(IOS_TARGETS): LDFLAGS += -Lout/openssl/ios/ios64-xcrun/usr/local/lib | ||||||
| else | else | ||||||
| $(IOS_TARGETS): CFLAGS += -Ideps/openssl/$(UNAME_S)/ios64-cross/usr/local/include | $(IOS_TARGETS): CFLAGS += -Iout/openssl/$(UNAME_S)/ios64-cross/usr/local/include | ||||||
| $(IOS_TARGETS): LDFLAGS += -Ldeps/openssl/$(UNAME_S)/ios64-cross/usr/local/lib | $(IOS_TARGETS): LDFLAGS += -Lout/openssl/$(UNAME_S)/ios64-cross/usr/local/lib | ||||||
|  | $(filter $(BUILD_DIR)/macosdebug-x86_64/%,$(ALL_TARGETS)): CFLAGS += -Iout/openssl/$(UNAME_S)/macos-x86_64/usr/local/include | ||||||
|  | $(filter $(BUILD_DIR)/macosdebug-arm/%,$(ALL_TARGETS)): CFLAGS += -Iout/openssl/$(UNAME_S)/macos-arm/usr/local/include | ||||||
|  | $(filter $(BUILD_DIR)/macosdebug-x86_64/%,$(ALL_TARGETS)): LDFLAGS += -Lout/openssl/$(UNAME_S)/macos-x86_64/usr/local/lib | ||||||
|  | $(filter $(BUILD_DIR)/macosdebug-arm/%,$(ALL_TARGETS)): LDFLAGS += -Lout/openssl/$(UNAME_S)/macos-arm/usr/local/lib | ||||||
|  | $(filter $(BUILD_DIR)/macosrelease-x86_64/%,$(ALL_TARGETS)): CFLAGS += -Iout/openssl/$(UNAME_S)/macos-x86_64/usr/local/include | ||||||
|  | $(filter $(BUILD_DIR)/macosrelease-arm/%,$(ALL_TARGETS)): CFLAGS += -Iout/openssl/$(UNAME_S)/macos-arm/usr/local/include | ||||||
|  | $(filter $(BUILD_DIR)/macosrelease-x86_64/%,$(ALL_TARGETS)): LDFLAGS += -Lout/openssl/$(UNAME_S)/macos-x86_64/usr/local/lib | ||||||
|  | $(filter $(BUILD_DIR)/macosrelease-arm/%,$(ALL_TARGETS)): LDFLAGS += -Lout/openssl/$(UNAME_S)/macos-arm/usr/local/lib | ||||||
| endif | endif | ||||||
| $(IOSSIM_TARGETS): CFLAGS += -Ideps/openssl/ios/iossimulator-xcrun/usr/local/include | $(IOSSIM_TARGETS): CFLAGS += -Iout/openssl/ios/iossimulator-xcrun/usr/local/include | ||||||
| $(IOSSIM_TARGETS): LDFLAGS += -Ldeps/openssl/ios/iossimulator-xcrun/usr/local/lib | $(IOSSIM_TARGETS): LDFLAGS += -Lout/openssl/ios/iossimulator-xcrun/usr/local/lib | ||||||
| $(LINUX_TARGETS) $(MACOS_TARGETS): CFLAGS += -Ideps/openssl/$(UNAME_S)/$(UNAME_M)/usr/local/include | $(HOST_TARGETS): CFLAGS += -Iout/openssl/$(UNAME_S)/$(UNAME_M)/usr/local/include | ||||||
| $(LINUX_TARGETS) $(MACOS_TARGETS): LDFLAGS += -Ldeps/openssl/$(UNAME_S)/$(UNAME_M)/usr/local/lib | $(HOST_TARGETS): LDFLAGS += -Lout/openssl/$(UNAME_S)/$(UNAME_M)/usr/local/lib | ||||||
|  |  | ||||||
| ifeq ($(UNAME_M),x86_64) | ifeq ($(UNAME_M),x86_64) | ||||||
| ifeq ($(UNAME_S),Linux) | ifeq ($(UNAME_S),Linux) | ||||||
| @@ -292,13 +344,14 @@ endif | |||||||
|  |  | ||||||
| get_objs = \ | get_objs = \ | ||||||
| 	$(foreach build_type,$(BUILD_TYPES),$(addprefix $(BUILD_DIR)/$(build_type)/,$(addsuffix .o,$(basename $(value $(1)))))) \ | 	$(foreach build_type,$(BUILD_TYPES),$(addprefix $(BUILD_DIR)/$(build_type)/,$(addsuffix .o,$(basename $(value $(1)))))) \ | ||||||
| 	$(foreach build_type,debug release armdebug armrelease,$(addprefix $(BUILD_DIR)/$(build_type)/,$(addsuffix .o,$(basename $(value $(1)_unix))))) \ |  | ||||||
| 	$(foreach build_type,windebug winrelease,$(addprefix $(BUILD_DIR)/$(build_type)/,$(addsuffix .o,$(basename $(value $(1)_win))))) \ | 	$(foreach build_type,windebug winrelease,$(addprefix $(BUILD_DIR)/$(build_type)/,$(addsuffix .o,$(basename $(value $(1)_win))))) \ | ||||||
| 	$(foreach build_type,androiddebug androidrelease androiddebug-x86 androidrelease-x86 androiddebug-x86_64 androidrelease-x86_64 androiddebug-armv7a androiddebug-armv7a,$(addprefix $(BUILD_DIR)/$(build_type)/,$(addsuffix .o,$(basename $(value $(1)_android))))) \ | 	$(foreach build_type,androiddebug androidrelease androiddebug-x86 androidrelease-x86 androiddebug-x86_64 androidrelease-x86_64 androiddebug-armv7a androidrelease-armv7a,$(addprefix $(BUILD_DIR)/$(build_type)/,$(addsuffix .o,$(basename $(value $(1)_android))))) \ | ||||||
| 	$(foreach build_type,androiddebug androidrelease androiddebug-x86 androidrelease-x86 androiddebug-x86_64 androidrelease-x86_64 androiddebug-armv7a androidrelease-armv7a,$(addprefix $(BUILD_DIR)/$(build_type)/,$(addsuffix .o,$(basename $(value $(1)_unix))))) \ | 	$(foreach build_type,androiddebug androidrelease androiddebug-x86 androidrelease-x86 androiddebug-x86_64 androidrelease-x86_64 androiddebug-armv7a androidrelease-armv7a,$(addprefix $(BUILD_DIR)/$(build_type)/,$(addsuffix .o,$(basename $(value $(1)_unix))))) \ | ||||||
| 	$(foreach build_type,macosdebug macosrelease iosdebug iosrelease iossimdebug iossimrelease,$(addprefix $(BUILD_DIR)/$(build_type)/,$(addsuffix .o,$(basename $(value $(1)_macos))))) \ |  | ||||||
| 	$(foreach build_type,iosdebug iosrelease iossimdebug iossimrelease,$(addprefix $(BUILD_DIR)/$(build_type)/,$(addsuffix .o,$(basename $(value $(1)_ios))))) \ | 	$(foreach build_type,iosdebug iosrelease iossimdebug iossimrelease,$(addprefix $(BUILD_DIR)/$(build_type)/,$(addsuffix .o,$(basename $(value $(1)_ios))))) \ | ||||||
| 	$(foreach build_type,androiddebug-x86 androidrelease-x86,$(addprefix $(BUILD_DIR)/$(build_type)/,$(addsuffix .o,$(basename $(value $(1)_x86))))) | 	$(foreach build_type,iosdebug iosrelease iossimdebug iossimrelease macosdebug-arm macosrelease-arm macosdebug-x86_64 macosrelease-x86_64,$(addprefix $(BUILD_DIR)/$(build_type)/,$(addsuffix .o,$(basename $(value $(1)_macos))))) \ | ||||||
|  | 	$(foreach build_type,androiddebug-x86 androidrelease-x86,$(addprefix $(BUILD_DIR)/$(build_type)/,$(addsuffix .o,$(basename $(value $(1)_x86))))) \ | ||||||
|  | 	$(if $(findstring Darwin,$(UNAME_S)),$(foreach build_type,debug release,$(addprefix $(BUILD_DIR)/$(build_type)/,$(addsuffix .o,$(basename $(value $(1)_macos)))))) \ | ||||||
|  | 	$(if $(findstring Darwin,$(UNAME_S)),,$(foreach build_type,debug release armdebug armrelease,$(addprefix $(BUILD_DIR)/$(build_type)/,$(addsuffix .o,$(basename $(value $(1)_unix)))))) | ||||||
|  |  | ||||||
| APP_SOURCES := $(wildcard src/*.c) | APP_SOURCES := $(wildcard src/*.c) | ||||||
| APP_SOURCES_ios := $(wildcard src/*.m) | APP_SOURCES_ios := $(wildcard src/*.m) | ||||||
| @@ -320,10 +373,12 @@ $(APP_OBJS): CFLAGS += \ | |||||||
| 	-Ideps/valgrind \ | 	-Ideps/valgrind \ | ||||||
| 	-Wdouble-promotion \ | 	-Wdouble-promotion \ | ||||||
| 	-Werror | 	-Werror | ||||||
|  | ifneq ($(UNAME_S),Darwin) | ||||||
| ifeq ($(UNAME_M),x86_64) | ifeq ($(UNAME_M),x86_64) | ||||||
| $(filter-out $(BUILD_DIR)/android% $(BUILD_DIR)/macos% $(BUILD_DIR)/ios%,$(APP_OBJS)): CFLAGS += \ | $(filter-out $(BUILD_DIR)/android% $(BUILD_DIR)/ios% $(BUILD_DIR)/macos%,$(APP_OBJS)): CFLAGS += \ | ||||||
| 	-fanalyzer | 	-fanalyzer | ||||||
| endif | endif | ||||||
|  | endif | ||||||
|  |  | ||||||
| ARES_SOURCES := \ | ARES_SOURCES := \ | ||||||
| 	deps/c-ares/src/lib/ares_addrinfo2hostent.c \ | 	deps/c-ares/src/lib/ares_addrinfo2hostent.c \ | ||||||
| @@ -756,12 +811,12 @@ $(MINIUNZIP_OBJS): CFLAGS += \ | |||||||
| LDFLAGS += \ | LDFLAGS += \ | ||||||
| 	-pthread \ | 	-pthread \ | ||||||
| 	-lm | 	-lm | ||||||
| $(LINUX_TARGETS) $(MACOS_TARGETS) $(IOS_TARGETS) $(IOSSIM_TARGETS) $(AARCH64_TARGETS): LDFLAGS += \ | $(HOST_TARGETS) $(IOS_TARGETS) $(IOSSIM_TARGETS) $(AARCH64_TARGETS) $(filter-out $(HOST_TARGETS),$(MACOS_TARGETS)): LDFLAGS += \ | ||||||
| 	-lssl \ | 	-lssl \ | ||||||
| 	-lcrypto | 	-lcrypto | ||||||
| ifneq ($(UNAME_S),Haiku) | ifneq ($(UNAME_S),Haiku) | ||||||
| ifneq ($(UNAME_S),OpenBSD) | ifneq ($(UNAME_S),OpenBSD) | ||||||
| debug release $(MACOS_TARGETS) $(IOS_TARGETS) $(IOSSIM_TARGETS): LDFLAGS += \ | $(HOST_TARGETS) $(IOS_TARGETS) $(IOSSIM_TARGETS): LDFLAGS += \ | ||||||
| 	-ldl | 	-ldl | ||||||
| endif | endif | ||||||
| endif | endif | ||||||
| @@ -794,27 +849,15 @@ $(IOS_TARGETS) $(IOSSIM_TARGETS): LDFLAGS += \ | |||||||
| ## | ## | ||||||
| ## Common targets: | ## Common targets: | ||||||
| ## | ## | ||||||
| debug: ## Build a debug executable for the current platform. | all: $(BUILD_TYPES) ## Build all targets that appear possible to build on this machine. | ||||||
| release: ## Build a release executable for the current platform. | debug: ## Build a debug executable for the current host platform. | ||||||
|  | release: ## Build a release executable for the current host platform. | ||||||
| armdebug: ## Cross-compile aarch64 debug on Linux. | armdebug: ## Cross-compile aarch64 debug on Linux. | ||||||
| armrelease: ## Cross-compile aarch64 release on Linux. | armrelease: ## Cross-compile aarch64 release on Linux. | ||||||
| all: $(BUILD_TYPES) ## Build all targets that appear possible to build on this machine. | windebug: ## Cross-compile a debug win32 executable on Linux. | ||||||
| unix: debug release ## Build all UNIX targets. | winrelease: ## Cross-compile a release win32 executable on Linux. | ||||||
| win: windebug winrelease ## Build all Windows targets. |  | ||||||
| .PHONY: all win unix | .PHONY: all win unix | ||||||
|  |  | ||||||
| ## |  | ||||||
| ## Windows targets: |  | ||||||
| ## |  | ||||||
| windebug: ## Build a debug win32 executable. |  | ||||||
| winrelease: ## Build a release win32 executable. |  | ||||||
|  |  | ||||||
| ## |  | ||||||
| ## MacOS targets: |  | ||||||
| ## |  | ||||||
| macosdebug: ## Build a MacOS debug executable. |  | ||||||
| macosrelease: ## Build a MacOS release executable. |  | ||||||
|  |  | ||||||
| ALL_APP_OBJS := \ | ALL_APP_OBJS := \ | ||||||
| 	$(APP_OBJS) \ | 	$(APP_OBJS) \ | ||||||
| 	$(ARES_OBJS) \ | 	$(ARES_OBJS) \ | ||||||
| @@ -870,6 +913,17 @@ src/android/AndroidManifest.xml : $(firstword $(MAKEFILE_LIST)) | |||||||
| 		-e 's/android:targetSdkVersion="[[:digit:]]*"/android:targetSdkVersion="$(ANDROID_TARGET_SDK_VERSION)"/' \ | 		-e 's/android:targetSdkVersion="[[:digit:]]*"/android:targetSdkVersion="$(ANDROID_TARGET_SDK_VERSION)"/' \ | ||||||
| 		$@ | 		$@ | ||||||
|  |  | ||||||
|  | src/ios/Info.plist : $(firstword $(MAKEFILE_LIST)) | ||||||
|  | 	@echo "[ios_version] $@" | ||||||
|  | 	@cat $@ | \ | ||||||
|  | 		tr '\n' '^' | \ | ||||||
|  | 		sed -r \ | ||||||
|  | 			-e 's@(<key>CFBundleShortVersionString</key>\^[[:space:]]*<string>)[0-9.]*(</string>)@\1$(VERSION_NUMBER:%-wip=%)\2@' \ | ||||||
|  | 			-e 's@(<key>CFBundleVersion</key>\^[[:space:]]*<string>)[[:digit:]]+(</string>)@\1$(VERSION_CODE_IOS)\2@' \ | ||||||
|  | 			-e 's@(<key>MinimumOSVersion</key>\^[[:space:]]*<string>)[0-9.]*(</string>)@\1$(IPHONEOS_VERSION_MIN)\2@' | \ | ||||||
|  | 		tr '^' '\n' > \ | ||||||
|  | 		$@.tmp && mv $@.tmp $@ || rm -f $@.tmp | ||||||
|  |  | ||||||
| ## | ## | ||||||
| ## Android targets: | ## Android targets: | ||||||
| ## | ## | ||||||
| @@ -887,29 +941,30 @@ out/res/layout_activity_main.xml.flat: src/android/res/layout/activity_main.xml | |||||||
| 	@echo "[aapt2] $@" | 	@echo "[aapt2] $@" | ||||||
| 	@$(ANDROID_BUILD_TOOLS)/aapt2 compile -o out/res/ src/android/res/layout/activity_main.xml | 	@$(ANDROID_BUILD_TOOLS)/aapt2 compile -o out/res/ src/android/res/layout/activity_main.xml | ||||||
|  |  | ||||||
| out/res/drawable_icon.xml.flat: src/android/res/drawable/icon.xml | out/res/drawable_%.xml.flat: src/android/res/drawable/%.xml | ||||||
| 	@mkdir -p $(dir $@) | 	@mkdir -p $(dir $@) | ||||||
| 	@echo "[aapt2] $@" | 	@echo "[aapt2] $@" | ||||||
| 	@$(ANDROID_BUILD_TOOLS)/aapt2 compile -o out/res/ src/android/res/drawable/icon.xml | 	@$(ANDROID_BUILD_TOOLS)/aapt2 compile -o out/res/ $(<) | ||||||
|  |  | ||||||
| out/apk/res.apk out/gen/com/unprompted/tildefriends/R.java: out/res/layout_activity_main.xml.flat out/res/drawable_icon.xml.flat src/android/AndroidManifest.xml | out/apk/res.apk out/gen/com/unprompted/tildefriends/R.java: out/res/layout_activity_main.xml.flat out/res/drawable_icon.xml.flat out/res/drawable_logo.xml.flat src/android/AndroidManifest.xml | ||||||
| 	@echo [aapt2 link] res.apk | 	@echo [aapt2 link] res.apk | ||||||
| 	@mkdir -p out/apk/ | 	@mkdir -p out/apk/ | ||||||
| 	@$(ANDROID_BUILD_TOOLS)/aapt2 link -I $(ANDROID_PLATFORM)/android.jar out/res/layout_activity_main.xml.flat out/res/drawable_icon.xml.flat \ | 	@$(ANDROID_BUILD_TOOLS)/aapt2 link -I $(ANDROID_PLATFORM)/android.jar out/res/layout_activity_main.xml.flat out/res/drawable_icon.xml.flat out/res/drawable_logo.xml.flat \ | ||||||
| 		--min-sdk-version $(ANDROID_MIN_SDK_VERSION) \ | 		--min-sdk-version $(ANDROID_MIN_SDK_VERSION) \ | ||||||
| 		--target-sdk-version $(ANDROID_TARGET_SDK_VERSION) \ | 		--target-sdk-version $(ANDROID_TARGET_SDK_VERSION) \ | ||||||
| 		--manifest src/android/AndroidManifest.xml \ | 		--manifest src/android/AndroidManifest.xml \ | ||||||
| 		-o out/apk/res.apk \ | 		-o out/apk/res.apk \ | ||||||
| 		--java out/gen/ | 		--java out/gen/ | ||||||
|  |  | ||||||
| out/apk/res.fdroid.apk out/gen_fdroid/com/unprompted/tildefriends/R.java: out/res/layout_activity_main.xml.flat out/res/drawable_icon.xml.flat src/android/AndroidManifest.xml | out/apk/res.fdroid.apk out/gen_fdroid/com/unprompted/tildefriends/R.java: out/res/layout_activity_main.xml.flat out/res/drawable_icon_fdroid.xml.flat out/res/drawable_logo.xml.flat src/android/AndroidManifest.xml | ||||||
| 	@echo [aapt2 link] res.fdroid.apk | 	@echo [aapt2 link] res.fdroid.apk | ||||||
| 	@mkdir -p out/apk/ | 	@mkdir -p out/apk/ | ||||||
| 	@$(ANDROID_BUILD_TOOLS)/aapt2 link -I $(ANDROID_PLATFORM)/android.jar out/res/layout_activity_main.xml.flat out/res/drawable_icon.xml.flat \ | 	@sed -e 's@drawable/icon@drawable/icon_fdroid@' src/android/AndroidManifest.xml > out/apk/AndroidManifest.fdroid.xml | ||||||
|  | 	@$(ANDROID_BUILD_TOOLS)/aapt2 link -I $(ANDROID_PLATFORM)/android.jar out/res/layout_activity_main.xml.flat out/res/drawable_icon_fdroid.xml.flat out/res/drawable_logo.xml.flat \ | ||||||
| 		--min-sdk-version $(ANDROID_MIN_SDK_VERSION) \ | 		--min-sdk-version $(ANDROID_MIN_SDK_VERSION) \ | ||||||
| 		--target-sdk-version $(ANDROID_TARGET_SDK_VERSION) \ | 		--target-sdk-version $(ANDROID_TARGET_SDK_VERSION) \ | ||||||
| 		--rename-manifest-package com.unprompted.tildefriends.fdroid \ | 		--rename-manifest-package com.unprompted.tildefriends.fdroid \ | ||||||
| 		--manifest src/android/AndroidManifest.xml \ | 		--manifest out/apk/AndroidManifest.fdroid.xml \ | ||||||
| 		-o out/apk/res.fdroid.apk \ | 		-o out/apk/res.fdroid.apk \ | ||||||
| 		--java out/gen_fdroid/ | 		--java out/gen_fdroid/ | ||||||
|  |  | ||||||
| @@ -926,11 +981,11 @@ out/apk/classes.dex: $(CLASS_FILES) | |||||||
| 	@$(ANDROID_BUILD_TOOLS)/d8 --lib $(ANDROID_PLATFORM)/android.jar --output $(dir $@) out/classes/com/unprompted/tildefriends/*.class | 	@$(ANDROID_BUILD_TOOLS)/d8 --lib $(ANDROID_PLATFORM)/android.jar --output $(dir $@) out/classes/com/unprompted/tildefriends/*.class | ||||||
|  |  | ||||||
| PACKAGE_DIRS := \ | PACKAGE_DIRS := \ | ||||||
| 	apps/ \ | 	apps \ | ||||||
| 	core/ \ | 	core \ | ||||||
| 	deps/codemirror/ \ | 	deps/codemirror \ | ||||||
| 	deps/prettier/ \ | 	deps/prettier \ | ||||||
| 	deps/lit/ | 	deps/lit | ||||||
|  |  | ||||||
| RAW_FILES := $(sort $(filter-out apps/blog% apps/issues% apps/welcome% apps/journal% %.map, $(shell find $(PACKAGE_DIRS) -type f -not -name '.*'))) | RAW_FILES := $(sort $(filter-out apps/blog% apps/issues% apps/welcome% apps/journal% %.map, $(shell find $(PACKAGE_DIRS) -type f -not -name '.*'))) | ||||||
|  |  | ||||||
| @@ -960,6 +1015,7 @@ out/TildeFriends.aab: out/apk/classes.dex $(filter-out %debug%, $(ANDROID_TARGET | |||||||
| 		--manifest src/android/AndroidManifest.xml \ | 		--manifest src/android/AndroidManifest.xml \ | ||||||
| 		-R out/res/layout_activity_main.xml.flat \ | 		-R out/res/layout_activity_main.xml.flat \ | ||||||
| 		-R out/res/drawable_icon.xml.flat \ | 		-R out/res/drawable_icon.xml.flat \ | ||||||
|  | 		-R out/res/drawable_logo.xml.flat \ | ||||||
| 		--auto-add-overlay | 		--auto-add-overlay | ||||||
| 	@unzip out/aab/temporary.apk -d out/aab/staging/ | 	@unzip out/aab/temporary.apk -d out/aab/staging/ | ||||||
| 	@mkdir -p out/aab/staging/root/deps | 	@mkdir -p out/aab/staging/root/deps | ||||||
| @@ -986,7 +1042,7 @@ out/TildeFriends.aab: out/apk/classes.dex $(filter-out %debug%, $(ANDROID_TARGET | |||||||
| 	@cp -r deps/codemirror/ out/aab/staging/root/deps/ | 	@cp -r deps/codemirror/ out/aab/staging/root/deps/ | ||||||
| 	@cd out/aab/staging/; zip -r ../base.zip *; cd ../../../ | 	@cd out/aab/staging/; zip -r ../base.zip *; cd ../../../ | ||||||
| 	@java -jar $(BUNDLETOOL) build-bundle --overwrite --config=src/android/BundleConfig.json --modules=out/aab/base.zip --output=$@ | 	@java -jar $(BUNDLETOOL) build-bundle --overwrite --config=src/android/BundleConfig.json --modules=out/aab/base.zip --output=$@ | ||||||
| 	@jarsigner -keystore .keys/android.jks $@ androidKey -storepass android | 	@$(ANDROID_BUILD_TOOLS)/apksigner sign -ks .keys/android.jks --ks-key-alias androidKey -ks-pass pass:android --min-sdk-version=$(ANDROID_MIN_SDK_VERSION) $@ | ||||||
|  |  | ||||||
| aab: out/TildeFriends.aab ## Build an Android App Bundle. | aab: out/TildeFriends.aab ## Build an Android App Bundle. | ||||||
| .PHONY: aab | .PHONY: aab | ||||||
| @@ -1092,12 +1148,18 @@ out/data.zip: $(RAW_FILES) | |||||||
| 	@echo [zip] $@ | 	@echo [zip] $@ | ||||||
| 	@zip -u $@ -q -9 $(RAW_FILES) | 	@zip -u $@ -q -9 $(RAW_FILES) | ||||||
|  |  | ||||||
| out/tildefriends-%.app/tildefriends: out/%/tildefriends out/tildefriends-%.app/Info.plist out/tildefriends-%.app/tildefriends.png out/data.zip | out/zsign_build/zsign: $(wildcard deps/zsign/*.cpp deps/zsign/*.h deps/zsign/*.txt deps/zsign/common/*) | ||||||
|  | 	@+echo [cmake] $@ | ||||||
|  | 	@cmake -B out/zsign_build deps/zsign | ||||||
|  | 	@cmake --build out/zsign_build -- COLOR=0 VERBOSE=0 MAKESILENT=-s | ||||||
|  |  | ||||||
|  | out/tildefriends-%.app/tildefriends: out/%/tildefriends out/tildefriends-%.app/Info.plist out/tildefriends-%.app/tildefriends.png out/data.zip $(if $(HAVE_LINUX_IOS),out/zsign_build/zsign) | ||||||
| 	@mkdir -p $(dir $@) | 	@mkdir -p $(dir $@) | ||||||
| 	@cp -v $< $@ | 	@cp -v $(filter-out out/zsign%,$<) $@ | ||||||
| 	@cp -v out/data.zip $(@D)/ | 	@cp -v out/data.zip $(@D)/ | ||||||
| ifeq ($(HAVE_LINUX_IOS),1) | ifeq ($(HAVE_LINUX_IOS),1) | ||||||
| 	@zsign -q -k .keys/apple.p12 -f -m src/ios/embedded.mobileprovision $(realpath $(dir $@)) | 	@mkdir -p $(realpath $(dir $@))/_CodeSignature | ||||||
|  | 	@out/zsign_build/zsign -q -k .keys/apple.p12 -f -m src/ios/embedded.mobileprovision $(realpath $(dir $@)) | ||||||
| endif | endif | ||||||
| .SECONDARY: | .SECONDARY: | ||||||
| out/tildefriends-%.ipa: out/tildefriends-ios%.app/tildefriends | out/tildefriends-%.ipa: out/tildefriends-ios%.app/tildefriends | ||||||
| @@ -1135,26 +1197,34 @@ iossimdebuggo: out/tildefriends-iossimdebug.app/tildefriends ## Build, install, | |||||||
| 	xcrun simctl launch booted com.unprompted.tildefriends | 	xcrun simctl launch booted com.unprompted.tildefriends | ||||||
| .PHONY: iossimdebuggo | .PHONY: iossimdebuggo | ||||||
|  |  | ||||||
| ANDROID_DEPS := deps/openssl/android/arm64-v8a/usr/local/lib/libssl.a | ANDROID_DEPS := out/openssl/android/arm64-v8a/usr/local/lib/libssl.a | ||||||
| $(ANDROID_DEPS): | $(ANDROID_DEPS): | ||||||
| 	+@ANDROID_NDK_ROOT=$(ANDROID_NDK) tools/ssl-android | 	+@export ANDROID_NDK_ROOT=$(ANDROID_NDK) | ||||||
|  | 	+@export BUILD_PLATFORM=android | ||||||
|  | 	+@export TOOLCHAIN=$(ANDROID_NDK)/toolchains/llvm/prebuilt/linux-x86_64 | ||||||
|  | 	+@PATH="$$TOOLCHAIN/x86_64-linux-android/bin:$$TOOLCHAIN/bin:$$PATH" BUILD_TARGET=x86_64 SSL_TARGET=android-x86_64 OPTIONS="-D__ANDROID_API__=$(ANDROID_MIN_SDK_VERSION) -Wno-macro-redefined" tools/ssl-local | ||||||
|  | 	+@PATH="$$TOOLCHAIN/i686-linux-android/bin:$$TOOLCHAIN/bin:$$PATH" BUILD_TARGET=x86 SSL_TARGET=android-x86 OPTIONS="-D__ANDROID_API__=$(ANDROID_MIN_SDK_VERSION) -Wno-macro-redefined" tools/ssl-local | ||||||
|  | 	+@PATH="$$TOOLCHAIN/arm-linux-androideabi/bin:$$TOOLCHAIN/bin:$$PATH" BUILD_TARGET=armeabi-v7a SSL_TARGET=android-arm OPTIONS="--target=armv7a-linux-androideabi -Wl,--fix-cortex-a8 -D__ANDROID_API__=$(ANDROID_MIN_SDK_VERSION) -Wno-macro-redefined" tools/ssl-local | ||||||
|  | 	+@PATH="$$TOOLCHAIN/aarch64-linux-android/bin:$$TOOLCHAIN/bin:$$PATH" BUILD_TARGET=arm64-v8a SSL_TARGET=android-arm64 OPTIONS="-D__ANDROID_API__=$(ANDROID_MIN_SDK_VERSION) -Wno-macro-redefined" tools/ssl-local | ||||||
| $(filter $(BUILD_DIR)/android%,$(APP_OBJS)): | $(ANDROID_DEPS) | $(filter $(BUILD_DIR)/android%,$(APP_OBJS)): | $(ANDROID_DEPS) | ||||||
|  |  | ||||||
| ifeq ($(UNAME_S),Linux) | ifeq ($(UNAME_S),Linux) | ||||||
| LOCAL_DEPS := deps/openssl/$(UNAME_S)/$(UNAME_M)/usr/local/lib/libssl.a | ifneq ($(USE_SYSTEM_SSL),1) | ||||||
|  | LOCAL_DEPS := out/openssl/$(UNAME_S)/$(UNAME_M)/usr/local/lib/libssl.a | ||||||
| $(LOCAL_DEPS): | $(LOCAL_DEPS): | ||||||
| 	+@/usr/bin/env bash tools/ssl-local | 	+@tools/ssl-local | ||||||
| $(filter $(BUILD_DIR)/debug/%,$(APP_OBJS)) $(filter $(BUILD_DIR)/release/%,$(APP_OBJS)): | $(LOCAL_DEPS) | $(filter $(BUILD_DIR)/debug/%,$(APP_OBJS)) $(filter $(BUILD_DIR)/release/%,$(APP_OBJS)): | $(LOCAL_DEPS) | ||||||
|  | endif | ||||||
|  |  | ||||||
| ifeq ($(HAVE_CROSS_AARCH64),1) | ifeq ($(HAVE_CROSS_AARCH64),1) | ||||||
| LOCAL_DEPS := deps/openssl/$(UNAME_S)/aarch64/usr/local/lib/libssl.a | LOCAL_DEPS := out/openssl/$(UNAME_S)/aarch64/usr/local/lib/libssl.a | ||||||
| $(LOCAL_DEPS): | $(LOCAL_DEPS): | ||||||
| 	+@OPTIONS="--cross-compile-prefix=aarch64-linux-gnu-" BUILD_TARGET=aarch64 tools/ssl-local | 	+@OPTIONS="--cross-compile-prefix=aarch64-linux-gnu-" BUILD_TARGET=aarch64 SSL_TARGET=linux-aarch64 tools/ssl-local | ||||||
| $(filter $(BUILD_DIR)/armdebug/%,$(APP_OBJS)) $(filter $(BUILD_DIR)/armrelease/%,$(APP_OBJS)): | $(LOCAL_DEPS) | $(filter $(BUILD_DIR)/armdebug/%,$(APP_OBJS)) $(filter $(BUILD_DIR)/armrelease/%,$(APP_OBJS)): | $(LOCAL_DEPS) | ||||||
| endif | endif | ||||||
|  |  | ||||||
| ifeq ($(HAVE_LINUX_IOS),1) | ifeq ($(HAVE_LINUX_IOS),1) | ||||||
| LOCAL_DEPS := deps/openssl/$(UNAME_S)/ios64-cross/usr/local/lib/libssl.a | LOCAL_DEPS := out/openssl/$(UNAME_S)/ios64-cross/usr/local/lib/libssl.a | ||||||
| $(LOCAL_DEPS): | $(LOCAL_DEPS): | ||||||
| 	+@PATH=deps/ios_toolchain/target/bin:$$PATH \ | 	+@PATH=deps/ios_toolchain/target/bin:$$PATH \ | ||||||
| 		BUILD_TARGET=ios64-cross \ | 		BUILD_TARGET=ios64-cross \ | ||||||
| @@ -1163,33 +1233,63 @@ $(LOCAL_DEPS): | |||||||
| 		CROSS_TOP=../../deps/ios_toolchain/target \ | 		CROSS_TOP=../../deps/ios_toolchain/target \ | ||||||
| 		CROSS_SDK=iPhoneOS18.2.sdk \ | 		CROSS_SDK=iPhoneOS18.2.sdk \ | ||||||
| 		CC=clang \ | 		CC=clang \ | ||||||
| 		OPTIONS=-miphoneos-version-min=9.0 \ | 		OPTIONS=-miphoneos-version-min=$(IPHONEOS_VERSION_MIN) \ | ||||||
| 		tools/ssl-local | 		tools/ssl-local | ||||||
| $(filter $(BUILD_DIR)/ios%,$(APP_OBJS)): | $(LOCAL_DEPS) | $(filter $(BUILD_DIR)/ios%,$(APP_OBJS)): | $(LOCAL_DEPS) | ||||||
| endif | endif | ||||||
|  |  | ||||||
|  | ifeq ($(HAVE_LINUX_MACOS),1) | ||||||
|  | LOCAL_DEPS := out/openssl/$(UNAME_S)/macos-arm/usr/local/lib/libssl.a | ||||||
|  | $(LOCAL_DEPS): | ||||||
|  | 	+@PATH=../../deps/macos_toolchain/bin:$$PATH \ | ||||||
|  | 		BUILD_TARGET=macos-arm \ | ||||||
|  | 		SSL_TARGET=darwin64-arm64 \ | ||||||
|  | 		CC=../../deps/macos_toolchain/bin/oa64-clang \ | ||||||
|  | 		RANLIB=../../deps/macos_toolchain/bin/x86_64-apple-darwin24-ranlib \ | ||||||
|  | 		AR=../../deps/macos_toolchain/bin/arm64-apple-darwin24-ar \ | ||||||
|  | 		tools/ssl-local | ||||||
|  | $(filter $(BUILD_DIR)/macosrelease-arm/% $(BUILD_DIR)/macosdebug-arm/%,$(APP_OBJS)): | $(LOCAL_DEPS) | ||||||
|  |  | ||||||
|  | LOCAL_DEPS := out/openssl/$(UNAME_S)/macos-x86_64/usr/local/lib/libssl.a | ||||||
|  | $(LOCAL_DEPS): | ||||||
|  | 	+@PATH=../../deps/macos_toolchain/bin:$$PATH \ | ||||||
|  | 		BUILD_TARGET=macos-x86_64 \ | ||||||
|  | 		SSL_TARGET=darwin64-x86_64 \ | ||||||
|  | 		CC=../../deps/macos_toolchain/bin/o64-clang \ | ||||||
|  | 		RANLIB=../../deps/macos_toolchain/bin/x86_64-apple-darwin24-ranlib \ | ||||||
|  | 		AR=../../deps/macos_toolchain/bin/x86_64-apple-darwin24-ar \ | ||||||
|  | 		tools/ssl-local | ||||||
|  | $(filter $(BUILD_DIR)/macosrelease-x86_64/% $(BUILD_DIR)/macosdebug-x86_64/%,$(APP_OBJS)): | $(LOCAL_DEPS) | ||||||
|  | endif | ||||||
| endif | endif | ||||||
|  |  | ||||||
| ifeq ($(UNAME_S),Darwin) | ifeq ($(UNAME_S),Darwin) | ||||||
| LOCAL_DEPS := deps/openssl/$(UNAME_S)/$(UNAME_M)/usr/local/lib/libssl.a | LOCAL_DEPS := out/openssl/$(UNAME_S)/$(UNAME_M)/usr/local/lib/libssl.a | ||||||
| $(LOCAL_DEPS): | $(LOCAL_DEPS): | ||||||
| 	+@tools/ssl-local | 	+@tools/ssl-local | ||||||
| $(filter $(BUILD_DIR)/macosdebug/%,$(APP_OBJS)) $(filter $(BUILD_DIR)/macosrelease/%,$(APP_OBJS)): | $(LOCAL_DEPS) | $(filter $(BUILD_DIR)/debug/%,$(APP_OBJS)) $(filter $(BUILD_DIR)/release/%,$(APP_OBJS)): | $(LOCAL_DEPS) | ||||||
| endif | endif | ||||||
|  |  | ||||||
| ifeq ($(HAVE_WIN),1) | ifeq ($(HAVE_WIN),1) | ||||||
| WINDOWS_DEPS := deps/openssl/mingw64/usr/local/lib/libssl.a | WINDOWS_DEPS := out/openssl/$(UNAME_S)/mingw64/usr/local/lib/libssl.a | ||||||
| $(WINDOWS_DEPS): | $(WINDOWS_DEPS): | ||||||
| 	+@tools/ssl-mingw64 | 	+@BUILD_TARGET=mingw64 SSL_TARGET=mingw64 OPTIONS="--cross-compile-prefix=x86_64-w64-mingw32-" tools/ssl-local | ||||||
| $(filter $(BUILD_DIR)/win%,$(APP_OBJS)): | $(WINDOWS_DEPS) | $(filter $(BUILD_DIR)/win%,$(APP_OBJS)): | $(WINDOWS_DEPS) | ||||||
| endif | endif | ||||||
|  |  | ||||||
| ifeq ($(UNAME_S),Darwin) | ifeq ($(UNAME_S),Darwin) | ||||||
| IOS_DEPS := deps/openssl/ios/ios64-xcrun/usr/local/lib/libssl.a | IOS_DEPS := out/openssl/ios/ios64-xcrun/usr/local/lib/libssl.a | ||||||
| $(IOS_DEPS): | $(IOS_DEPS): | ||||||
| 	+@tools/ssl-ios | 	+@BUILD_PLATFORM=ios BUILD_TARGET=ios64-xcrun SSL_TARGET=ios64-xcrun OPTIONS="-fPIC -Wno-macro-redefined -miphoneos-version-min=$(IPHONEOS_VERSION_MIN)" tools/ssl-local | ||||||
|  | 	+@BUILD_PLATFORM=ios BUILD_TARGET=iossimulator-xcrun SSL_TARGET=iossimulator-xcrun OPTIONS="-fPIC -Wno-macro-redefined" tools/ssl-local | ||||||
| $(filter $(BUILD_DIR)/ios%,$(APP_OBJS)): | $(IOS_DEPS) | $(filter $(BUILD_DIR)/ios%,$(APP_OBJS)): | $(IOS_DEPS) | ||||||
| endif | endif | ||||||
|  |  | ||||||
|  | out/macos%/tildefriends: out/macos%-arm/tildefriends out/macos%-x86_64/tildefriends | ||||||
|  | 	@echo [lipo] $@ | ||||||
|  | 	@mkdir -p $(@D) | ||||||
|  | 	@deps/macos_toolchain/bin/lipo -create -output $@ $^ | ||||||
|  |  | ||||||
| ## | ## | ||||||
| ## Linux package targets: | ## Linux package targets: | ||||||
| ## | ## | ||||||
| @@ -1258,7 +1358,6 @@ tarball: ## Build an all-inclusive source tarball (.tar.xz). | |||||||
| 		--exclude=deps/libsodium/test \ | 		--exclude=deps/libsodium/test \ | ||||||
| 		--exclude=deps/libuv/docs \ | 		--exclude=deps/libuv/docs \ | ||||||
| 		--exclude=deps/libuv/test \ | 		--exclude=deps/libuv/test \ | ||||||
| 		--exclude=deps/openssl \ |  | ||||||
| 		--exclude=deps/speedscope/*.map \ | 		--exclude=deps/speedscope/*.map \ | ||||||
| 		--exclude=deps/sqlite/shell.c \ | 		--exclude=deps/sqlite/shell.c \ | ||||||
| 		--exclude=deps/zlib/contrib/vstudio \ | 		--exclude=deps/zlib/contrib/vstudio \ | ||||||
| @@ -1269,7 +1368,20 @@ tarball: ## Build an all-inclusive source tarball (.tar.xz). | |||||||
| .PHONY: tarball | .PHONY: tarball | ||||||
|  |  | ||||||
| dist: ## Build versions of all distributables for release. | dist: ## Build versions of all distributables for release. | ||||||
| dist: release-apk iosrelease-ipa aab $(if $(HAVE_WIN), out/winrelease/tildefriends.standalone.exe) out/TildeFriends-release.fdroid.apk appimage tarball out/release/tildefriends.standalone $(if $(HAVE_CROSS_AARCH64), out/armrelease/tildefriends.standalone) | dist: release-apk aab out/TildeFriends-release.fdroid.apk appimage tarball out/release/tildefriends.standalone | ||||||
|  | ifeq ($(HAVE_LINUX_IOS),1) | ||||||
|  | dist: iosrelease-ipa | ||||||
|  | endif | ||||||
|  | ifeq ($(HAVE_LINUX_MACOS),1) | ||||||
|  | dist: out/macosrelease/tildefriends.standalone | ||||||
|  | endif | ||||||
|  | ifeq ($(HAVE_WIN),1) | ||||||
|  | dist: out/winrelease/tildefriends.standalone.exe | ||||||
|  | endif | ||||||
|  | ifeq ($(HAVE_CROSS_AARCH64),1) | ||||||
|  | dist: out/armrelease/tildefriends.standalone | ||||||
|  | endif | ||||||
|  | dist: | ||||||
| 	@mkdir -p dist/ | 	@mkdir -p dist/ | ||||||
| 	@echo "[cp] tildefriends-$(VERSION_NUMBER).tar.xz" | 	@echo "[cp] tildefriends-$(VERSION_NUMBER).tar.xz" | ||||||
| 	@cp out/tildefriends-$(VERSION_NUMBER).tar.xz dist/tildefriends-$(VERSION_NUMBER).tar.xz | 	@cp out/tildefriends-$(VERSION_NUMBER).tar.xz dist/tildefriends-$(VERSION_NUMBER).tar.xz | ||||||
| @@ -1277,8 +1389,10 @@ dist: release-apk iosrelease-ipa aab $(if $(HAVE_WIN), out/winrelease/tildefrien | |||||||
| 	@cp out/TildeFriends-x86-release.zopfli.apk dist/TildeFriends-x86-$(VERSION_NUMBER).apk | 	@cp out/TildeFriends-x86-release.zopfli.apk dist/TildeFriends-x86-$(VERSION_NUMBER).apk | ||||||
| 	@echo "[cp] TildeFriends-arm-$(VERSION_NUMBER).apk" | 	@echo "[cp] TildeFriends-arm-$(VERSION_NUMBER).apk" | ||||||
| 	@cp out/TildeFriends-arm-release.zopfli.apk dist/TildeFriends-arm-$(VERSION_NUMBER).apk | 	@cp out/TildeFriends-arm-release.zopfli.apk dist/TildeFriends-arm-$(VERSION_NUMBER).apk | ||||||
| 	@echo "[cp] TildeFriends-$(VERSION_NUMBER).ipa" | 	@test $(HAVE_LINUX_IOS) && echo "[cp] TildeFriends-$(VERSION_NUMBER).ipa" | ||||||
| 	@cp out/tildefriends-release.ipa dist/TildeFriends-$(VERSION_NUMBER).ipa | 	@test $(HAVE_LINUX_IOS) && cp out/tildefriends-release.ipa dist/TildeFriends-$(VERSION_NUMBER).ipa | ||||||
|  | 	@test $(HAVE_LINUX_MACOS) && echo "[cp] tildefriends-macos-$(VERSION_NUMBER)" | ||||||
|  | 	@test $(HAVE_LINUX_MACOS) && cp out/macosrelease/tildefriends.standalone dist/tildefriends-macos-$(VERSION_NUMBER) | ||||||
| 	@test $(HAVE_WIN) && echo "[cp] tildefriends-$(VERSION_NUMBER).exe" | 	@test $(HAVE_WIN) && echo "[cp] tildefriends-$(VERSION_NUMBER).exe" | ||||||
| 	@test $(HAVE_WIN) && cp out/winrelease/tildefriends.standalone.exe dist/tildefriends-$(VERSION_NUMBER).exe | 	@test $(HAVE_WIN) && cp out/winrelease/tildefriends.standalone.exe dist/tildefriends-$(VERSION_NUMBER).exe | ||||||
| 	@echo "[cp] TildeFriends-$(VERSION_NUMBER).aab" | 	@echo "[cp] TildeFriends-$(VERSION_NUMBER).aab" | ||||||
| @@ -1300,6 +1414,17 @@ dist-test: dist ## Exercise some built distributable files, making sure they wor | |||||||
| 	@rm -rf tildefriends-$(VERSION_NUMBER) | 	@rm -rf tildefriends-$(VERSION_NUMBER) | ||||||
| .PHONY: dist-test | .PHONY: dist-test | ||||||
|  |  | ||||||
|  | dist-ios: iosrelease-app | ||||||
|  | 	rm -rfv out/Payload out/tildefriends.ipa | ||||||
|  | 	mkdir -p out/Payload/tildefriends.app | ||||||
|  | 	cp -avR out/tildefriends-iosrelease.app/* out/Payload/tildefriends.app/ | ||||||
|  | 	cp src/ios/tildefriends.png out/Payload/tildefriends.app/ | ||||||
|  | 	cp src/ios/icons/Assets.car out/Payload/tildefriends.app/ | ||||||
|  | 	cp src/ios/distribution.mobileprovision out/Payload/tildefriends.app/embedded.mobileprovision | ||||||
|  | 	xcrun -sdk iphoneos codesign -f -s 'Apple Distribution' --entitlements src/ios/Entitlements.plist --generate-entitlement-der out/Payload/tildefriends.app | ||||||
|  | 	cd out; zip -r tildefriends.ipa Payload; cd .. | ||||||
|  | 	xcrun -sdk iphoneos altool --upload-app -f out/tildefriends.ipa -t ios -u $$(cat .keys/altool-user) -p $$(cat .keys/altool-password) | ||||||
|  |  | ||||||
| ## | ## | ||||||
| ## Targets for tidying up: | ## Targets for tidying up: | ||||||
| ## | ## | ||||||
|   | |||||||
							
								
								
									
										22
									
								
								README.md
									
									
									
									
									
								
							
							
						
						| @@ -1,18 +1,16 @@ | |||||||
| # Tilde Friends | # Tilde Friends | ||||||
|  |  | ||||||
| Tilde Friends is a tool for making and sharing. | Tilde Friends participates in the Secure Scuttlebutt decentralized social | ||||||
|  | network while also functioning as a platform for making, sharing, and running | ||||||
|  | web applications. | ||||||
|  |  | ||||||
| A public instance lives at https://www.tildefriends.net/. | A public instance lives at https://www.tildefriends.net/. | ||||||
|  |  | ||||||
| It is both a peer-to-peer social network client, participating in Secure |  | ||||||
| Scuttlebutt, as well as a platform for writing and running web applications. |  | ||||||
|  |  | ||||||
| ## Goals | ## Goals | ||||||
|  |  | ||||||
| 1. Make it easy and fun to run all sorts of web applications. | 1. Be the fanciest, best-maintained Secure Scuttlebutt client in town. | ||||||
| 2. Provide security that is easy to understand and protects your data. | 1. Make it easy to make, share, and run all sorts of applications while | ||||||
| 3. Make creating and sharing web applications accessible to anyone with a |    respecting the privacy and safety of your data. | ||||||
|    browser. |  | ||||||
|  |  | ||||||
| ## Getting the Source | ## Getting the Source | ||||||
|  |  | ||||||
| @@ -40,8 +38,7 @@ dependencies in the right places. | |||||||
|  |  | ||||||
| ### Requirements | ### Requirements | ||||||
|  |  | ||||||
| On Linux only, system OpenSSL libraries (`libssl-dev`, in debian-speak) are | System OpenSSL libraries are assumed to be available on Haiku and OpenSSL. | ||||||
| assumed to be available. |  | ||||||
|  |  | ||||||
| On MacOS, Xcode's command-line tools are expected to be available. | On MacOS, Xcode's command-line tools are expected to be available. | ||||||
|  |  | ||||||
| @@ -63,13 +60,12 @@ repository root as the current working directory. `tildefriends -h` lists | |||||||
| further options. | further options. | ||||||
|  |  | ||||||
| The first user to create an account and log in will be granted administrative | The first user to create an account and log in will be granted administrative | ||||||
| privileges. Further administration can be done at | privileges. Further administration can be done in the `admin` app at | ||||||
| <http://localhost:12345/~core/admin/>. | <http://localhost:12345/~core/admin/>. | ||||||
|  |  | ||||||
| ## Documentation | ## Documentation | ||||||
|  |  | ||||||
| Docs are a work in progress: | Docs live here: <https://docs.tildefriends.net/>. | ||||||
| <https://dev.tildefriends.net/cory/tildefriends/wiki>. |  | ||||||
|  |  | ||||||
| ## License | ## License | ||||||
|  |  | ||||||
|   | |||||||
| @@ -1,5 +1,5 @@ | |||||||
| { | { | ||||||
| 	"type": "tildefriends-app", | 	"type": "tildefriends-app", | ||||||
| 	"emoji": "🎛", | 	"emoji": "🎛", | ||||||
| 	"previous": "&R49FywYF8CXPhoSEydLbSCgvCddeyTiBwGuDU/gqY+M=.sha256" | 	"previous": "&kmKNyb/uaXNb24gCinJtfS8iWx4cLUWdtl0y2DwEUas=.sha256" | ||||||
| } | } | ||||||
|   | |||||||
| @@ -72,7 +72,7 @@ ${description.value}</textarea | |||||||
| 					</button> | 					</button> | ||||||
| 				</li> | 				</li> | ||||||
| 			`; | 			`; | ||||||
| 		} else { | 		} else if (description.type != 'hidden') { | ||||||
| 			return html` | 			return html` | ||||||
| 				<li class="w3-row"> | 				<li class="w3-row"> | ||||||
| 					<label class="w3-quarter" for=${'gs_' + key} style="font-weight: bold">${title_case(key)}</label> | 					<label class="w3-quarter" for=${'gs_' + key} style="font-weight: bold">${title_case(key)}</label> | ||||||
|   | |||||||
| @@ -1,5 +1,5 @@ | |||||||
| { | { | ||||||
| 	"type": "tildefriends-app", | 	"type": "tildefriends-app", | ||||||
| 	"emoji": "🦀", | 	"emoji": "🦀", | ||||||
| 	"previous": "&m5ZrkzoyAjv7AY6AukzVoWIyUFUL97uuDDWCtxCloAU=.sha256" | 	"previous": "&jAAzd36Nmpw0sRA1Dx9wLiIwGX+q//+S/Han+RLlEOw=.sha256" | ||||||
| } | } | ||||||
|   | |||||||
| @@ -20,6 +20,7 @@ class TfElement extends LitElement { | |||||||
| 			channels_latest: {type: Object}, | 			channels_latest: {type: Object}, | ||||||
| 			guest: {type: Boolean}, | 			guest: {type: Boolean}, | ||||||
| 			url: {type: String}, | 			url: {type: String}, | ||||||
|  | 			private_messages: {type: Array}, | ||||||
| 		}; | 		}; | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| @@ -334,7 +335,7 @@ class TfElement extends LitElement { | |||||||
| 				JSON.stringify(cache) | 				JSON.stringify(cache) | ||||||
| 			); | 			); | ||||||
| 		} | 		} | ||||||
| 		return cache.latest; | 		return [cache.latest, cache.messages]; | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	async load_channels_latest(following) { | 	async load_channels_latest(following) { | ||||||
| @@ -378,9 +379,11 @@ class TfElement extends LitElement { | |||||||
| 		start_time = new Date(); | 		start_time = new Date(); | ||||||
| 		latest_private.then(function (latest) { | 		latest_private.then(function (latest) { | ||||||
| 			self.channels_latest = Object.assign({}, self.channels_latest, { | 			self.channels_latest = Object.assign({}, self.channels_latest, { | ||||||
| 				'🔐': latest, | 				'🔐': latest[0], | ||||||
| 			}); | 			}); | ||||||
| 			console.log('private took', (new Date() - start_time) / 1000.0); | 			console.log('private took', (new Date() - start_time) / 1000.0); | ||||||
|  | 			console.log(latest); | ||||||
|  | 			self.private_messages = latest[1]; | ||||||
| 		}); | 		}); | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| @@ -513,6 +516,7 @@ class TfElement extends LitElement { | |||||||
| 					.channels_unread=${this.channels_unread} | 					.channels_unread=${this.channels_unread} | ||||||
| 					@channelsetunread=${this.channel_set_unread} | 					@channelsetunread=${this.channel_set_unread} | ||||||
| 					.connections=${this.connections} | 					.connections=${this.connections} | ||||||
|  | 					.private_messages=${this.private_messages} | ||||||
| 				></tf-tab-news> | 				></tf-tab-news> | ||||||
| 			`; | 			`; | ||||||
| 		} else if (this.tab === 'connections') { | 		} else if (this.tab === 'connections') { | ||||||
|   | |||||||
| @@ -775,7 +775,9 @@ class TfMessageElement extends LitElement { | |||||||
| 				} | 				} | ||||||
| 			} else { | 			} else { | ||||||
| 				return this.render_small_frame( | 				return this.render_small_frame( | ||||||
| 					html`<div class="w3-container"><b>type</b>: ${content.type}</div>` | 					html`<div class="w3-container"> | ||||||
|  | 						<p><b>type</b>: ${content.type}</p> | ||||||
|  | 					</div>` | ||||||
| 				); | 				); | ||||||
| 			} | 			} | ||||||
| 		} else if (typeof this.message.content == 'string') { | 		} else if (typeof this.message.content == 'string') { | ||||||
|   | |||||||
| @@ -62,6 +62,7 @@ class TfProfileElement extends LitElement { | |||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	modify(change) { | 	modify(change) { | ||||||
|  | 		let self = this; | ||||||
| 		tfrpc.rpc | 		tfrpc.rpc | ||||||
| 			.appendMessage( | 			.appendMessage( | ||||||
| 				this.whoami, | 				this.whoami, | ||||||
| @@ -73,6 +74,10 @@ class TfProfileElement extends LitElement { | |||||||
| 					change | 					change | ||||||
| 				) | 				) | ||||||
| 			) | 			) | ||||||
|  | 			.then(function () { | ||||||
|  | 				self._follow_whoami = undefined; | ||||||
|  | 				self.load(); | ||||||
|  | 			}) | ||||||
| 			.catch(function (error) { | 			.catch(function (error) { | ||||||
| 				alert(error?.message); | 				alert(error?.message); | ||||||
| 			}); | 			}); | ||||||
|   | |||||||
| @@ -17,6 +17,7 @@ class TfTabNewsFeedElement extends LitElement { | |||||||
| 			loading: {type: Number}, | 			loading: {type: Number}, | ||||||
| 			time_range: {type: Array}, | 			time_range: {type: Array}, | ||||||
| 			time_loading: {type: Array}, | 			time_loading: {type: Array}, | ||||||
|  | 			private_messages: {type: Array}, | ||||||
| 		}; | 		}; | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| @@ -148,13 +149,13 @@ class TfTabNewsFeedElement extends LitElement { | |||||||
| 				` | 				` | ||||||
| 					SELECT TRUE AS is_primary, messages.rowid, messages.id, previous, author, sequence, timestamp, hash, json(content) AS content, signature | 					SELECT TRUE AS is_primary, messages.rowid, messages.id, previous, author, sequence, timestamp, hash, json(content) AS content, signature | ||||||
| 					FROM messages | 					FROM messages | ||||||
| 					JOIN json_each(?1) AS following ON messages.author = following.value | 					JOIN json_each(?1) AS private_messages ON messages.id = private_messages.value | ||||||
| 					WHERE | 					WHERE | ||||||
| 						(?2 IS NULL OR (messages.timestamp >= ?2)) AND messages.timestamp < ?3 AND | 						(?2 IS NULL OR (messages.timestamp >= ?2)) AND messages.timestamp < ?3 AND | ||||||
| 						json(messages.content) LIKE '"%' | 						json(messages.content) LIKE '"%' | ||||||
| 					ORDER BY messages.sequence DESC LIMIT 20 | 					ORDER BY messages.sequence DESC LIMIT 20 | ||||||
| 				`, | 				`, | ||||||
| 				[JSON.stringify(this.following), start_time, end_time] | 				[JSON.stringify(this.private_messages), start_time, end_time] | ||||||
| 			); | 			); | ||||||
| 			result = (await this.decrypt(result)).filter((x) => x.decrypted); | 			result = (await this.decrypt(result)).filter((x) => x.decrypted); | ||||||
| 		} else { | 		} else { | ||||||
| @@ -298,7 +299,7 @@ class TfTabNewsFeedElement extends LitElement { | |||||||
| 			let now = new Date().valueOf(); | 			let now = new Date().valueOf(); | ||||||
| 			let start_time = now - 24 * 60 * 60 * 1000; | 			let start_time = now - 24 * 60 * 60 * 1000; | ||||||
| 			this.start_time = start_time; | 			this.start_time = start_time; | ||||||
| 			this.time_range = [this.start_time, now + 24 * 60 * 60 * 1000]; | 			this.time_range = [now + 24 * 60 * 60 * 1000, now + 24 * 60 * 60 * 1000]; | ||||||
| 			messages = await this.fetch_messages(null, this.time_range[1]); | 			messages = await this.fetch_messages(null, this.time_range[1]); | ||||||
| 			this.update_time_range_from_messages( | 			this.update_time_range_from_messages( | ||||||
| 				messages.filter((x) => x.timestamp < this.time_range[1]) | 				messages.filter((x) => x.timestamp < this.time_range[1]) | ||||||
|   | |||||||
| @@ -23,6 +23,7 @@ class TfTabNewsElement extends LitElement { | |||||||
| 			channels_unread: {type: Object}, | 			channels_unread: {type: Object}, | ||||||
| 			channels_latest: {type: Object}, | 			channels_latest: {type: Object}, | ||||||
| 			connections: {type: Array}, | 			connections: {type: Array}, | ||||||
|  | 			private_messages: {type: Array}, | ||||||
| 		}; | 		}; | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| @@ -337,6 +338,7 @@ class TfTabNewsElement extends LitElement { | |||||||
| 						@tf-expand=${this.on_expand} | 						@tf-expand=${this.on_expand} | ||||||
| 						.channels_unread=${this.channels_unread} | 						.channels_unread=${this.channels_unread} | ||||||
| 						.channels_latest=${this.channels_latest} | 						.channels_latest=${this.channels_latest} | ||||||
|  | 						.private_messages=${this.private_messages} | ||||||
| 					></tf-tab-news-feed> | 					></tf-tab-news-feed> | ||||||
| 				</div> | 				</div> | ||||||
| 			</div> | 			</div> | ||||||
|   | |||||||
| @@ -21,7 +21,9 @@ class TfUserElement extends LitElement { | |||||||
| 	render() { | 	render() { | ||||||
| 		let user = this.users[this.id]; | 		let user = this.users[this.id]; | ||||||
| 		let shape = | 		let shape = | ||||||
| 			!user?.follow_depth || user.follow_depth >= 2 ? 'w3-circle' : 'w3-round'; | 			user?.follow_depth === undefined || user.follow_depth >= 2 | ||||||
|  | 				? 'w3-circle' | ||||||
|  | 				: 'w3-round'; | ||||||
| 		let image = html`<span | 		let image = html`<span | ||||||
| 			class=${'w3-theme-l4 ' + shape} | 			class=${'w3-theme-l4 ' + shape} | ||||||
| 			style="display: inline-block; width: 2em; height: 2em; text-align: center; line-height: 2em" | 			style="display: inline-block; width: 2em; height: 2em; text-align: center; line-height: 2em" | ||||||
|   | |||||||
| @@ -1,5 +1,5 @@ | |||||||
| { | { | ||||||
| 	"type": "tildefriends-app", | 	"type": "tildefriends-app", | ||||||
| 	"emoji": "👋", | 	"emoji": "👋", | ||||||
| 	"previous": "&7gFmLW5zSMhmxWWY1+jeRcHdullgujSqGJg94lVgr1k=.sha256" | 	"previous": "&wAb7J6E35xEXpiXsQ6t1RaWTGIvlatUnyH8ipF6pVic=.sha256" | ||||||
| } | } | ||||||
|   | |||||||
| @@ -10,17 +10,6 @@ | |||||||
| 		<link rel="stylesheet" href="brands.min.css" /> | 		<link rel="stylesheet" href="brands.min.css" /> | ||||||
|  |  | ||||||
| 		<style> | 		<style> | ||||||
| 			body, |  | ||||||
| 			h1, |  | ||||||
| 			h2, |  | ||||||
| 			h3, |  | ||||||
| 			h4, |  | ||||||
| 			h5 { |  | ||||||
| 				font-family: 'Poppins', sans-serif; |  | ||||||
| 			} |  | ||||||
| 			body { |  | ||||||
| 				font-size: 16px; |  | ||||||
| 			} |  | ||||||
| 			img { | 			img { | ||||||
| 				margin-bottom: -8px; | 				margin-bottom: -8px; | ||||||
| 			} | 			} | ||||||
| @@ -39,11 +28,14 @@ | |||||||
| 						<b>😎 Tilde Friends</b> | 						<b>😎 Tilde Friends</b> | ||||||
| 					</h1> | 					</h1> | ||||||
| 					<h1 class="w3-xxlarge w3-text-green"> | 					<h1 class="w3-xxlarge w3-text-green"> | ||||||
| 						<b>Make apps and friends from the comfort of your web browser.</b> | 						<b | ||||||
|  | 							>the Secure Scuttlebutt decentralized social network client that's | ||||||
|  | 							<i>fancy🎩</i></b | ||||||
|  | 						> | ||||||
| 					</h1> | 					</h1> | ||||||
| 					<p> | 					<p> | ||||||
| 						Tilde Friends is a platform for building, running, and sharing web | 						In addition to participating in Secure Scuttlebutt, Tilde Friends is | ||||||
| 						applications. | 						a platform for building, running, and sharing applications. | ||||||
| 					</p> | 					</p> | ||||||
| 					<p> | 					<p> | ||||||
| 						Available for lots of devices: | 						Available for lots of devices: | ||||||
| @@ -60,7 +52,7 @@ | |||||||
| 					> | 					> | ||||||
| 					<a | 					<a | ||||||
| 						class="w3-button w3-black w3-padding-large" | 						class="w3-button w3-black w3-padding-large" | ||||||
| 						href="https://www.tildefriends.net/~cory/apps/" | 						href="https://www.tildefriends.net/~core/ssb/" | ||||||
| 						><i class="fa fa-link"></i> Try It</a | 						><i class="fa fa-link"></i> Try It</a | ||||||
| 					> | 					> | ||||||
| 					<a | 					<a | ||||||
| @@ -68,6 +60,11 @@ | |||||||
| 						href="https://dev.tildefriends.net/" | 						href="https://dev.tildefriends.net/" | ||||||
| 						><i class="fa fa-mug-hot"></i> Development</a | 						><i class="fa fa-mug-hot"></i> Development</a | ||||||
| 					> | 					> | ||||||
|  | 					<a | ||||||
|  | 						class="w3-button w3-black w3-padding-large" | ||||||
|  | 						href="https://docs.tildefriends.net/" | ||||||
|  | 						><i class="fa fa-book"></i> Documentation</a | ||||||
|  | 					> | ||||||
| 					<p> | 					<p> | ||||||
| 						<a | 						<a | ||||||
| 							class="w3-button w3-round-large w3-padding w3-blue-gray w3-margin-top" | 							class="w3-button w3-round-large w3-padding w3-blue-gray w3-margin-top" | ||||||
| @@ -223,16 +220,15 @@ | |||||||
|  |  | ||||||
| 		<!-- Technlology Section --> | 		<!-- Technlology Section --> | ||||||
| 		<div class="w3-container w3-padding-64 w3-light-grey w3-center"> | 		<div class="w3-container w3-padding-64 w3-light-grey w3-center"> | ||||||
| 			<h1 class="w3-jumbo"><b>Boring Technology</b></h1> | 			<h1 class="w3-jumbo"><b>Built the Old Fashioned Way</b></h1> | ||||||
| 			<p> | 			<p> | ||||||
| 				Tilde Friends is built using boring, trusted tech. Unless a better | 				Tilde Friends strives to use only simple and widely adopted dependencies | ||||||
| 				reason presents itself, it strives to use only simple and widely adopted | 				in order to keep it easy to build for all sorts of platforms and | ||||||
| 				dependencies in order to keep it easy to build for all sorts of | 				maintainable for a very long time. | ||||||
| 				platforms and maintainable for a very long time. |  | ||||||
| 			</p> | 			</p> | ||||||
| 			<p> | 			<p> | ||||||
| 				Though of course for building Tilde Friends apps, you are free to use | 				Though of course for building Tilde Friends apps, you are free to use | ||||||
| 				whatever fits. | 				whatever fits on top. | ||||||
| 			</p> | 			</p> | ||||||
|  |  | ||||||
| 			<div class="w3-row" style="margin-top: 64px"> | 			<div class="w3-row" style="margin-top: 64px"> | ||||||
|   | |||||||
| @@ -81,9 +81,8 @@ App.prototype.send = function (message) { | |||||||
|  * TODOC |  * TODOC | ||||||
|  * @param {*} request |  * @param {*} request | ||||||
|  * @param {*} response |  * @param {*} response | ||||||
|  * @param {*} client |  | ||||||
|  */ |  */ | ||||||
| async function socket(request, response, client) { | exports.app_socket = async function socket(request, response) { | ||||||
| 	let process; | 	let process; | ||||||
| 	let options = {}; | 	let options = {}; | ||||||
| 	let credentials = await httpd.auth_query(request.headers); | 	let credentials = await httpd.auth_query(request.headers); | ||||||
| @@ -245,6 +244,6 @@ async function socket(request, response, client) { | |||||||
| 	}; | 	}; | ||||||
|  |  | ||||||
| 	response.upgrade(100, {}); | 	response.upgrade(100, {}); | ||||||
| } | }; | ||||||
|  |  | ||||||
| export {socket, App}; | export {App}; | ||||||
|   | |||||||
							
								
								
									
										57
									
								
								core/core.js
									
									
									
									
									
								
							
							
						
						| @@ -1,5 +1,4 @@ | |||||||
| import * as app from './app.js'; | import * as app from './app.js'; | ||||||
| import * as form from './form.js'; |  | ||||||
| import * as http from './http.js'; | import * as http from './http.js'; | ||||||
|  |  | ||||||
| let gProcesses = {}; | let gProcesses = {}; | ||||||
| @@ -471,7 +470,6 @@ async function getProcessBlob(blobId, key, options) { | |||||||
| 			imports.ssb = Object.fromEntries( | 			imports.ssb = Object.fromEntries( | ||||||
| 				Object.keys(ssb).map((key) => [key, ssb[key].bind(ssb)]) | 				Object.keys(ssb).map((key) => [key, ssb[key].bind(ssb)]) | ||||||
| 			); | 			); | ||||||
| 			imports.ssb.port = tildefriends.ssb_port; |  | ||||||
| 			imports.ssb.createIdentity = () => process.createIdentity(); | 			imports.ssb.createIdentity = () => process.createIdentity(); | ||||||
| 			imports.ssb.addIdentity = function (id) { | 			imports.ssb.addIdentity = function (id) { | ||||||
| 				if ( | 				if ( | ||||||
| @@ -835,59 +833,4 @@ exports.callAppHandler = async function callAppHandler( | |||||||
| 	response.end(answer?.data); | 	response.end(answer?.data); | ||||||
| }; | }; | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * TODOC |  | ||||||
|  */ |  | ||||||
| loadSettings() |  | ||||||
| 	.then(function (settings) { |  | ||||||
| 		if (tildefriends.https_port && settings.http_redirect) { |  | ||||||
| 			httpd.set_http_redirect(settings.http_redirect); |  | ||||||
| 		} |  | ||||||
| 		httpd.all('/app/socket', app.socket); |  | ||||||
| 		let port = httpd.start(tildefriends.http_port); |  | ||||||
| 		if (tildefriends.args.out_http_port_file) { |  | ||||||
| 			print('Writing the port file.'); |  | ||||||
| 			File.writeFile( |  | ||||||
| 				tildefriends.args.out_http_port_file, |  | ||||||
| 				port.toString() + '\n' |  | ||||||
| 			) |  | ||||||
| 				.then(function (r) { |  | ||||||
| 					print( |  | ||||||
| 						'Wrote the port file:', |  | ||||||
| 						tildefriends.args.out_http_port_file, |  | ||||||
| 						r |  | ||||||
| 					); |  | ||||||
| 				}) |  | ||||||
| 				.catch(function () { |  | ||||||
| 					print('Failed to write the port file.'); |  | ||||||
| 				}); |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		if (tildefriends.https_port) { |  | ||||||
| 			async function start_tls() { |  | ||||||
| 				const kCertificatePath = 'data/httpd/certificate.pem'; |  | ||||||
| 				const kPrivateKeyPath = 'data/httpd/privatekey.pem'; |  | ||||||
| 				let privateKey; |  | ||||||
| 				let certificate; |  | ||||||
| 				try { |  | ||||||
| 					privateKey = utf8Decode(await File.readFile(kPrivateKeyPath)); |  | ||||||
| 					certificate = utf8Decode(await File.readFile(kCertificatePath)); |  | ||||||
| 				} catch (e) { |  | ||||||
| 					print(`TLS disabled (${e.message}).`); |  | ||||||
| 					return; |  | ||||||
| 				} |  | ||||||
| 				let context = new TlsContext(); |  | ||||||
| 				context.setPrivateKey(privateKey); |  | ||||||
| 				context.setCertificate(certificate); |  | ||||||
| 				httpd.start(tildefriends.https_port, context); |  | ||||||
| 			} |  | ||||||
| 			start_tls(); |  | ||||||
| 		} |  | ||||||
| 	}) |  | ||||||
| 	.catch(function (error) { |  | ||||||
| 		print('Failed to load settings.'); |  | ||||||
| 		printError({print: print}, error); |  | ||||||
| 		exit(1); |  | ||||||
| 	}); |  | ||||||
|  |  | ||||||
| export {invoke, getProcessBlob}; | export {invoke, getProcessBlob}; | ||||||
|   | |||||||
							
								
								
									
										44
									
								
								core/form.js
									
									
									
									
									
								
							
							
						
						| @@ -1,44 +0,0 @@ | |||||||
| /** |  | ||||||
|  * TODOC |  | ||||||
|  * @param {*} encoded |  | ||||||
|  * @returns |  | ||||||
|  */ |  | ||||||
| function decode(encoded) { |  | ||||||
| 	let result = ''; |  | ||||||
| 	for (let i = 0; i < encoded.length; i++) { |  | ||||||
| 		let c = encoded[i]; |  | ||||||
| 		if (c == '+') { |  | ||||||
| 			result += ' '; |  | ||||||
| 		} else if (c == '%') { |  | ||||||
| 			result += String.fromCharCode(parseInt(encoded.slice(i + 1, i + 3), 16)); |  | ||||||
| 			i += 2; |  | ||||||
| 		} else { |  | ||||||
| 			result += c; |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
| 	return result; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * TODOC |  | ||||||
|  * @param {*} encoded |  | ||||||
|  * @param {*} initial |  | ||||||
|  * @returns |  | ||||||
|  */ |  | ||||||
| function decodeForm(encoded, initial) { |  | ||||||
| 	let result = initial || {}; |  | ||||||
| 	if (encoded) { |  | ||||||
| 		encoded = encoded.trim(); |  | ||||||
| 		let items = encoded.split('&'); |  | ||||||
| 		for (let i = 0; i < items.length; i++) { |  | ||||||
| 			let item = items[i]; |  | ||||||
| 			let equals = item.indexOf('='); |  | ||||||
| 			let key = decode(item.slice(0, equals)); |  | ||||||
| 			let value = decode(item.slice(equals + 1)); |  | ||||||
| 			result[key] = value; |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
| 	return result; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| export {decodeForm}; |  | ||||||
							
								
								
									
										15
									
								
								default.nix
									
									
									
									
									
								
							
							
						
						| @@ -1,4 +1,8 @@ | |||||||
| # How to upgrade to a newer version | # How to upgrade to a newer version | ||||||
|  | # - On the june and december release, you'll have to update nixpkgs to the current branch | ||||||
|  | # Change `nixpkgs.url = "github:NixOS/nixpkgs/nixos-24.11";` | ||||||
|  | # to the latest release (see https://nixos.org/) | ||||||
|  | # - Run `$ nix flake update` | ||||||
| # - Comment `src.hash` | # - Comment `src.hash` | ||||||
| # - Change `version` | # - Change `version` | ||||||
| # - Run `$ nix build` | # - Run `$ nix build` | ||||||
| @@ -21,31 +25,32 @@ | |||||||
| }: | }: | ||||||
| pkgs.stdenv.mkDerivation rec { | pkgs.stdenv.mkDerivation rec { | ||||||
|   pname = "tildefriends"; |   pname = "tildefriends"; | ||||||
|   version = "0.0.27"; |   version = "0.0.28"; | ||||||
|  |  | ||||||
|   src = pkgs.fetchFromGitea { |   src = pkgs.fetchFromGitea { | ||||||
|     domain = "dev.tildefriends.net"; |     domain = "dev.tildefriends.net"; | ||||||
|     owner = "cory"; |     owner = "cory"; | ||||||
|     repo = "tildefriends"; |     repo = "tildefriends"; | ||||||
|     rev = "v${version}"; |     rev = "f02423d0846fefd5ab21fa4542fb77ce5714547c"; | ||||||
|     hash = "sha256-NhoTBWYsWc206f1+M9haxHSJU6ZSGoP5nPStymAIyRk="; |     hash = "sha256-QyM7wmViXJc4r8uTu4oE/HO3Z9tzNbFIX2+AOTQz9ZY="; | ||||||
|     fetchSubmodules = true; |     fetchSubmodules = true; | ||||||
|   }; |   }; | ||||||
|  |  | ||||||
|   nativeBuildInputs = with pkgs; [ |   nativeBuildInputs = with pkgs; [ | ||||||
|     bash |  | ||||||
|     glibc |     glibc | ||||||
|     gnumake |     gnumake | ||||||
|  |     openssl | ||||||
|     which |     which | ||||||
|   ]; |   ]; | ||||||
|  |  | ||||||
|   buildInputs = with pkgs; [ |   buildInputs = with pkgs; [ | ||||||
|     glibc |     glibc | ||||||
|  |     openssl | ||||||
|     which |     which | ||||||
|   ]; |   ]; | ||||||
|  |  | ||||||
|   buildPhase = '' |   buildPhase = '' | ||||||
|     make -j $NIX_BUILD_CORES release |     make -j $NIX_BUILD_CORES release USE_SYSTEM_SSL=1 | ||||||
|   ''; |   ''; | ||||||
|  |  | ||||||
|   installPhase = '' |   installPhase = '' | ||||||
|   | |||||||
							
								
								
									
										2
									
								
								deps/codemirror/cm6.js
									
									
									
									
										vendored
									
									
								
							
							
						
						
							
								
								
									
										194
									
								
								deps/codemirror_src/package-lock.json
									
									
									
										generated
									
									
										vendored
									
									
								
							
							
						
						| @@ -19,9 +19,9 @@ | |||||||
|       } |       } | ||||||
|     }, |     }, | ||||||
|     "node_modules/@codemirror/autocomplete": { |     "node_modules/@codemirror/autocomplete": { | ||||||
|       "version": "6.18.4", |       "version": "6.18.6", | ||||||
|       "resolved": "https://registry.npmjs.org/@codemirror/autocomplete/-/autocomplete-6.18.4.tgz", |       "resolved": "https://registry.npmjs.org/@codemirror/autocomplete/-/autocomplete-6.18.6.tgz", | ||||||
|       "integrity": "sha512-sFAphGQIqyQZfP2ZBsSHV7xQvo9Py0rV0dW7W3IMRdS+zDuNb2l3no78CvUaWKGfzFjI4FTrLdUSj86IGb2hRA==", |       "integrity": "sha512-PHHBXFomUs5DF+9tCOM/UoW6XQ4R44lLNNhRaW9PKPTU0D7lIjRg3ElxaJnTwsl/oHiR93WSXDBrekhoUGCPtg==", | ||||||
|       "dependencies": { |       "dependencies": { | ||||||
|         "@codemirror/language": "^6.0.0", |         "@codemirror/language": "^6.0.0", | ||||||
|         "@codemirror/state": "^6.0.0", |         "@codemirror/state": "^6.0.0", | ||||||
| @@ -69,9 +69,9 @@ | |||||||
|       } |       } | ||||||
|     }, |     }, | ||||||
|     "node_modules/@codemirror/lang-javascript": { |     "node_modules/@codemirror/lang-javascript": { | ||||||
|       "version": "6.2.2", |       "version": "6.2.3", | ||||||
|       "resolved": "https://registry.npmjs.org/@codemirror/lang-javascript/-/lang-javascript-6.2.2.tgz", |       "resolved": "https://registry.npmjs.org/@codemirror/lang-javascript/-/lang-javascript-6.2.3.tgz", | ||||||
|       "integrity": "sha512-VGQfY+FCc285AhWuwjYxQyUQcYurWlxdKYT4bqwr3Twnd5wP5WSeu52t4tvvuWmljT4EmgEgZCqSieokhtY8hg==", |       "integrity": "sha512-8PR3vIWg7pSu7ur8A07pGiYHgy3hHj+mRYRCSG8q+mPIrl0F02rgpGv+DsQTHRTc30rydOsf5PZ7yjKFg2Ackw==", | ||||||
|       "dependencies": { |       "dependencies": { | ||||||
|         "@codemirror/autocomplete": "^6.0.0", |         "@codemirror/autocomplete": "^6.0.0", | ||||||
|         "@codemirror/language": "^6.6.0", |         "@codemirror/language": "^6.6.0", | ||||||
| @@ -115,9 +115,9 @@ | |||||||
|       } |       } | ||||||
|     }, |     }, | ||||||
|     "node_modules/@codemirror/search": { |     "node_modules/@codemirror/search": { | ||||||
|       "version": "6.5.8", |       "version": "6.5.9", | ||||||
|       "resolved": "https://registry.npmjs.org/@codemirror/search/-/search-6.5.8.tgz", |       "resolved": "https://registry.npmjs.org/@codemirror/search/-/search-6.5.9.tgz", | ||||||
|       "integrity": "sha512-PoWtZvo7c1XFeZWmmyaOp2G0XVbOnm+fJzvghqGAktBW3cufwJUWvSCcNG0ppXiBEM05mZu6RhMtXPv2hpllig==", |       "integrity": "sha512-7DdQ9aaZMMxuWB1u6IIFWWuK9NocVZwvo4nG8QjJTS6oZGvteoLSiXw3EbVZVlO08Ri2ltO89JVInMpfcJxhtg==", | ||||||
|       "dependencies": { |       "dependencies": { | ||||||
|         "@codemirror/state": "^6.0.0", |         "@codemirror/state": "^6.0.0", | ||||||
|         "@codemirror/view": "^6.0.0", |         "@codemirror/view": "^6.0.0", | ||||||
| @@ -125,9 +125,9 @@ | |||||||
|       } |       } | ||||||
|     }, |     }, | ||||||
|     "node_modules/@codemirror/state": { |     "node_modules/@codemirror/state": { | ||||||
|       "version": "6.5.1", |       "version": "6.5.2", | ||||||
|       "resolved": "https://registry.npmjs.org/@codemirror/state/-/state-6.5.1.tgz", |       "resolved": "https://registry.npmjs.org/@codemirror/state/-/state-6.5.2.tgz", | ||||||
|       "integrity": "sha512-3rA9lcwciEB47ZevqvD8qgbzhM9qMb8vCcQCNmDfVRPQG4JT9mSb0Jg8H7YjKGGQcFnLN323fj9jdnG59Kx6bg==", |       "integrity": "sha512-FVqsPqtPWKVVL3dPSxy8wEF/ymIEuVzF1PK3VbUgrxXpJUSHQWWZz4JMToquRxnkw+36LTamCZG2iua2Ptq0fA==", | ||||||
|       "dependencies": { |       "dependencies": { | ||||||
|         "@marijn/find-cluster-break": "^1.0.0" |         "@marijn/find-cluster-break": "^1.0.0" | ||||||
|       } |       } | ||||||
| @@ -144,9 +144,9 @@ | |||||||
|       } |       } | ||||||
|     }, |     }, | ||||||
|     "node_modules/@codemirror/view": { |     "node_modules/@codemirror/view": { | ||||||
|       "version": "6.36.2", |       "version": "6.36.3", | ||||||
|       "resolved": "https://registry.npmjs.org/@codemirror/view/-/view-6.36.2.tgz", |       "resolved": "https://registry.npmjs.org/@codemirror/view/-/view-6.36.3.tgz", | ||||||
|       "integrity": "sha512-DZ6ONbs8qdJK0fdN7AB82CgI6tYXf4HWk1wSVa0+9bhVznCuuvhQtX8bFBoy3dv8rZSQqUd8GvhVAcielcidrA==", |       "integrity": "sha512-N2bilM47QWC8Hnx0rMdDxO2x2ImJ1FvZWXubwKgjeoOrWwEiFrtpA7SFHcuZ+o2Ze2VzbkgbzWVj4+V18LVkeg==", | ||||||
|       "dependencies": { |       "dependencies": { | ||||||
|         "@codemirror/state": "^6.5.0", |         "@codemirror/state": "^6.5.0", | ||||||
|         "style-mod": "^4.1.0", |         "style-mod": "^4.1.0", | ||||||
| @@ -344,9 +344,9 @@ | |||||||
|       } |       } | ||||||
|     }, |     }, | ||||||
|     "node_modules/@rollup/rollup-android-arm-eabi": { |     "node_modules/@rollup/rollup-android-arm-eabi": { | ||||||
|       "version": "4.32.1", |       "version": "4.34.8", | ||||||
|       "resolved": "https://registry.npmjs.org/@rollup/rollup-android-arm-eabi/-/rollup-android-arm-eabi-4.32.1.tgz", |       "resolved": "https://registry.npmjs.org/@rollup/rollup-android-arm-eabi/-/rollup-android-arm-eabi-4.34.8.tgz", | ||||||
|       "integrity": "sha512-/pqA4DmqyCm8u5YIDzIdlLcEmuvxb0v8fZdFhVMszSpDTgbQKdw3/mB3eMUHIbubtJ6F9j+LtmyCnHTEqIHyzA==", |       "integrity": "sha512-q217OSE8DTp8AFHuNHXo0Y86e1wtlfVrXiAlwkIvGRQv9zbc6mE3sjIVfwI8sYUyNxwOg0j/Vm1RKM04JcWLJw==", | ||||||
|       "cpu": [ |       "cpu": [ | ||||||
|         "arm" |         "arm" | ||||||
|       ], |       ], | ||||||
| @@ -356,9 +356,9 @@ | |||||||
|       ] |       ] | ||||||
|     }, |     }, | ||||||
|     "node_modules/@rollup/rollup-android-arm64": { |     "node_modules/@rollup/rollup-android-arm64": { | ||||||
|       "version": "4.32.1", |       "version": "4.34.8", | ||||||
|       "resolved": "https://registry.npmjs.org/@rollup/rollup-android-arm64/-/rollup-android-arm64-4.32.1.tgz", |       "resolved": "https://registry.npmjs.org/@rollup/rollup-android-arm64/-/rollup-android-arm64-4.34.8.tgz", | ||||||
|       "integrity": "sha512-If3PDskT77q7zgqVqYuj7WG3WC08G1kwXGVFi9Jr8nY6eHucREHkfpX79c0ACAjLj3QIWKPJR7w4i+f5EdLH5Q==", |       "integrity": "sha512-Gigjz7mNWaOL9wCggvoK3jEIUUbGul656opstjaUSGC3eT0BM7PofdAJaBfPFWWkXNVAXbaQtC99OCg4sJv70Q==", | ||||||
|       "cpu": [ |       "cpu": [ | ||||||
|         "arm64" |         "arm64" | ||||||
|       ], |       ], | ||||||
| @@ -368,9 +368,9 @@ | |||||||
|       ] |       ] | ||||||
|     }, |     }, | ||||||
|     "node_modules/@rollup/rollup-darwin-arm64": { |     "node_modules/@rollup/rollup-darwin-arm64": { | ||||||
|       "version": "4.32.1", |       "version": "4.34.8", | ||||||
|       "resolved": "https://registry.npmjs.org/@rollup/rollup-darwin-arm64/-/rollup-darwin-arm64-4.32.1.tgz", |       "resolved": "https://registry.npmjs.org/@rollup/rollup-darwin-arm64/-/rollup-darwin-arm64-4.34.8.tgz", | ||||||
|       "integrity": "sha512-zCpKHioQ9KgZToFp5Wvz6zaWbMzYQ2LJHQ+QixDKq52KKrF65ueu6Af4hLlLWHjX1Wf/0G5kSJM9PySW9IrvHA==", |       "integrity": "sha512-02rVdZ5tgdUNRxIUrFdcMBZQoaPMrxtwSb+/hOfBdqkatYHR3lZ2A2EGyHq2sGOd0Owk80oV3snlDASC24He3Q==", | ||||||
|       "cpu": [ |       "cpu": [ | ||||||
|         "arm64" |         "arm64" | ||||||
|       ], |       ], | ||||||
| @@ -380,9 +380,9 @@ | |||||||
|       ] |       ] | ||||||
|     }, |     }, | ||||||
|     "node_modules/@rollup/rollup-darwin-x64": { |     "node_modules/@rollup/rollup-darwin-x64": { | ||||||
|       "version": "4.32.1", |       "version": "4.34.8", | ||||||
|       "resolved": "https://registry.npmjs.org/@rollup/rollup-darwin-x64/-/rollup-darwin-x64-4.32.1.tgz", |       "resolved": "https://registry.npmjs.org/@rollup/rollup-darwin-x64/-/rollup-darwin-x64-4.34.8.tgz", | ||||||
|       "integrity": "sha512-sFvF+t2+TyUo/ZQqUcifrJIgznx58oFZbdHS9TvHq3xhPVL9nOp+yZ6LKrO9GWTP+6DbFtoyLDbjTpR62Mbr3Q==", |       "integrity": "sha512-qIP/elwR/tq/dYRx3lgwK31jkZvMiD6qUtOycLhTzCvrjbZ3LjQnEM9rNhSGpbLXVJYQ3rq39A6Re0h9tU2ynw==", | ||||||
|       "cpu": [ |       "cpu": [ | ||||||
|         "x64" |         "x64" | ||||||
|       ], |       ], | ||||||
| @@ -392,9 +392,9 @@ | |||||||
|       ] |       ] | ||||||
|     }, |     }, | ||||||
|     "node_modules/@rollup/rollup-freebsd-arm64": { |     "node_modules/@rollup/rollup-freebsd-arm64": { | ||||||
|       "version": "4.32.1", |       "version": "4.34.8", | ||||||
|       "resolved": "https://registry.npmjs.org/@rollup/rollup-freebsd-arm64/-/rollup-freebsd-arm64-4.32.1.tgz", |       "resolved": "https://registry.npmjs.org/@rollup/rollup-freebsd-arm64/-/rollup-freebsd-arm64-4.34.8.tgz", | ||||||
|       "integrity": "sha512-NbOa+7InvMWRcY9RG+B6kKIMD/FsnQPH0MWUvDlQB1iXnF/UcKSudCXZtv4lW+C276g3w5AxPbfry5rSYvyeYA==", |       "integrity": "sha512-IQNVXL9iY6NniYbTaOKdrlVP3XIqazBgJOVkddzJlqnCpRi/yAeSOa8PLcECFSQochzqApIOE1GHNu3pCz+BDA==", | ||||||
|       "cpu": [ |       "cpu": [ | ||||||
|         "arm64" |         "arm64" | ||||||
|       ], |       ], | ||||||
| @@ -404,9 +404,9 @@ | |||||||
|       ] |       ] | ||||||
|     }, |     }, | ||||||
|     "node_modules/@rollup/rollup-freebsd-x64": { |     "node_modules/@rollup/rollup-freebsd-x64": { | ||||||
|       "version": "4.32.1", |       "version": "4.34.8", | ||||||
|       "resolved": "https://registry.npmjs.org/@rollup/rollup-freebsd-x64/-/rollup-freebsd-x64-4.32.1.tgz", |       "resolved": "https://registry.npmjs.org/@rollup/rollup-freebsd-x64/-/rollup-freebsd-x64-4.34.8.tgz", | ||||||
|       "integrity": "sha512-JRBRmwvHPXR881j2xjry8HZ86wIPK2CcDw0EXchE1UgU0ubWp9nvlT7cZYKc6bkypBt745b4bglf3+xJ7hXWWw==", |       "integrity": "sha512-TYXcHghgnCqYFiE3FT5QwXtOZqDj5GmaFNTNt3jNC+vh22dc/ukG2cG+pi75QO4kACohZzidsq7yKTKwq/Jq7Q==", | ||||||
|       "cpu": [ |       "cpu": [ | ||||||
|         "x64" |         "x64" | ||||||
|       ], |       ], | ||||||
| @@ -416,9 +416,9 @@ | |||||||
|       ] |       ] | ||||||
|     }, |     }, | ||||||
|     "node_modules/@rollup/rollup-linux-arm-gnueabihf": { |     "node_modules/@rollup/rollup-linux-arm-gnueabihf": { | ||||||
|       "version": "4.32.1", |       "version": "4.34.8", | ||||||
|       "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm-gnueabihf/-/rollup-linux-arm-gnueabihf-4.32.1.tgz", |       "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm-gnueabihf/-/rollup-linux-arm-gnueabihf-4.34.8.tgz", | ||||||
|       "integrity": "sha512-PKvszb+9o/vVdUzCCjL0sKHukEQV39tD3fepXxYrHE3sTKrRdCydI7uldRLbjLmDA3TFDmh418XH19NOsDRH8g==", |       "integrity": "sha512-A4iphFGNkWRd+5m3VIGuqHnG3MVnqKe7Al57u9mwgbyZ2/xF9Jio72MaY7xxh+Y87VAHmGQr73qoKL9HPbXj1g==", | ||||||
|       "cpu": [ |       "cpu": [ | ||||||
|         "arm" |         "arm" | ||||||
|       ], |       ], | ||||||
| @@ -428,9 +428,9 @@ | |||||||
|       ] |       ] | ||||||
|     }, |     }, | ||||||
|     "node_modules/@rollup/rollup-linux-arm-musleabihf": { |     "node_modules/@rollup/rollup-linux-arm-musleabihf": { | ||||||
|       "version": "4.32.1", |       "version": "4.34.8", | ||||||
|       "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm-musleabihf/-/rollup-linux-arm-musleabihf-4.32.1.tgz", |       "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm-musleabihf/-/rollup-linux-arm-musleabihf-4.34.8.tgz", | ||||||
|       "integrity": "sha512-9WHEMV6Y89eL606ReYowXuGF1Yb2vwfKWKdD1A5h+OYnPZSJvxbEjxTRKPgi7tkP2DSnW0YLab1ooy+i/FQp/Q==", |       "integrity": "sha512-S0lqKLfTm5u+QTxlFiAnb2J/2dgQqRy/XvziPtDd1rKZFXHTyYLoVL58M/XFwDI01AQCDIevGLbQrMAtdyanpA==", | ||||||
|       "cpu": [ |       "cpu": [ | ||||||
|         "arm" |         "arm" | ||||||
|       ], |       ], | ||||||
| @@ -440,9 +440,9 @@ | |||||||
|       ] |       ] | ||||||
|     }, |     }, | ||||||
|     "node_modules/@rollup/rollup-linux-arm64-gnu": { |     "node_modules/@rollup/rollup-linux-arm64-gnu": { | ||||||
|       "version": "4.32.1", |       "version": "4.34.8", | ||||||
|       "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm64-gnu/-/rollup-linux-arm64-gnu-4.32.1.tgz", |       "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm64-gnu/-/rollup-linux-arm64-gnu-4.34.8.tgz", | ||||||
|       "integrity": "sha512-tZWc9iEt5fGJ1CL2LRPw8OttkCBDs+D8D3oEM8mH8S1ICZCtFJhD7DZ3XMGM8kpqHvhGUTvNUYVDnmkj4BDXnw==", |       "integrity": "sha512-jpz9YOuPiSkL4G4pqKrus0pn9aYwpImGkosRKwNi+sJSkz+WU3anZe6hi73StLOQdfXYXC7hUfsQlTnjMd3s1A==", | ||||||
|       "cpu": [ |       "cpu": [ | ||||||
|         "arm64" |         "arm64" | ||||||
|       ], |       ], | ||||||
| @@ -452,9 +452,9 @@ | |||||||
|       ] |       ] | ||||||
|     }, |     }, | ||||||
|     "node_modules/@rollup/rollup-linux-arm64-musl": { |     "node_modules/@rollup/rollup-linux-arm64-musl": { | ||||||
|       "version": "4.32.1", |       "version": "4.34.8", | ||||||
|       "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm64-musl/-/rollup-linux-arm64-musl-4.32.1.tgz", |       "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm64-musl/-/rollup-linux-arm64-musl-4.34.8.tgz", | ||||||
|       "integrity": "sha512-FTYc2YoTWUsBz5GTTgGkRYYJ5NGJIi/rCY4oK/I8aKowx1ToXeoVVbIE4LGAjsauvlhjfl0MYacxClLld1VrOw==", |       "integrity": "sha512-KdSfaROOUJXgTVxJNAZ3KwkRc5nggDk+06P6lgi1HLv1hskgvxHUKZ4xtwHkVYJ1Rep4GNo+uEfycCRRxht7+Q==", | ||||||
|       "cpu": [ |       "cpu": [ | ||||||
|         "arm64" |         "arm64" | ||||||
|       ], |       ], | ||||||
| @@ -464,9 +464,9 @@ | |||||||
|       ] |       ] | ||||||
|     }, |     }, | ||||||
|     "node_modules/@rollup/rollup-linux-loongarch64-gnu": { |     "node_modules/@rollup/rollup-linux-loongarch64-gnu": { | ||||||
|       "version": "4.32.1", |       "version": "4.34.8", | ||||||
|       "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-loongarch64-gnu/-/rollup-linux-loongarch64-gnu-4.32.1.tgz", |       "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-loongarch64-gnu/-/rollup-linux-loongarch64-gnu-4.34.8.tgz", | ||||||
|       "integrity": "sha512-F51qLdOtpS6P1zJVRzYM0v6MrBNypyPEN1GfMiz0gPu9jN8ScGaEFIZQwteSsGKg799oR5EaP7+B2jHgL+d+Kw==", |       "integrity": "sha512-NyF4gcxwkMFRjgXBM6g2lkT58OWztZvw5KkV2K0qqSnUEqCVcqdh2jN4gQrTn/YUpAcNKyFHfoOZEer9nwo6uQ==", | ||||||
|       "cpu": [ |       "cpu": [ | ||||||
|         "loong64" |         "loong64" | ||||||
|       ], |       ], | ||||||
| @@ -476,9 +476,9 @@ | |||||||
|       ] |       ] | ||||||
|     }, |     }, | ||||||
|     "node_modules/@rollup/rollup-linux-powerpc64le-gnu": { |     "node_modules/@rollup/rollup-linux-powerpc64le-gnu": { | ||||||
|       "version": "4.32.1", |       "version": "4.34.8", | ||||||
|       "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-powerpc64le-gnu/-/rollup-linux-powerpc64le-gnu-4.32.1.tgz", |       "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-powerpc64le-gnu/-/rollup-linux-powerpc64le-gnu-4.34.8.tgz", | ||||||
|       "integrity": "sha512-wO0WkfSppfX4YFm5KhdCCpnpGbtgQNj/tgvYzrVYFKDpven8w2N6Gg5nB6w+wAMO3AIfSTWeTjfVe+uZ23zAlg==", |       "integrity": "sha512-LMJc999GkhGvktHU85zNTDImZVUCJ1z/MbAJTnviiWmmjyckP5aQsHtcujMjpNdMZPT2rQEDBlJfubhs3jsMfw==", | ||||||
|       "cpu": [ |       "cpu": [ | ||||||
|         "ppc64" |         "ppc64" | ||||||
|       ], |       ], | ||||||
| @@ -488,9 +488,9 @@ | |||||||
|       ] |       ] | ||||||
|     }, |     }, | ||||||
|     "node_modules/@rollup/rollup-linux-riscv64-gnu": { |     "node_modules/@rollup/rollup-linux-riscv64-gnu": { | ||||||
|       "version": "4.32.1", |       "version": "4.34.8", | ||||||
|       "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-riscv64-gnu/-/rollup-linux-riscv64-gnu-4.32.1.tgz", |       "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-riscv64-gnu/-/rollup-linux-riscv64-gnu-4.34.8.tgz", | ||||||
|       "integrity": "sha512-iWswS9cIXfJO1MFYtI/4jjlrGb/V58oMu4dYJIKnR5UIwbkzR0PJ09O0PDZT0oJ3LYWXBSWahNf/Mjo6i1E5/g==", |       "integrity": "sha512-xAQCAHPj8nJq1PI3z8CIZzXuXCstquz7cIOL73HHdXiRcKk8Ywwqtx2wrIy23EcTn4aZ2fLJNBB8d0tQENPCmw==", | ||||||
|       "cpu": [ |       "cpu": [ | ||||||
|         "riscv64" |         "riscv64" | ||||||
|       ], |       ], | ||||||
| @@ -500,9 +500,9 @@ | |||||||
|       ] |       ] | ||||||
|     }, |     }, | ||||||
|     "node_modules/@rollup/rollup-linux-s390x-gnu": { |     "node_modules/@rollup/rollup-linux-s390x-gnu": { | ||||||
|       "version": "4.32.1", |       "version": "4.34.8", | ||||||
|       "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-s390x-gnu/-/rollup-linux-s390x-gnu-4.32.1.tgz", |       "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-s390x-gnu/-/rollup-linux-s390x-gnu-4.34.8.tgz", | ||||||
|       "integrity": "sha512-RKt8NI9tebzmEthMnfVgG3i/XeECkMPS+ibVZjZ6mNekpbbUmkNWuIN2yHsb/mBPyZke4nlI4YqIdFPgKuoyQQ==", |       "integrity": "sha512-DdePVk1NDEuc3fOe3dPPTb+rjMtuFw89gw6gVWxQFAuEqqSdDKnrwzZHrUYdac7A7dXl9Q2Vflxpme15gUWQFA==", | ||||||
|       "cpu": [ |       "cpu": [ | ||||||
|         "s390x" |         "s390x" | ||||||
|       ], |       ], | ||||||
| @@ -512,9 +512,9 @@ | |||||||
|       ] |       ] | ||||||
|     }, |     }, | ||||||
|     "node_modules/@rollup/rollup-linux-x64-gnu": { |     "node_modules/@rollup/rollup-linux-x64-gnu": { | ||||||
|       "version": "4.32.1", |       "version": "4.34.8", | ||||||
|       "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-x64-gnu/-/rollup-linux-x64-gnu-4.32.1.tgz", |       "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-x64-gnu/-/rollup-linux-x64-gnu-4.34.8.tgz", | ||||||
|       "integrity": "sha512-WQFLZ9c42ECqEjwg/GHHsouij3pzLXkFdz0UxHa/0OM12LzvX7DzedlY0SIEly2v18YZLRhCRoHZDxbBSWoGYg==", |       "integrity": "sha512-8y7ED8gjxITUltTUEJLQdgpbPh1sUQ0kMTmufRF/Ns5tI9TNMNlhWtmPKKHCU0SilX+3MJkZ0zERYYGIVBYHIA==", | ||||||
|       "cpu": [ |       "cpu": [ | ||||||
|         "x64" |         "x64" | ||||||
|       ], |       ], | ||||||
| @@ -524,9 +524,9 @@ | |||||||
|       ] |       ] | ||||||
|     }, |     }, | ||||||
|     "node_modules/@rollup/rollup-linux-x64-musl": { |     "node_modules/@rollup/rollup-linux-x64-musl": { | ||||||
|       "version": "4.32.1", |       "version": "4.34.8", | ||||||
|       "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-x64-musl/-/rollup-linux-x64-musl-4.32.1.tgz", |       "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-x64-musl/-/rollup-linux-x64-musl-4.34.8.tgz", | ||||||
|       "integrity": "sha512-BLoiyHDOWoS3uccNSADMza6V6vCNiphi94tQlVIL5de+r6r/CCQuNnerf+1g2mnk2b6edp5dk0nhdZ7aEjOBsA==", |       "integrity": "sha512-SCXcP0ZpGFIe7Ge+McxY5zKxiEI5ra+GT3QRxL0pMMtxPfpyLAKleZODi1zdRHkz5/BhueUrYtYVgubqe9JBNQ==", | ||||||
|       "cpu": [ |       "cpu": [ | ||||||
|         "x64" |         "x64" | ||||||
|       ], |       ], | ||||||
| @@ -536,9 +536,9 @@ | |||||||
|       ] |       ] | ||||||
|     }, |     }, | ||||||
|     "node_modules/@rollup/rollup-win32-arm64-msvc": { |     "node_modules/@rollup/rollup-win32-arm64-msvc": { | ||||||
|       "version": "4.32.1", |       "version": "4.34.8", | ||||||
|       "resolved": "https://registry.npmjs.org/@rollup/rollup-win32-arm64-msvc/-/rollup-win32-arm64-msvc-4.32.1.tgz", |       "resolved": "https://registry.npmjs.org/@rollup/rollup-win32-arm64-msvc/-/rollup-win32-arm64-msvc-4.34.8.tgz", | ||||||
|       "integrity": "sha512-w2l3UnlgYTNNU+Z6wOR8YdaioqfEnwPjIsJ66KxKAf0p+AuL2FHeTX6qvM+p/Ue3XPBVNyVSfCrfZiQh7vZHLQ==", |       "integrity": "sha512-YHYsgzZgFJzTRbth4h7Or0m5O74Yda+hLin0irAIobkLQFRQd1qWmnoVfwmKm9TXIZVAD0nZ+GEb2ICicLyCnQ==", | ||||||
|       "cpu": [ |       "cpu": [ | ||||||
|         "arm64" |         "arm64" | ||||||
|       ], |       ], | ||||||
| @@ -548,9 +548,9 @@ | |||||||
|       ] |       ] | ||||||
|     }, |     }, | ||||||
|     "node_modules/@rollup/rollup-win32-ia32-msvc": { |     "node_modules/@rollup/rollup-win32-ia32-msvc": { | ||||||
|       "version": "4.32.1", |       "version": "4.34.8", | ||||||
|       "resolved": "https://registry.npmjs.org/@rollup/rollup-win32-ia32-msvc/-/rollup-win32-ia32-msvc-4.32.1.tgz", |       "resolved": "https://registry.npmjs.org/@rollup/rollup-win32-ia32-msvc/-/rollup-win32-ia32-msvc-4.34.8.tgz", | ||||||
|       "integrity": "sha512-Am9H+TGLomPGkBnaPWie4F3x+yQ2rr4Bk2jpwy+iV+Gel9jLAu/KqT8k3X4jxFPW6Zf8OMnehyutsd+eHoq1WQ==", |       "integrity": "sha512-r3NRQrXkHr4uWy5TOjTpTYojR9XmF0j/RYgKCef+Ag46FWUTltm5ziticv8LdNsDMehjJ543x/+TJAek/xBA2w==", | ||||||
|       "cpu": [ |       "cpu": [ | ||||||
|         "ia32" |         "ia32" | ||||||
|       ], |       ], | ||||||
| @@ -560,9 +560,9 @@ | |||||||
|       ] |       ] | ||||||
|     }, |     }, | ||||||
|     "node_modules/@rollup/rollup-win32-x64-msvc": { |     "node_modules/@rollup/rollup-win32-x64-msvc": { | ||||||
|       "version": "4.32.1", |       "version": "4.34.8", | ||||||
|       "resolved": "https://registry.npmjs.org/@rollup/rollup-win32-x64-msvc/-/rollup-win32-x64-msvc-4.32.1.tgz", |       "resolved": "https://registry.npmjs.org/@rollup/rollup-win32-x64-msvc/-/rollup-win32-x64-msvc-4.34.8.tgz", | ||||||
|       "integrity": "sha512-ar80GhdZb4DgmW3myIS9nRFYcpJRSME8iqWgzH2i44u+IdrzmiXVxeFnExQ5v4JYUSpg94bWjevMG8JHf1Da5Q==", |       "integrity": "sha512-U0FaE5O1BCpZSeE6gBl3c5ObhePQSfk9vDRToMmTkbhCOgW4jqvtS5LGyQ76L1fH8sM0keRp4uDTsbjiUyjk0g==", | ||||||
|       "cpu": [ |       "cpu": [ | ||||||
|         "x64" |         "x64" | ||||||
|       ], |       ], | ||||||
| @@ -733,9 +733,9 @@ | |||||||
|       } |       } | ||||||
|     }, |     }, | ||||||
|     "node_modules/rollup": { |     "node_modules/rollup": { | ||||||
|       "version": "4.32.1", |       "version": "4.34.8", | ||||||
|       "resolved": "https://registry.npmjs.org/rollup/-/rollup-4.32.1.tgz", |       "resolved": "https://registry.npmjs.org/rollup/-/rollup-4.34.8.tgz", | ||||||
|       "integrity": "sha512-z+aeEsOeEa3mEbS1Tjl6sAZ8NE3+AalQz1RJGj81M+fizusbdDMoEJwdJNHfaB40Scr4qNu+welOfes7maKonA==", |       "integrity": "sha512-489gTVMzAYdiZHFVA/ig/iYFllCcWFHMvUHI1rpFmkoUtRlQxqh6/yiNqnYibjMZ2b/+FUQwldG+aLsEt6bglQ==", | ||||||
|       "dependencies": { |       "dependencies": { | ||||||
|         "@types/estree": "1.0.6" |         "@types/estree": "1.0.6" | ||||||
|       }, |       }, | ||||||
| @@ -747,25 +747,25 @@ | |||||||
|         "npm": ">=8.0.0" |         "npm": ">=8.0.0" | ||||||
|       }, |       }, | ||||||
|       "optionalDependencies": { |       "optionalDependencies": { | ||||||
|         "@rollup/rollup-android-arm-eabi": "4.32.1", |         "@rollup/rollup-android-arm-eabi": "4.34.8", | ||||||
|         "@rollup/rollup-android-arm64": "4.32.1", |         "@rollup/rollup-android-arm64": "4.34.8", | ||||||
|         "@rollup/rollup-darwin-arm64": "4.32.1", |         "@rollup/rollup-darwin-arm64": "4.34.8", | ||||||
|         "@rollup/rollup-darwin-x64": "4.32.1", |         "@rollup/rollup-darwin-x64": "4.34.8", | ||||||
|         "@rollup/rollup-freebsd-arm64": "4.32.1", |         "@rollup/rollup-freebsd-arm64": "4.34.8", | ||||||
|         "@rollup/rollup-freebsd-x64": "4.32.1", |         "@rollup/rollup-freebsd-x64": "4.34.8", | ||||||
|         "@rollup/rollup-linux-arm-gnueabihf": "4.32.1", |         "@rollup/rollup-linux-arm-gnueabihf": "4.34.8", | ||||||
|         "@rollup/rollup-linux-arm-musleabihf": "4.32.1", |         "@rollup/rollup-linux-arm-musleabihf": "4.34.8", | ||||||
|         "@rollup/rollup-linux-arm64-gnu": "4.32.1", |         "@rollup/rollup-linux-arm64-gnu": "4.34.8", | ||||||
|         "@rollup/rollup-linux-arm64-musl": "4.32.1", |         "@rollup/rollup-linux-arm64-musl": "4.34.8", | ||||||
|         "@rollup/rollup-linux-loongarch64-gnu": "4.32.1", |         "@rollup/rollup-linux-loongarch64-gnu": "4.34.8", | ||||||
|         "@rollup/rollup-linux-powerpc64le-gnu": "4.32.1", |         "@rollup/rollup-linux-powerpc64le-gnu": "4.34.8", | ||||||
|         "@rollup/rollup-linux-riscv64-gnu": "4.32.1", |         "@rollup/rollup-linux-riscv64-gnu": "4.34.8", | ||||||
|         "@rollup/rollup-linux-s390x-gnu": "4.32.1", |         "@rollup/rollup-linux-s390x-gnu": "4.34.8", | ||||||
|         "@rollup/rollup-linux-x64-gnu": "4.32.1", |         "@rollup/rollup-linux-x64-gnu": "4.34.8", | ||||||
|         "@rollup/rollup-linux-x64-musl": "4.32.1", |         "@rollup/rollup-linux-x64-musl": "4.34.8", | ||||||
|         "@rollup/rollup-win32-arm64-msvc": "4.32.1", |         "@rollup/rollup-win32-arm64-msvc": "4.34.8", | ||||||
|         "@rollup/rollup-win32-ia32-msvc": "4.32.1", |         "@rollup/rollup-win32-ia32-msvc": "4.34.8", | ||||||
|         "@rollup/rollup-win32-x64-msvc": "4.32.1", |         "@rollup/rollup-win32-x64-msvc": "4.34.8", | ||||||
|         "fsevents": "~2.3.2" |         "fsevents": "~2.3.2" | ||||||
|       } |       } | ||||||
|     }, |     }, | ||||||
| @@ -840,9 +840,9 @@ | |||||||
|       } |       } | ||||||
|     }, |     }, | ||||||
|     "node_modules/terser": { |     "node_modules/terser": { | ||||||
|       "version": "5.37.0", |       "version": "5.39.0", | ||||||
|       "resolved": "https://registry.npmjs.org/terser/-/terser-5.37.0.tgz", |       "resolved": "https://registry.npmjs.org/terser/-/terser-5.39.0.tgz", | ||||||
|       "integrity": "sha512-B8wRRkmre4ERucLM/uXx4MOV5cbnOlVAqUst+1+iLKPI0dOgFO28f84ptoQt9HEI537PMzfYa/d+GEPKTRXmYA==", |       "integrity": "sha512-LBAhFyLho16harJoWMg/nZsQYgTrg5jXOn2nCYjRUcZZEdE3qa2zb8QEDRUGVZBW4rlazf2fxkg8tztybTaqWw==", | ||||||
|       "dev": true, |       "dev": true, | ||||||
|       "dependencies": { |       "dependencies": { | ||||||
|         "@jridgewell/source-map": "^0.3.3", |         "@jridgewell/source-map": "^0.3.3", | ||||||
|   | |||||||
							
								
								
									
										2
									
								
								deps/libbacktrace
									
									
									
									
										vendored
									
									
								
							
							
								
								
								
								
								
							
						
						
							
								
								
									
										2
									
								
								deps/openssl_src
									
									
									
									
										vendored
									
									
								
							
							
								
								
								
								
								
							
						
						
							
								
								
									
										2
									
								
								deps/speedscope/index.html
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -11,7 +11,7 @@ | |||||||
|     <link rel="icon" type="image/x-icon" href="favicon-FOKUP5Y5.ico"> |     <link rel="icon" type="image/x-icon" href="favicon-FOKUP5Y5.ico"> | ||||||
|   </head> |   </head> | ||||||
|   <body> |   <body> | ||||||
|     <script src="speedscope-CAEVGCWN.js"></script> |     <script src="speedscope-VHEG2FVF.js"></script> | ||||||
|      |      | ||||||
|      |      | ||||||
|      |      | ||||||
|   | |||||||
							
								
								
									
										6
									
								
								deps/speedscope/release.txt
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -1,3 +1,3 @@ | |||||||
| speedscope@1.22.0 | speedscope@1.22.2 | ||||||
| Thu Jan 16 16:49:47 PST 2025 | Sat Feb 15 13:02:38 PST 2025 | ||||||
| bd7b44a0a7d63375ee6ea0a0d1b96e65a456642f | 1c254dcb3e2b4f6d921340d20e972d9d27b788f4 | ||||||
|   | |||||||
							
								
								
									
										24
									
								
								deps/sqlite/shell.c
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -357,6 +357,11 @@ void sqlite3_fsetmode(FILE *stream, int mode); | |||||||
| ** use O_U8TEXT when writing to the Windows console (or anything | ** use O_U8TEXT when writing to the Windows console (or anything | ||||||
| ** else for which _isatty() returns true) and to use O_BINARY or O_TEXT | ** else for which _isatty() returns true) and to use O_BINARY or O_TEXT | ||||||
| ** for all other output channels. | ** for all other output channels. | ||||||
|  | ** | ||||||
|  | ** The SQLITE_USE_W32_FOR_CONSOLE_IO macro is also available.  If | ||||||
|  | ** defined, it forces the use of Win32 APIs for all console I/O, both | ||||||
|  | ** input and output.  This is necessary for some non-Microsoft run-times | ||||||
|  | ** that implement stdio differently from Microsoft/Visual-Studio. | ||||||
| */ | */ | ||||||
| #if defined(SQLITE_U8TEXT_ONLY) | #if defined(SQLITE_U8TEXT_ONLY) | ||||||
| # define UseWtextForOutput(fd) 1 | # define UseWtextForOutput(fd) 1 | ||||||
| @@ -459,10 +464,10 @@ char *sqlite3_fgets(char *buf, int sz, FILE *in){ | |||||||
|     */ |     */ | ||||||
|     wchar_t *b1 = sqlite3_malloc( sz*sizeof(wchar_t) ); |     wchar_t *b1 = sqlite3_malloc( sz*sizeof(wchar_t) ); | ||||||
|     if( b1==0 ) return 0; |     if( b1==0 ) return 0; | ||||||
| #ifndef SQLITE_USE_STDIO_FOR_CONSOLE | #ifdef SQLITE_USE_W32_FOR_CONSOLE_IO | ||||||
|     DWORD nRead = 0; |     DWORD nRead = 0; | ||||||
|     if( IsConsole(in) |     if( IsConsole(in) | ||||||
|      && ReadConsoleW(GetStdHandle(STD_INPUT_HANDLE), b1, sz, &nRead, 0) |      && ReadConsoleW(GetStdHandle(STD_INPUT_HANDLE), b1, sz-1, &nRead, 0) | ||||||
|     ){ |     ){ | ||||||
|       b1[nRead] = 0; |       b1[nRead] = 0; | ||||||
|     }else |     }else | ||||||
| @@ -537,7 +542,7 @@ int sqlite3_fputs(const char *z, FILE *out){ | |||||||
|     sz = MultiByteToWideChar(CP_UTF8, 0, z, sz, b1, sz); |     sz = MultiByteToWideChar(CP_UTF8, 0, z, sz, b1, sz); | ||||||
|     b1[sz] = 0; |     b1[sz] = 0; | ||||||
| 
 | 
 | ||||||
| #ifndef SQLITE_STDIO_FOR_CONSOLE | #ifdef SQLITE_USE_W32_FOR_CONSOLE_IO | ||||||
|     DWORD nWr = 0; |     DWORD nWr = 0; | ||||||
|     if( IsConsole(out) |     if( IsConsole(out) | ||||||
|       && WriteConsoleW(GetStdHandle(STD_OUTPUT_HANDLE),b1,sz,&nWr,0) |       && WriteConsoleW(GetStdHandle(STD_OUTPUT_HANDLE),b1,sz,&nWr,0) | ||||||
| @@ -547,8 +552,9 @@ int sqlite3_fputs(const char *z, FILE *out){ | |||||||
|     }else |     }else | ||||||
| #endif | #endif | ||||||
|     { |     { | ||||||
|       /* For non-console I/O, or if SQLITE_USE_STDIO_FOR_CONSOLE is defined
 |       /* As long as SQLITE_USE_W32_FOR_CONSOLE_IO is not defined, or for
 | ||||||
|       ** then write using the standard library. */ |       ** non-console I/O even if that macro is defined, write using the | ||||||
|  |       ** standard library. */ | ||||||
|       _setmode(_fileno(out), _O_U8TEXT); |       _setmode(_fileno(out), _O_U8TEXT); | ||||||
|       if( UseBinaryWText(out) ){ |       if( UseBinaryWText(out) ){ | ||||||
|         piecemealOutput(b1, sz, out); |         piecemealOutput(b1, sz, out); | ||||||
| @@ -5314,7 +5320,7 @@ static u8* fromBase64( char *pIn, int ncIn, u8 *pOut ){ | |||||||
|       deliberate_fall_through; /* FALLTHRU */ |       deliberate_fall_through; /* FALLTHRU */ | ||||||
|     case 1: |     case 1: | ||||||
|       pOut[0] = (qv>>16) & 0xff; |       pOut[0] = (qv>>16) & 0xff; | ||||||
|       deliberate_fall_through; /* FALLTHRU */ |       break; | ||||||
|     } |     } | ||||||
|     pOut += nbo; |     pOut += nbo; | ||||||
|   } |   } | ||||||
| @@ -17199,7 +17205,7 @@ static void vfstraceDlClose(sqlite3_vfs *pVfs, void *pHandle){ | |||||||
|   vfstrace_info *pInfo = (vfstrace_info*)pVfs->pAppData; |   vfstrace_info *pInfo = (vfstrace_info*)pVfs->pAppData; | ||||||
|   sqlite3_vfs *pRoot = pInfo->pRootVfs; |   sqlite3_vfs *pRoot = pInfo->pRootVfs; | ||||||
|   vfstraceOnOff(pInfo, VTR_DLCLOSE); |   vfstraceOnOff(pInfo, VTR_DLCLOSE); | ||||||
|   vfstrace_printf(pInfo, "%s.xDlOpen()\n", pInfo->zVfsName); |   vfstrace_printf(pInfo, "%s.xDlClose()\n", pInfo->zVfsName); | ||||||
|   pRoot->xDlClose(pRoot, pHandle); |   pRoot->xDlClose(pRoot, pHandle); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| @@ -28885,6 +28891,9 @@ static int do_meta_command(char *zLine, ShellState *p){ | |||||||
|       const char *zName; |       const char *zName; | ||||||
|       int op; |       int op; | ||||||
|     } aDbConfig[] = { |     } aDbConfig[] = { | ||||||
|  |         { "attach_create",      SQLITE_DBCONFIG_ENABLE_ATTACH_CREATE  }, | ||||||
|  |         { "attach_write",       SQLITE_DBCONFIG_ENABLE_ATTACH_WRITE   }, | ||||||
|  |         { "comments",           SQLITE_DBCONFIG_ENABLE_COMMENTS       }, | ||||||
|         { "defensive",          SQLITE_DBCONFIG_DEFENSIVE             }, |         { "defensive",          SQLITE_DBCONFIG_DEFENSIVE             }, | ||||||
|         { "dqs_ddl",            SQLITE_DBCONFIG_DQS_DDL               }, |         { "dqs_ddl",            SQLITE_DBCONFIG_DQS_DDL               }, | ||||||
|         { "dqs_dml",            SQLITE_DBCONFIG_DQS_DML               }, |         { "dqs_dml",            SQLITE_DBCONFIG_DQS_DML               }, | ||||||
| @@ -31635,6 +31644,7 @@ static int do_meta_command(char *zLine, ShellState *p){ | |||||||
|             { 0x04000000, 1, "NullUnusedCols" }, |             { 0x04000000, 1, "NullUnusedCols" }, | ||||||
|             { 0x08000000, 1, "OnePass" }, |             { 0x08000000, 1, "OnePass" }, | ||||||
|             { 0x10000000, 1, "OrderBySubq" }, |             { 0x10000000, 1, "OrderBySubq" }, | ||||||
|  |             { 0x20000000, 1, "StarQuery" }, | ||||||
|             { 0xffffffff, 0, "All" }, |             { 0xffffffff, 0, "All" }, | ||||||
|           }; |           }; | ||||||
|           unsigned int curOpt; |           unsigned int curOpt; | ||||||
|   | |||||||
							
								
								
									
										4064
									
								
								deps/sqlite/sqlite3.c
									
									
									
									
										vendored
									
									
								
							
							
						
						
							
								
								
									
										153
									
								
								deps/sqlite/sqlite3.h
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -146,9 +146,9 @@ extern "C" { | |||||||
| ** [sqlite3_libversion_number()], [sqlite3_sourceid()], | ** [sqlite3_libversion_number()], [sqlite3_sourceid()], | ||||||
| ** [sqlite_version()] and [sqlite_source_id()]. | ** [sqlite_version()] and [sqlite_source_id()]. | ||||||
| */ | */ | ||||||
| #define SQLITE_VERSION        "3.48.0" | #define SQLITE_VERSION        "3.49.1" | ||||||
| #define SQLITE_VERSION_NUMBER 3048000 | #define SQLITE_VERSION_NUMBER 3049001 | ||||||
| #define SQLITE_SOURCE_ID      "2025-01-14 11:05:00 d2fe6b05f38d9d7cd78c5d252e99ac59f1aea071d669830c1ffe4e8966e84010" | #define SQLITE_SOURCE_ID      "2025-02-18 13:38:58 873d4e274b4988d260ba8354a9718324a1c26187a4ab4c1cc0227c03d0f10e70" | ||||||
|  |  | ||||||
| /* | /* | ||||||
| ** CAPI3REF: Run-Time Library Version Numbers | ** CAPI3REF: Run-Time Library Version Numbers | ||||||
| @@ -2211,7 +2211,15 @@ struct sqlite3_mem_methods { | |||||||
| ** CAPI3REF: Database Connection Configuration Options | ** CAPI3REF: Database Connection Configuration Options | ||||||
| ** | ** | ||||||
| ** These constants are the available integer configuration options that | ** These constants are the available integer configuration options that | ||||||
| ** can be passed as the second argument to the [sqlite3_db_config()] interface. | ** can be passed as the second parameter to the [sqlite3_db_config()] interface. | ||||||
|  | ** | ||||||
|  | ** The [sqlite3_db_config()] interface is a var-args functions.  It takes a | ||||||
|  | ** variable number of parameters, though always at least two.  The number of | ||||||
|  | ** parameters passed into sqlite3_db_config() depends on which of these | ||||||
|  | ** constants is given as the second parameter.  This documentation page | ||||||
|  | ** refers to parameters beyond the second as "arguments".  Thus, when this | ||||||
|  | ** page says "the N-th argument" it means "the N-th parameter past the | ||||||
|  | ** configuration option" or "the (N+2)-th parameter to sqlite3_db_config()". | ||||||
| ** | ** | ||||||
| ** New configuration options may be added in future releases of SQLite. | ** New configuration options may be added in future releases of SQLite. | ||||||
| ** Existing configuration options might be discontinued.  Applications | ** Existing configuration options might be discontinued.  Applications | ||||||
| @@ -2223,8 +2231,14 @@ struct sqlite3_mem_methods { | |||||||
| ** <dl> | ** <dl> | ||||||
| ** [[SQLITE_DBCONFIG_LOOKASIDE]] | ** [[SQLITE_DBCONFIG_LOOKASIDE]] | ||||||
| ** <dt>SQLITE_DBCONFIG_LOOKASIDE</dt> | ** <dt>SQLITE_DBCONFIG_LOOKASIDE</dt> | ||||||
| ** <dd> ^This option takes three additional arguments that determine the | ** <dd> The SQLITE_DBCONFIG_LOOKASIDE option is used to adjust the | ||||||
| ** [lookaside memory allocator] configuration for the [database connection]. | ** configuration of the lookaside memory allocator within a database | ||||||
|  | ** connection. | ||||||
|  | ** The arguments to the SQLITE_DBCONFIG_LOOKASIDE option are <i>not</i> | ||||||
|  | ** in the [DBCONFIG arguments|usual format]. | ||||||
|  | ** The SQLITE_DBCONFIG_LOOKASIDE option takes three arguments, not two, | ||||||
|  | ** so that a call to [sqlite3_db_config()] that uses SQLITE_DBCONFIG_LOOKASIDE | ||||||
|  | ** should have a total of five parameters. | ||||||
| ** ^The first argument (the third parameter to [sqlite3_db_config()] is a | ** ^The first argument (the third parameter to [sqlite3_db_config()] is a | ||||||
| ** pointer to a memory buffer to use for lookaside memory. | ** pointer to a memory buffer to use for lookaside memory. | ||||||
| ** ^The first argument after the SQLITE_DBCONFIG_LOOKASIDE verb | ** ^The first argument after the SQLITE_DBCONFIG_LOOKASIDE verb | ||||||
| @@ -2247,7 +2261,8 @@ struct sqlite3_mem_methods { | |||||||
| ** [[SQLITE_DBCONFIG_ENABLE_FKEY]] | ** [[SQLITE_DBCONFIG_ENABLE_FKEY]] | ||||||
| ** <dt>SQLITE_DBCONFIG_ENABLE_FKEY</dt> | ** <dt>SQLITE_DBCONFIG_ENABLE_FKEY</dt> | ||||||
| ** <dd> ^This option is used to enable or disable the enforcement of | ** <dd> ^This option is used to enable or disable the enforcement of | ||||||
| ** [foreign key constraints].  There should be two additional arguments. | ** [foreign key constraints].  This is the same setting that is | ||||||
|  | ** enabled or disabled by the [PRAGMA foreign_keys] statement. | ||||||
| ** The first argument is an integer which is 0 to disable FK enforcement, | ** The first argument is an integer which is 0 to disable FK enforcement, | ||||||
| ** positive to enable FK enforcement or negative to leave FK enforcement | ** positive to enable FK enforcement or negative to leave FK enforcement | ||||||
| ** unchanged.  The second parameter is a pointer to an integer into which | ** unchanged.  The second parameter is a pointer to an integer into which | ||||||
| @@ -2269,13 +2284,13 @@ struct sqlite3_mem_methods { | |||||||
| ** <p>Originally this option disabled all triggers.  ^(However, since | ** <p>Originally this option disabled all triggers.  ^(However, since | ||||||
| ** SQLite version 3.35.0, TEMP triggers are still allowed even if | ** SQLite version 3.35.0, TEMP triggers are still allowed even if | ||||||
| ** this option is off.  So, in other words, this option now only disables | ** this option is off.  So, in other words, this option now only disables | ||||||
| ** triggers in the main database schema or in the schemas of ATTACH-ed | ** triggers in the main database schema or in the schemas of [ATTACH]-ed | ||||||
| ** databases.)^ </dd> | ** databases.)^ </dd> | ||||||
| ** | ** | ||||||
| ** [[SQLITE_DBCONFIG_ENABLE_VIEW]] | ** [[SQLITE_DBCONFIG_ENABLE_VIEW]] | ||||||
| ** <dt>SQLITE_DBCONFIG_ENABLE_VIEW</dt> | ** <dt>SQLITE_DBCONFIG_ENABLE_VIEW</dt> | ||||||
| ** <dd> ^This option is used to enable or disable [CREATE VIEW | views]. | ** <dd> ^This option is used to enable or disable [CREATE VIEW | views]. | ||||||
| ** There should be two additional arguments. | ** There must be two additional arguments. | ||||||
| ** The first argument is an integer which is 0 to disable views, | ** The first argument is an integer which is 0 to disable views, | ||||||
| ** positive to enable views or negative to leave the setting unchanged. | ** positive to enable views or negative to leave the setting unchanged. | ||||||
| ** The second parameter is a pointer to an integer into which | ** The second parameter is a pointer to an integer into which | ||||||
| @@ -2294,7 +2309,7 @@ struct sqlite3_mem_methods { | |||||||
| ** <dd> ^This option is used to enable or disable the | ** <dd> ^This option is used to enable or disable the | ||||||
| ** [fts3_tokenizer()] function which is part of the | ** [fts3_tokenizer()] function which is part of the | ||||||
| ** [FTS3] full-text search engine extension. | ** [FTS3] full-text search engine extension. | ||||||
| ** There should be two additional arguments. | ** There must be two additional arguments. | ||||||
| ** The first argument is an integer which is 0 to disable fts3_tokenizer() or | ** The first argument is an integer which is 0 to disable fts3_tokenizer() or | ||||||
| ** positive to enable fts3_tokenizer() or negative to leave the setting | ** positive to enable fts3_tokenizer() or negative to leave the setting | ||||||
| ** unchanged. | ** unchanged. | ||||||
| @@ -2309,7 +2324,7 @@ struct sqlite3_mem_methods { | |||||||
| ** interface independently of the [load_extension()] SQL function. | ** interface independently of the [load_extension()] SQL function. | ||||||
| ** The [sqlite3_enable_load_extension()] API enables or disables both the | ** The [sqlite3_enable_load_extension()] API enables or disables both the | ||||||
| ** C-API [sqlite3_load_extension()] and the SQL function [load_extension()]. | ** C-API [sqlite3_load_extension()] and the SQL function [load_extension()]. | ||||||
| ** There should be two additional arguments. | ** There must be two additional arguments. | ||||||
| ** When the first argument to this interface is 1, then only the C-API is | ** When the first argument to this interface is 1, then only the C-API is | ||||||
| ** enabled and the SQL function remains disabled.  If the first argument to | ** enabled and the SQL function remains disabled.  If the first argument to | ||||||
| ** this interface is 0, then both the C-API and the SQL function are disabled. | ** this interface is 0, then both the C-API and the SQL function are disabled. | ||||||
| @@ -2323,23 +2338,30 @@ struct sqlite3_mem_methods { | |||||||
| ** | ** | ||||||
| ** [[SQLITE_DBCONFIG_MAINDBNAME]] <dt>SQLITE_DBCONFIG_MAINDBNAME</dt> | ** [[SQLITE_DBCONFIG_MAINDBNAME]] <dt>SQLITE_DBCONFIG_MAINDBNAME</dt> | ||||||
| ** <dd> ^This option is used to change the name of the "main" database | ** <dd> ^This option is used to change the name of the "main" database | ||||||
| ** schema.  ^The sole argument is a pointer to a constant UTF8 string | ** schema.  This option does not follow the | ||||||
| ** which will become the new schema name in place of "main".  ^SQLite | ** [DBCONFIG arguments|usual SQLITE_DBCONFIG argument format]. | ||||||
| ** does not make a copy of the new main schema name string, so the application | ** This option takes exactly one additional argument so that the | ||||||
| ** must ensure that the argument passed into this DBCONFIG option is unchanged | ** [sqlite3_db_config()] call has a total of three parameters.  The | ||||||
| ** until after the database connection closes. | ** extra argument must be a pointer to a constant UTF8 string which | ||||||
|  | ** will become the new schema name in place of "main".  ^SQLite does | ||||||
|  | ** not make a copy of the new main schema name string, so the application | ||||||
|  | ** must ensure that the argument passed into SQLITE_DBCONFIG MAINDBNAME | ||||||
|  | ** is unchanged until after the database connection closes. | ||||||
| ** </dd> | ** </dd> | ||||||
| ** | ** | ||||||
| ** [[SQLITE_DBCONFIG_NO_CKPT_ON_CLOSE]] | ** [[SQLITE_DBCONFIG_NO_CKPT_ON_CLOSE]] | ||||||
| ** <dt>SQLITE_DBCONFIG_NO_CKPT_ON_CLOSE</dt> | ** <dt>SQLITE_DBCONFIG_NO_CKPT_ON_CLOSE</dt> | ||||||
| ** <dd> Usually, when a database in wal mode is closed or detached from a | ** <dd> Usually, when a database in [WAL mode] is closed or detached from a | ||||||
| ** database handle, SQLite checks if this will mean that there are now no | ** database handle, SQLite checks if if there are other connections to the | ||||||
| ** connections at all to the database. If so, it performs a checkpoint | ** same database, and if there are no other database connection (if the | ||||||
| ** operation before closing the connection. This option may be used to | ** connection being closed is the last open connection to the database), | ||||||
| ** override this behavior. The first parameter passed to this operation | ** then SQLite performs a [checkpoint] before closing the connection and | ||||||
| ** is an integer - positive to disable checkpoints-on-close, or zero (the | ** deletes the WAL file.  The SQLITE_DBCONFIG_NO_CKPT_ON_CLOSE option can | ||||||
| ** default) to enable them, and negative to leave the setting unchanged. | ** be used to override that behavior. The first argument passed to this | ||||||
| ** The second parameter is a pointer to an integer | ** operation (the third parameter to [sqlite3_db_config()]) is an integer | ||||||
|  | ** which is positive to disable checkpoints-on-close, or zero (the default) | ||||||
|  | ** to enable them, and negative to leave the setting unchanged. | ||||||
|  | ** The second argument (the fourth parameter) is a pointer to an integer | ||||||
| ** into which is written 0 or 1 to indicate whether checkpoints-on-close | ** into which is written 0 or 1 to indicate whether checkpoints-on-close | ||||||
| ** have been disabled - 0 if they are not disabled, 1 if they are. | ** have been disabled - 0 if they are not disabled, 1 if they are. | ||||||
| ** </dd> | ** </dd> | ||||||
| @@ -2500,7 +2522,7 @@ struct sqlite3_mem_methods { | |||||||
| ** statistics. For statistics to be collected, the flag must be set on | ** statistics. For statistics to be collected, the flag must be set on | ||||||
| ** the database handle both when the SQL statement is prepared and when it | ** the database handle both when the SQL statement is prepared and when it | ||||||
| ** is stepped. The flag is set (collection of statistics is enabled) | ** is stepped. The flag is set (collection of statistics is enabled) | ||||||
| ** by default.  This option takes two arguments: an integer and a pointer to | ** by default. <p>This option takes two arguments: an integer and a pointer to | ||||||
| ** an integer..  The first argument is 1, 0, or -1 to enable, disable, or | ** an integer..  The first argument is 1, 0, or -1 to enable, disable, or | ||||||
| ** leave unchanged the statement scanstatus option.  If the second argument | ** leave unchanged the statement scanstatus option.  If the second argument | ||||||
| ** is not NULL, then the value of the statement scanstatus setting after | ** is not NULL, then the value of the statement scanstatus setting after | ||||||
| @@ -2514,7 +2536,7 @@ struct sqlite3_mem_methods { | |||||||
| ** in which tables and indexes are scanned so that the scans start at the end | ** in which tables and indexes are scanned so that the scans start at the end | ||||||
| ** and work toward the beginning rather than starting at the beginning and | ** and work toward the beginning rather than starting at the beginning and | ||||||
| ** working toward the end. Setting SQLITE_DBCONFIG_REVERSE_SCANORDER is the | ** working toward the end. Setting SQLITE_DBCONFIG_REVERSE_SCANORDER is the | ||||||
| ** same as setting [PRAGMA reverse_unordered_selects].  This option takes | ** same as setting [PRAGMA reverse_unordered_selects]. <p>This option takes | ||||||
| ** two arguments which are an integer and a pointer to an integer.  The first | ** two arguments which are an integer and a pointer to an integer.  The first | ||||||
| ** argument is 1, 0, or -1 to enable, disable, or leave unchanged the | ** argument is 1, 0, or -1 to enable, disable, or leave unchanged the | ||||||
| ** reverse scan order flag, respectively.  If the second argument is not NULL, | ** reverse scan order flag, respectively.  If the second argument is not NULL, | ||||||
| @@ -2523,7 +2545,76 @@ struct sqlite3_mem_methods { | |||||||
| ** first argument. | ** first argument. | ||||||
| ** </dd> | ** </dd> | ||||||
| ** | ** | ||||||
|  | ** [[SQLITE_DBCONFIG_ENABLE_ATTACH_CREATE]] | ||||||
|  | ** <dt>SQLITE_DBCONFIG_ENABLE_ATTACH_CREATE</dt> | ||||||
|  | ** <dd>The SQLITE_DBCONFIG_ENABLE_ATTACH_CREATE option enables or disables | ||||||
|  | ** the ability of the [ATTACH DATABASE] SQL command to create a new database | ||||||
|  | ** file if the database filed named in the ATTACH command does not already | ||||||
|  | ** exist.  This ability of ATTACH to create a new database is enabled by | ||||||
|  | ** default.  Applications can disable or reenable the ability for ATTACH to | ||||||
|  | ** create new database files using this DBCONFIG option.<p> | ||||||
|  | ** This option takes two arguments which are an integer and a pointer | ||||||
|  | ** to an integer.  The first argument is 1, 0, or -1 to enable, disable, or | ||||||
|  | ** leave unchanged the attach-create flag, respectively.  If the second | ||||||
|  | ** argument is not NULL, then 0 or 1 is written into the integer that the | ||||||
|  | ** second argument points to depending on if the attach-create flag is set | ||||||
|  | ** after processing the first argument. | ||||||
|  | ** </dd> | ||||||
|  | ** | ||||||
|  | ** [[SQLITE_DBCONFIG_ENABLE_ATTACH_WRITE]] | ||||||
|  | ** <dt>SQLITE_DBCONFIG_ENABLE_ATTACH_WRITE</dt> | ||||||
|  | ** <dd>The SQLITE_DBCONFIG_ENABLE_ATTACH_WRITE option enables or disables the | ||||||
|  | ** ability of the [ATTACH DATABASE] SQL command to open a database for writing. | ||||||
|  | ** This capability is enabled by default.  Applications can disable or | ||||||
|  | ** reenable this capability using the current DBCONFIG option.  If the | ||||||
|  | ** the this capability is disabled, the [ATTACH] command will still work, | ||||||
|  | ** but the database will be opened read-only.  If this option is disabled, | ||||||
|  | ** then the ability to create a new database using [ATTACH] is also disabled, | ||||||
|  | ** regardless of the value of the [SQLITE_DBCONFIG_ENABLE_ATTACH_CREATE] | ||||||
|  | ** option.<p> | ||||||
|  | ** This option takes two arguments which are an integer and a pointer | ||||||
|  | ** to an integer.  The first argument is 1, 0, or -1 to enable, disable, or | ||||||
|  | ** leave unchanged the ability to ATTACH another database for writing, | ||||||
|  | ** respectively.  If the second argument is not NULL, then 0 or 1 is written | ||||||
|  | ** into the integer to which the second argument points, depending on whether | ||||||
|  | ** the ability to ATTACH a read/write database is enabled or disabled | ||||||
|  | ** after processing the first argument. | ||||||
|  | ** </dd> | ||||||
|  | ** | ||||||
|  | ** [[SQLITE_DBCONFIG_ENABLE_COMMENTS]] | ||||||
|  | ** <dt>SQLITE_DBCONFIG_ENABLE_COMMENTS</dt> | ||||||
|  | ** <dd>The SQLITE_DBCONFIG_ENABLE_COMMENTS option enables or disables the | ||||||
|  | ** ability to include comments in SQL text.  Comments are enabled by default. | ||||||
|  | ** An application can disable or reenable comments in SQL text using this | ||||||
|  | ** DBCONFIG option.<p> | ||||||
|  | ** This option takes two arguments which are an integer and a pointer | ||||||
|  | ** to an integer.  The first argument is 1, 0, or -1 to enable, disable, or | ||||||
|  | ** leave unchanged the ability to use comments in SQL text, | ||||||
|  | ** respectively.  If the second argument is not NULL, then 0 or 1 is written | ||||||
|  | ** into the integer that the second argument points to depending on if | ||||||
|  | ** comments are allowed in SQL text after processing the first argument. | ||||||
|  | ** </dd> | ||||||
|  | ** | ||||||
| ** </dl> | ** </dl> | ||||||
|  | ** | ||||||
|  | ** [[DBCONFIG arguments]] <h3>Arguments To SQLITE_DBCONFIG Options</h3> | ||||||
|  | ** | ||||||
|  | ** <p>Most of the SQLITE_DBCONFIG options take two arguments, so that the | ||||||
|  | ** overall call to [sqlite3_db_config()] has a total of four parameters. | ||||||
|  | ** The first argument (the third parameter to sqlite3_db_config()) is a integer. | ||||||
|  | ** The second argument is a pointer to an integer.  If the first argument is 1, | ||||||
|  | ** then the option becomes enabled.  If the first integer argument is 0, then the | ||||||
|  | ** option is disabled.  If the first argument is -1, then the option setting | ||||||
|  | ** is unchanged.  The second argument, the pointer to an integer, may be NULL. | ||||||
|  | ** If the second argument is not NULL, then a value of 0 or 1 is written into | ||||||
|  | ** the integer to which the second argument points, depending on whether the | ||||||
|  | ** setting is disabled or enabled after applying any changes specified by | ||||||
|  | ** the first argument. | ||||||
|  | ** | ||||||
|  | ** <p>While most SQLITE_DBCONFIG options use the argument format | ||||||
|  | ** described in the previous paragraph, the [SQLITE_DBCONFIG_MAINDBNAME] | ||||||
|  | ** and [SQLITE_DBCONFIG_LOOKASIDE] options are different.  See the | ||||||
|  | ** documentation of those exceptional options for details. | ||||||
| */ | */ | ||||||
| #define SQLITE_DBCONFIG_MAINDBNAME            1000 /* const char* */ | #define SQLITE_DBCONFIG_MAINDBNAME            1000 /* const char* */ | ||||||
| #define SQLITE_DBCONFIG_LOOKASIDE             1001 /* void* int int */ | #define SQLITE_DBCONFIG_LOOKASIDE             1001 /* void* int int */ | ||||||
| @@ -2545,7 +2636,10 @@ struct sqlite3_mem_methods { | |||||||
| #define SQLITE_DBCONFIG_TRUSTED_SCHEMA        1017 /* int int* */ | #define SQLITE_DBCONFIG_TRUSTED_SCHEMA        1017 /* int int* */ | ||||||
| #define SQLITE_DBCONFIG_STMT_SCANSTATUS       1018 /* int int* */ | #define SQLITE_DBCONFIG_STMT_SCANSTATUS       1018 /* int int* */ | ||||||
| #define SQLITE_DBCONFIG_REVERSE_SCANORDER     1019 /* int int* */ | #define SQLITE_DBCONFIG_REVERSE_SCANORDER     1019 /* int int* */ | ||||||
| #define SQLITE_DBCONFIG_MAX                   1019 /* Largest DBCONFIG */ | #define SQLITE_DBCONFIG_ENABLE_ATTACH_CREATE  1020 /* int int* */ | ||||||
|  | #define SQLITE_DBCONFIG_ENABLE_ATTACH_WRITE   1021 /* int int* */ | ||||||
|  | #define SQLITE_DBCONFIG_ENABLE_COMMENTS       1022 /* int int* */ | ||||||
|  | #define SQLITE_DBCONFIG_MAX                   1022 /* Largest DBCONFIG */ | ||||||
|  |  | ||||||
| /* | /* | ||||||
| ** CAPI3REF: Enable Or Disable Extended Result Codes | ** CAPI3REF: Enable Or Disable Extended Result Codes | ||||||
| @@ -10748,8 +10842,9 @@ SQLITE_API SQLITE_EXPERIMENTAL int sqlite3_snapshot_recover(sqlite3 *db, const c | |||||||
| /* | /* | ||||||
| ** CAPI3REF: Serialize a database | ** CAPI3REF: Serialize a database | ||||||
| ** | ** | ||||||
| ** The sqlite3_serialize(D,S,P,F) interface returns a pointer to memory | ** The sqlite3_serialize(D,S,P,F) interface returns a pointer to | ||||||
| ** that is a serialization of the S database on [database connection] D. | ** memory that is a serialization of the S database on | ||||||
|  | ** [database connection] D.  If S is a NULL pointer, the main database is used. | ||||||
| ** If P is not a NULL pointer, then the size of the database in bytes | ** If P is not a NULL pointer, then the size of the database in bytes | ||||||
| ** is written into *P. | ** is written into *P. | ||||||
| ** | ** | ||||||
|   | |||||||
							
								
								
									
										1
									
								
								deps/zsign
									
									
									
									
										vendored
									
									
										Submodule
									
								
							
							
								
								
								
								
								
							
						
						
							
								
								
									
										1
									
								
								docs
									
									
									
									
									
								
							
							
								
								
								
								
								
							
						
						
							
								
								
									
										63
									
								
								docs/app_development_cheat_sheet.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,63 @@ | |||||||
|  | # App Development Cheat Sheet | ||||||
|  |  | ||||||
|  | Making apps for the impatient tilde friend. | ||||||
|  |  | ||||||
|  | ## Prerequisites | ||||||
|  |  | ||||||
|  | - either run your own instance or use [tildefriends.net](https://www.tildefriends.net/) | ||||||
|  | - register and login | ||||||
|  | - [optional] use the `ssb` app to create yourself an SSB identity | ||||||
|  |  | ||||||
|  | ## Development Process | ||||||
|  |  | ||||||
|  | 1.  hit the `edit` link from any app or new app URL | ||||||
|  | 2.  make sure the path in the text box is under your username: `/~username/app/` | ||||||
|  | 3.  write server-side code in `app.js` | ||||||
|  | 4.  click the `save` button or press the save hotkey (Alt+S or _[browser-specific modifiers]_+S) | ||||||
|  | 5.  see the app reload on the right side | ||||||
|  |  | ||||||
|  | ## Output | ||||||
|  |  | ||||||
|  | - `app.setDocument(html)` - send HTML to the browser | ||||||
|  | - `print(...)` - send values to the browser's developer console | ||||||
|  |  | ||||||
|  | ## Persistence | ||||||
|  |  | ||||||
|  | - `app.localStorageGet(key)` -> `value` | ||||||
|  | - `app.localStorageSet(key, value)` | ||||||
|  | - `database()`, `shared_database(key)`, `my_shared_database(package, key)` | ||||||
|  |   - `db.get(key)` -> `value` | ||||||
|  |   - `db.set(key, value)` | ||||||
|  |   - `db.exchange(key, expected, value)` -> `exchanged` | ||||||
|  |   - `db.remove(key)` | ||||||
|  |   - `db.getAll()` -> `[key1, ...]` | ||||||
|  |   - `db.getLike(pattern)` -> `{key1: value1, ...}` | ||||||
|  |  | ||||||
|  | ## SSB | ||||||
|  |  | ||||||
|  | - `ssb.createIdentity()` -> `id` | ||||||
|  | - `ssb.getIdentities()` -> `[id1, ...]` | ||||||
|  | - `ssb.appendMessageWithIdentity(id, content)` -> `message_id` | ||||||
|  | - `ssb.blobStore(blob)` -> `blob_id` | ||||||
|  | - `ssb.blobGet(id)` -> `blob` | ||||||
|  | - `ssb.sqlAsync(query, args, row_callback)` | ||||||
|  |  | ||||||
|  | ## TF-RPC | ||||||
|  |  | ||||||
|  | Stock helper code for calling functions across the web server and browser boundary. | ||||||
|  |  | ||||||
|  | - on the server: `import * as tfrpc from "/tfrpc.js";` | ||||||
|  | - in the browser: `import * as tfrpc from "/static/tfrpc.js";` | ||||||
|  | - either direction: | ||||||
|  |   - register a function: `tfrpc.register(function my_function() {});` | ||||||
|  |   - call a remote function: `let promise = tfrpc.rpc.my_function();` | ||||||
|  |  | ||||||
|  | ## Share | ||||||
|  |  | ||||||
|  | - give out web links: [https://www.tildefriends.net/~cory/screwble/](https://www.tildefriends.net/~cory/screwble/) | ||||||
|  | - use the `Attach App` button when composing a post in [the SSB app](https://www.tildefriends.net/~core/ssb/) | ||||||
|  |  | ||||||
|  | ## More Docs | ||||||
|  |  | ||||||
|  | - [api reference](https://www.tildefriends.net/~cory/api/) | ||||||
|  | - [source code](https://dev.tildefriends.net/cory/tildefriends/releases) | ||||||
							
								
								
									
										166
									
								
								docs/app_development_guide.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,166 @@ | |||||||
|  | # App Development Guide | ||||||
|  |  | ||||||
|  | A Tilde Friends application starts with code that runs on a Tilde Friends server, possibly far away from where you wrote it, in a little JavaScript environment, in its own restricted process, with the only access to the outside world being the ability to send messages to the server. This document gives some recipes showing how that can be used to build a functional user-facing application in light of the unique constraints present. | ||||||
|  |  | ||||||
|  | ## Example 1: Hello, world! | ||||||
|  |  | ||||||
|  | Of course we must start with a classic. | ||||||
|  |  | ||||||
|  | ### app.js | ||||||
|  |  | ||||||
|  | ```js | ||||||
|  | app.setDocument('<h1 style="color: #fff">Hello, world!</h1>'); | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ### Output | ||||||
|  |  | ||||||
|  | <h1 style="color: #fff">Hello, world!</h1> | ||||||
|  |  | ||||||
|  | ### Explanation | ||||||
|  |  | ||||||
|  | At a glance, this might seem mundane, but for it to work: | ||||||
|  |  | ||||||
|  | - the server starts a real process for your app and loads your code into it | ||||||
|  | - your code runs | ||||||
|  | - `app.setDocument()` sends a message back to the server | ||||||
|  | - the server interprets the message and redirects it to the browser | ||||||
|  | - `core/client.js` in the browser receives the message and puts your HTML into an iframe | ||||||
|  | - your HTML is presented by the browser in an iframe sandbox | ||||||
|  |  | ||||||
|  | But you don't have to think about all that. Call a function, and you see the result. | ||||||
|  |  | ||||||
|  | ## Example 2: Hit Counter | ||||||
|  |  | ||||||
|  | Let's take advantage of code running on the server and create a little hit counter using a key value store shared between all visitors. | ||||||
|  |  | ||||||
|  | ### app.js | ||||||
|  |  | ||||||
|  | ```js | ||||||
|  | async function main() { | ||||||
|  | 	let db = await shared_database('visitors'); | ||||||
|  | 	let count = parseInt((await db.get('visitors')) ?? '0') + 1; | ||||||
|  | 	await db.set('visitors', count.toString()); | ||||||
|  | 	await app.setDocument(` | ||||||
|  | 		<h1 style="color: #fff">Welcome, visitor #${count}!</h1> | ||||||
|  | 	`); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | main(); | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ### Output | ||||||
|  |  | ||||||
|  | <h1 style="color: #fff">Welcome, visitor #1!</h1> | ||||||
|  |  | ||||||
|  | ### Explanation | ||||||
|  |  | ||||||
|  | Just as pure browser apps have access to `localStorage`, Tilde Friends apps have access to key-value storage on the server. | ||||||
|  |  | ||||||
|  | The interface is a bit clunky and will likely change someday, but this example gets a database object, from which you can get and set string values by key. There are various on `shared_database` that let you store data that is private to the user or shared by different criteria. | ||||||
|  |  | ||||||
|  | Also, even though any browser-side code is sandboxed, it is allowed to access browser local storage by going through Tilde Friends API, because sometimes that is useful. | ||||||
|  |  | ||||||
|  | ## Example 3: Files | ||||||
|  |  | ||||||
|  | Suppose you don't want to create your entire app in a single server-side file as we've done with the previous examples. There are some tools to allow you to begin to organize. | ||||||
|  |  | ||||||
|  | ### app.js | ||||||
|  |  | ||||||
|  | ```js | ||||||
|  | async function main() { | ||||||
|  | 	let html = utf8Decode(await getFile('index.html')); | ||||||
|  | 	app.setDocument(html); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | main(); | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ### index.html | ||||||
|  |  | ||||||
|  | ```html | ||||||
|  | <html> | ||||||
|  | 	<head> | ||||||
|  | 		<script type="module" src="script.js"></script> | ||||||
|  | 	</head> | ||||||
|  | 	<body style="color: #fff"> | ||||||
|  | 		<h1>File Test</h1> | ||||||
|  | 	</body> | ||||||
|  | </html> | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ### script.js | ||||||
|  |  | ||||||
|  | ```js | ||||||
|  | window.addEventListener('load', function() { | ||||||
|  | 	document.body.appendChild(document.createTextNode('Hello, world'); | ||||||
|  | }); | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ### Output | ||||||
|  |  | ||||||
|  | <h1>File Test</h1><p>Hello, world!</p> | ||||||
|  |  | ||||||
|  | ### Explanation | ||||||
|  |  | ||||||
|  | On the server, `utf8Decode(await getFile(fileName))` lets you load a file from your app. In the browser, your app files are made available by HTTP, so you can `<script src="my_script.js"></script>` and such to access them. | ||||||
|  |  | ||||||
|  | ## Example 4: Remote Procedure Call | ||||||
|  |  | ||||||
|  | While making calls between the client and the server, it is possible to pass functions across that boundary. `tfrpc.js` is a tiny script which builds on that feature to try to hide some of the complexities. | ||||||
|  |  | ||||||
|  | ### app.js | ||||||
|  |  | ||||||
|  | ```js | ||||||
|  | import * as tf from '/tfrpc.js'; | ||||||
|  |  | ||||||
|  | function sum() { | ||||||
|  | 	let s = 0; | ||||||
|  | 	for (let x of arguments) { | ||||||
|  | 		s += x; | ||||||
|  | 	} | ||||||
|  | 	return s; | ||||||
|  | } | ||||||
|  | tf.register(sum); | ||||||
|  |  | ||||||
|  | async function main() { | ||||||
|  | 	app.setDocument(utf8Decode(await getFile('index.html'))); | ||||||
|  | } | ||||||
|  | main(); | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ### index.html | ||||||
|  |  | ||||||
|  | ```html | ||||||
|  | <html> | ||||||
|  | 	<body> | ||||||
|  | 		<h1 id="result">Calculating...</h1> | ||||||
|  | 	</body> | ||||||
|  | 	<script type="module" src="script.js"></script> | ||||||
|  | </html> | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ### script.js | ||||||
|  |  | ||||||
|  | ```js | ||||||
|  | import * as tf from '/static/tfrpc.js'; | ||||||
|  |  | ||||||
|  | window.addEventListener('load', async function () { | ||||||
|  | 	document.getElementById('result').innerText = await tf.rpc.sum(1, 2, 3); | ||||||
|  | }); | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ### Output | ||||||
|  |  | ||||||
|  | <h1>6</h1> | ||||||
|  |  | ||||||
|  | ### Explanation | ||||||
|  |  | ||||||
|  | Here the browser makes an asynchronous call to the server to do some basic math and update its DOM with the result. | ||||||
|  |  | ||||||
|  | With your favorite Vue/Lit/React/... library on the client-side and your favorite Tilde Friends API calls registered with tfrpc, it becomes pretty easy to start extracting interesting information from, say, SQL queries over Secure Scuttlebutt data, and generating complicated, dynamic user interface. These are the building blocks I used to make the current Tilde Friends SSB client interface. | ||||||
|  |  | ||||||
|  | ## Conclusion | ||||||
|  |  | ||||||
|  | Tilde Friends is currently a pile of all the parts that I thought I needed to build interesting web applications, tied together by code that tries to walk the fine line between being secure enough to let us safely run code on the same device and being usable enough that you can open a tab in your browser and start building just by typing code. | ||||||
|  |  | ||||||
|  | I don't claim it thoroughly accomplishes either yet, but I believe it is at a stage where it is showing how promising this approach can be, and I am excited for you to take it for a spin and share. | ||||||
							
								
								
									
										15
									
								
								docs/inspiration.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,15 @@ | |||||||
|  | # Inspiration | ||||||
|  |  | ||||||
|  | This is an ever-growing list of software that is similar to what Tilde Friends tries to be but as far as I can tell don't quite fit the same niche. | ||||||
|  |  | ||||||
|  | - Secure Scuttlebutt Clients | ||||||
|  |   - [Manyverse](https://www.manyver.se/) | ||||||
|  |   - [Patchwork](https://github.com/ssbc/patchwork) | ||||||
|  |   - [Patchfox](https://patchfox.org/#/) | ||||||
|  |   - [Habitat](https://gitlab.com/quickdudley/habitat) | ||||||
|  |   - [Āhau](https://gitlab.com/ahau/ahau/) | ||||||
|  |   - [erlbutt](https://github.com/cmoid/erlbutt/) | ||||||
|  | - Web Application Platforms | ||||||
|  |   - [Glitch](https://glitch.com/) | ||||||
|  |   - [Val Town](https://www.val.town/) | ||||||
|  |   - [Clace](https://clace.io/) | ||||||
							
								
								
									
										23
									
								
								docs/release_checklist.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,23 @@ | |||||||
|  | # Release Checklist | ||||||
|  |  | ||||||
|  | - make sure ci is passing | ||||||
|  | - run the tests | ||||||
|  | - format + prettier | ||||||
|  | - update metadata/en-US/changelogs | ||||||
|  | - git tag | ||||||
|  | - push | ||||||
|  | - make a release on gitea | ||||||
|  | - upload the artifacts | ||||||
|  | - upload the AppImage and zsyncmake | ||||||
|  | - upload to Google | ||||||
|  | - upload to Apple with dist-ios on macos | ||||||
|  | - nix | ||||||
|  |   - june and december: update release version | ||||||
|  |   - run `nix flake update` | ||||||
|  |   - comment out the hash in default.nix | ||||||
|  |   - update the version | ||||||
|  |   - run `nix-build` | ||||||
|  |   - update the hash | ||||||
|  | - bump the versions in GNUmakefile for the next release | ||||||
|  | - make | ||||||
|  | - commit | ||||||
							
								
								
									
										64
									
								
								docs/vision.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,64 @@ | |||||||
|  | # Vision | ||||||
|  |  | ||||||
|  | Tilde Friends is a tool for making and sharing. | ||||||
|  |  | ||||||
|  | It is both a peer-to-peer social network client, participating in Secure | ||||||
|  | Scuttlebutt, and an environment for creating and running web applications. | ||||||
|  |  | ||||||
|  | ## Why | ||||||
|  |  | ||||||
|  | This is a thing that I wanted to exist and wanted to work on. No other reason. | ||||||
|  | There is not a business model. I believe it is interesting and unique. | ||||||
|  |  | ||||||
|  | ## Goals | ||||||
|  |  | ||||||
|  | 1. Make it **easy and fun** to run all sorts of web applications. | ||||||
|  |  | ||||||
|  | 2. Provide **security** that is easy to understand and protects your data. | ||||||
|  |  | ||||||
|  | 3. Make **creating and sharing** web applications accessible to anyone with a | ||||||
|  |    browser. | ||||||
|  |  | ||||||
|  | ## Ways to Use Tilde Friends | ||||||
|  |  | ||||||
|  | 1. **Social Network User**: This is a social network first. You are just here, | ||||||
|  |    because your friends are. Or you like how we limit your message length or | ||||||
|  |    short videos or whatever the trend is. If you are ambitious, you click links | ||||||
|  |    and see interactive experiences (apps) that you wouldn't see elsewhere. | ||||||
|  |  | ||||||
|  | 2. **Web Visitor**: You get links from a friend to meeting invites, polls, games, | ||||||
|  |    lists, wiki pages, ..., and you interact with them as though they were | ||||||
|  |    cloud-hosted by a megacorporation. They just work, and you don't think twice. | ||||||
|  |  | ||||||
|  | 3. **Group leader**: You host or use a small public instance, installing apps for | ||||||
|  |    a group of friends to use as web visitors. | ||||||
|  |  | ||||||
|  | 4. **Developer**: You like to write code and make or improve apps for fun or to | ||||||
|  |    solve problems. When you encounter a Tilde Friends app on a strange server, | ||||||
|  |    you know you can trivially modify it or download it to your own instance. | ||||||
|  |  | ||||||
|  | ## Future Goals / Endgame | ||||||
|  |  | ||||||
|  | 1. Mobile apps. This can run on your old phone. Maybe you won't be hosting | ||||||
|  |    the web interface publicly, but you can sync, install and edit apps, and | ||||||
|  |    otherwise get the full experience from a tiny touch screen. | ||||||
|  |  | ||||||
|  | 2. The universal application runtime. The web browser is the universal | ||||||
|  |    platform, but even for the simplest application that you might want to host | ||||||
|  |    for your friends, cloud hosting, containers, and complicated dependencies might | ||||||
|  |    all enter the mix. Tilde Friends, though it is yet another thing to host, | ||||||
|  |    includes everything you need out of the box to run a vast variety of interesting | ||||||
|  |    apps. | ||||||
|  |  | ||||||
|  |    Tilde Friends will be built out, gradually providing safe access to host | ||||||
|  |    resources and client resources the same way web browsers extended access to | ||||||
|  |    resources like GPU, persistent storage, cameras, ... over the years. | ||||||
|  |  | ||||||
|  |    Not much effort has been put forward yet to having a robust, long-lasting API, | ||||||
|  |    but since the client side longevity is already handled by web browsers, it | ||||||
|  |    seems possible that the server-side API can be managed in a similar way. | ||||||
|  |  | ||||||
|  | 3. An awesome development environment. Right now it runs JavaScript from the | ||||||
|  |    first embeddable text editor I could poorly configure enough to edit code, | ||||||
|  |    but it could incorporate a debugger, source control integration a la ssb-git, | ||||||
|  |    merge tools, and transpiling from all sorts of different languages. | ||||||
							
								
								
									
										14
									
								
								flake.lock
									
									
									
										generated
									
									
									
								
							
							
						
						| @@ -5,11 +5,11 @@ | |||||||
|         "systems": "systems" |         "systems": "systems" | ||||||
|       }, |       }, | ||||||
|       "locked": { |       "locked": { | ||||||
|         "lastModified": 1710146030, |         "lastModified": 1731533236, | ||||||
|         "narHash": "sha256-SZ5L6eA7HJ/nmkzGG7/ISclqe6oZdOZTNoesiInkXPQ=", |         "narHash": "sha256-l0KFg5HjrsfsO/JpG+r7fRrqm12kzFHyUHqHCVpMMbI=", | ||||||
|         "owner": "numtide", |         "owner": "numtide", | ||||||
|         "repo": "flake-utils", |         "repo": "flake-utils", | ||||||
|         "rev": "b1d9ab70662946ef0850d488da1c9019f3a9752a", |         "rev": "11707dc2f618dd54ca8739b309ec4fc024de578b", | ||||||
|         "type": "github" |         "type": "github" | ||||||
|       }, |       }, | ||||||
|       "original": { |       "original": { | ||||||
| @@ -20,16 +20,16 @@ | |||||||
|     }, |     }, | ||||||
|     "nixpkgs": { |     "nixpkgs": { | ||||||
|       "locked": { |       "locked": { | ||||||
|         "lastModified": 1717281328, |         "lastModified": 1739758141, | ||||||
|         "narHash": "sha256-evZPzpf59oNcDUXxh2GHcxHkTEG4fjae2ytWP85jXRo=", |         "narHash": "sha256-uq6A2L7o1/tR6VfmYhZWoVAwb3gTy7j4Jx30MIrH0rE=", | ||||||
|         "owner": "NixOS", |         "owner": "NixOS", | ||||||
|         "repo": "nixpkgs", |         "repo": "nixpkgs", | ||||||
|         "rev": "b3b2b28c1daa04fe2ae47c21bb76fd226eac4ca1", |         "rev": "c618e28f70257593de75a7044438efc1c1fc0791", | ||||||
|         "type": "github" |         "type": "github" | ||||||
|       }, |       }, | ||||||
|       "original": { |       "original": { | ||||||
|         "owner": "NixOS", |         "owner": "NixOS", | ||||||
|         "ref": "nixos-24.05", |         "ref": "nixos-24.11", | ||||||
|         "repo": "nixpkgs", |         "repo": "nixpkgs", | ||||||
|         "type": "github" |         "type": "github" | ||||||
|       } |       } | ||||||
|   | |||||||
							
								
								
									
										24
									
								
								flake.nix
									
									
									
									
									
								
							
							
						
						| @@ -2,7 +2,7 @@ | |||||||
|   description = "Tilde Friends is a platform for making, running, and sharing web applications."; |   description = "Tilde Friends is a platform for making, running, and sharing web applications."; | ||||||
|  |  | ||||||
|   inputs = { |   inputs = { | ||||||
|     nixpkgs.url = "github:NixOS/nixpkgs/nixos-24.05"; |     nixpkgs.url = "github:NixOS/nixpkgs/nixos-24.11"; | ||||||
|     flake-utils.url = "github:numtide/flake-utils"; |     flake-utils.url = "github:numtide/flake-utils"; | ||||||
|   }; |   }; | ||||||
|  |  | ||||||
| @@ -35,5 +35,27 @@ | |||||||
|             graphviz |             graphviz | ||||||
|           ]; |           ]; | ||||||
|         }; |         }; | ||||||
|  |  | ||||||
|  |         nixosModules.default = { | ||||||
|  |           config, | ||||||
|  |           lib, | ||||||
|  |           ... | ||||||
|  |         }: let | ||||||
|  |           # Shorter name to access final settings a | ||||||
|  |           # user of hello.nix module HAS ACTUALLY SET. | ||||||
|  |           # cfg is a typical convention. | ||||||
|  |           cfg = config.services.tildefriends; | ||||||
|  |         in { | ||||||
|  |           options.services.tildefriends = { | ||||||
|  |             enable = lib.mkEnableOption "Enable Tilde Friends"; | ||||||
|  |           }; | ||||||
|  |  | ||||||
|  |           config = lib.mkIf cfg.enable { | ||||||
|  |             systemd.services.tildefriends = { | ||||||
|  |               wantedBy = ["multi-user.target"]; | ||||||
|  |               serviceConfig.ExecStart = "${pkgs.tildefriends}/bin/tildefriends"; | ||||||
|  |             }; | ||||||
|  |           }; | ||||||
|  |         }; | ||||||
|       }); |       }); | ||||||
| } | } | ||||||
|   | |||||||
							
								
								
									
										14
									
								
								metadata/en-US/changelogs/33.txt
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,14 @@ | |||||||
|  | * An iOS build is on TestFlight. | ||||||
|  | * macOS targets are debug and release like everywhere else. | ||||||
|  | * Running from a subdirectory is fine. | ||||||
|  | * Patchwork compatibility improvements. | ||||||
|  | * Invite fixes. | ||||||
|  | * Follow/block UI fixes. | ||||||
|  | * Mobile automatically logs in. | ||||||
|  | * Allow specifying all global settings from the command-line. | ||||||
|  | * UpdateS: | ||||||
|  |   * CodeMirror | ||||||
|  |   * OpenSSL 3.4.1 | ||||||
|  |   * libbacktrace | ||||||
|  |   * sqlite 3.49.1 | ||||||
|  |   * speedscope 1.22.2 | ||||||
							
								
								
									
										6
									
								
								package-lock.json
									
									
									
										generated
									
									
									
								
							
							
						
						| @@ -11,9 +11,9 @@ | |||||||
| 			} | 			} | ||||||
| 		}, | 		}, | ||||||
| 		"node_modules/prettier": { | 		"node_modules/prettier": { | ||||||
| 			"version": "3.4.2", | 			"version": "3.5.1", | ||||||
| 			"resolved": "https://registry.npmjs.org/prettier/-/prettier-3.4.2.tgz", | 			"resolved": "https://registry.npmjs.org/prettier/-/prettier-3.5.1.tgz", | ||||||
| 			"integrity": "sha512-e9MewbtFo+Fevyuxn/4rrcDAaq0IYxPGLvObpQjiZBMAzB9IGmzlnG9RZy3FFas+eBMu2vA0CszMeduow5dIuQ==", | 			"integrity": "sha512-hPpFQvHwL3Qv5AdRvBFMhnKo4tYxp0ReXiPn2bxkiohEX6mBeBwEpBSQTkD458RaaDKQMYSp4hX4UtfUTA5wDw==", | ||||||
| 			"bin": { | 			"bin": { | ||||||
| 				"prettier": "bin/prettier.cjs" | 				"prettier": "bin/prettier.cjs" | ||||||
| 			}, | 			}, | ||||||
|   | |||||||
| @@ -1,8 +1,8 @@ | |||||||
| <?xml version="1.0" encoding="utf-8"?> | <?xml version="1.0" encoding="utf-8"?> | ||||||
| <manifest xmlns:android="http://schemas.android.com/apk/res/android" | <manifest xmlns:android="http://schemas.android.com/apk/res/android" | ||||||
| 	package="com.unprompted.tildefriends" | 	package="com.unprompted.tildefriends" | ||||||
| 	android:versionCode="32" | 	android:versionCode="33" | ||||||
| 	android:versionName="0.0.27"> | 	android:versionName="0.0.28-wip"> | ||||||
| 	<uses-permission android:name="android.permission.ACCESS_NETWORK_STATE"/> | 	<uses-permission android:name="android.permission.ACCESS_NETWORK_STATE"/> | ||||||
| 	<uses-permission android:name="android.permission.INTERNET"/> | 	<uses-permission android:name="android.permission.INTERNET"/> | ||||||
| 	<application | 	<application | ||||||
|   | |||||||
| @@ -421,7 +421,7 @@ public class TildeFriendsActivity extends Activity { | |||||||
| 			base_url = "http://127.0.0.1:" + String.valueOf(port) + "/"; | 			base_url = "http://127.0.0.1:" + String.valueOf(port) + "/"; | ||||||
| 			runOnUiThread(() -> { | 			runOnUiThread(() -> { | ||||||
| 				hide_status(); | 				hide_status(); | ||||||
| 				web_view.loadUrl(base_url); | 				web_view.loadUrl(base_url + "login/auto"); | ||||||
| 			}); | 			}); | ||||||
| 			observer = null; | 			observer = null; | ||||||
| 		} else { | 		} else { | ||||||
|   | |||||||
| @@ -1,55 +1,16 @@ | |||||||
| <vector xmlns:android="http://schemas.android.com/apk/res/android" | <?xml version="1.0" encoding="utf-8"?> | ||||||
|     android:width="65dp" | <layer-list xmlns:android="http://schemas.android.com/apk/res/android"> | ||||||
|     android:height="65dp" |     <item | ||||||
|     android:viewportWidth="61" |         android:gravity="center" | ||||||
|     android:viewportHeight="65"> |         android:width="256dp" | ||||||
|   <path |         android:height="256dp"> | ||||||
|       android:pathData="M6,0h49a8,8 45,0 1,8 8v49a8,8 135,0 1,-8 8H6a8,8 45,0 1,-8 -8V8a8,8 135,0 1,8 -8Z" |         <shape> | ||||||
|       android:strokeWidth=".712717" |             <solid | ||||||
|       android:fillColor="#0af" |                 android:color="#0af"/> | ||||||
|       android:fillAlpha="1"/> |         </shape> | ||||||
|   <path |     </item> | ||||||
|       android:pathData="m1.6762,36.6891v-4.0039q2.0703,-2.3438 5.4297,-2.3438 1.1719,0 2.4609,0.3516 1.2891,0.332 3.6719,1.3477 1.3477,0.5664 2.0117,0.7422 0.6836,0.1758 1.3672,0.1758 1.2695,0 2.6172,-0.7617 1.3672,-0.7617 2.4219,-1.9141v4.1406q-1.25,1.1719 -2.5391,1.6992 -1.2695,0.5273 -2.8711,0.5273 -1.1719,0 -2.2461,-0.2734 -1.0547,-0.2734 -3.3789,-1.3086 -2.3047,-1.0352 -3.8477,-1.0352 -1.25,0 -2.3633,0.5469 -1.0938,0.5273 -2.7344,2.1094z" |     <item android:drawable="@drawable/logo" | ||||||
|       android:fillColor="#000000"/> |         android:width="65dp" | ||||||
|   <path |         android:height="65dp" | ||||||
|       android:pathData="M42.4653,32.2273m-16.7723,0a16.7723,16.7723 0,1 1,33.5446 0a16.7723,16.7723 0,1 1,-33.5446 0" |         android:gravity="center"/> | ||||||
|       android:fillColor="#fcea2b"/> | </layer-list> | ||||||
|   <path |  | ||||||
|       android:pathData="M49.2697,34.097c2.8899,0 5.2344,-2.0871 5.2344,-4.6591 0,-1.2871 0.3267,-2.5742 -0.6213,-3.4164 -0.9473,-0.843 -3.1685,-1.2426 -4.6131,-1.2426 -1.7188,0 -3.7504,0.1043 -4.7043,1.2426 -0.6519,0.7766 -0.5302,2.3722 -0.5302,3.4164 0,2.572 2.343,4.6591 5.2344,4.6591zM34.9819,34.097c2.8899,0 5.2351,-2.0871 5.2351,-4.6591 0,-1.2871 0.326,-2.5742 -0.6213,-3.4164 -0.948,-0.843 -3.1685,-1.2426 -4.6138,-1.2426 -1.7181,0 -3.7497,0.1043 -4.7043,1.2426 -0.6512,0.7766 -0.5302,2.3722 -0.5302,3.4164 0,2.572 2.343,4.6591 5.2344,4.6591z" |  | ||||||
|       android:fillColor="#3f3f3f"/> |  | ||||||
|   <path |  | ||||||
|       android:pathData="M42.3829,32.2681m-16.7723,0a16.7723,16.7723 0,1 1,33.5446 0a16.7723,16.7723 0,1 1,-33.5446 0" |  | ||||||
|       android:strokeLineJoin="round" |  | ||||||
|       android:strokeWidth="2" |  | ||||||
|       android:fillColor="#00000000" |  | ||||||
|       android:strokeColor="#000" |  | ||||||
|       android:strokeLineCap="round"/> |  | ||||||
|   <path |  | ||||||
|       android:pathData="M49.5403,38.6897c-4.794,2.5705 -10.242,2.6675 -14.3148,0M29.983,28.1903s-0.695,6.2349 5.0025,5.774c1.9106,-0.1546 5.7004,-0.474 5.7369,-6.0832 0.0036,-0.509 -0.0051,-1.1668 -0.5907,-1.9179 -0.7766,-0.9969 -2.6048,-1.4373 -7.2522,-1.037 0,0 -2.5129,-0.0729 -2.8965,3.264z" |  | ||||||
|       android:strokeLineJoin="round" |  | ||||||
|       android:strokeWidth="2" |  | ||||||
|       android:fillColor="#00000000" |  | ||||||
|       android:strokeColor="#000" |  | ||||||
|       android:strokeLineCap="round"/> |  | ||||||
|   <path |  | ||||||
|       android:pathData="m30.0341,27.8016 l-0.3158,-2.459 2.7951,-0.3843M54.6733,28.1903s0.695,6.2349 -5.0025,5.774c-1.9106,-0.1546 -5.7004,-0.474 -5.7376,-6.0832 -0.0029,-0.509 0.0058,-1.1668 0.5914,-1.9179 0.7766,-0.9969 2.6048,-1.4373 7.2522,-1.037 0,0 2.5129,-0.0729 2.8965,3.264z" |  | ||||||
|       android:strokeLineJoin="round" |  | ||||||
|       android:strokeWidth="2" |  | ||||||
|       android:fillColor="#00000000" |  | ||||||
|       android:strokeColor="#000" |  | ||||||
|       android:strokeLineCap="round"/> |  | ||||||
|   <path |  | ||||||
|       android:pathData="M39.1874,25.2383s3.0073,1.8479 6.3129,0M40.6685,28.813s1.6058,-2.7353 3.3078,0M54.6172,27.803l0.3158,-2.4582 -2.7951,-0.385" |  | ||||||
|       android:strokeLineJoin="round" |  | ||||||
|       android:strokeWidth="2" |  | ||||||
|       android:fillColor="#00000000" |  | ||||||
|       android:strokeColor="#000" |  | ||||||
|       android:strokeLineCap="round"/> |  | ||||||
|   <path |  | ||||||
|       android:pathData="M40.974,27.8716s1.309,-2.7346 2.6974,0" |  | ||||||
|       android:strokeLineJoin="round" |  | ||||||
|       android:strokeWidth="2" |  | ||||||
|       android:fillColor="#00000000" |  | ||||||
|       android:strokeColor="#000" |  | ||||||
|       android:strokeLineCap="round"/> |  | ||||||
| </vector> |  | ||||||
|   | |||||||
							
								
								
									
										16
									
								
								src/android/res/drawable/icon_fdroid.xml
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,16 @@ | |||||||
|  | <?xml version="1.0" encoding="utf-8"?> | ||||||
|  | <layer-list xmlns:android="http://schemas.android.com/apk/res/android"> | ||||||
|  |     <item | ||||||
|  |         android:gravity="center" | ||||||
|  |         android:width="256dp" | ||||||
|  |         android:height="256dp"> | ||||||
|  |         <shape> | ||||||
|  |             <solid | ||||||
|  |                 android:color="#0fa"/> | ||||||
|  |         </shape> | ||||||
|  |     </item> | ||||||
|  |     <item android:drawable="@drawable/logo" | ||||||
|  |         android:width="65dp" | ||||||
|  |         android:height="65dp" | ||||||
|  |         android:gravity="center"/> | ||||||
|  | </layer-list> | ||||||
							
								
								
									
										51
									
								
								src/android/res/drawable/logo.xml
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,51 @@ | |||||||
|  | <?xml version="1.0" encoding="utf-8"?> | ||||||
|  | <vector xmlns:android="http://schemas.android.com/apk/res/android" | ||||||
|  |     android:width="65dp" | ||||||
|  |     android:height="65dp" | ||||||
|  |     android:viewportWidth="65" | ||||||
|  |     android:viewportHeight="65"> | ||||||
|  |   <path | ||||||
|  |       android:pathData="m1.6762,36.6891v-4.0039q2.0703,-2.3438 5.4297,-2.3438 1.1719,0 2.4609,0.3516 1.2891,0.332 3.6719,1.3477 1.3477,0.5664 2.0117,0.7422 0.6836,0.1758 1.3672,0.1758 1.2695,0 2.6172,-0.7617 1.3672,-0.7617 2.4219,-1.9141v4.1406q-1.25,1.1719 -2.5391,1.6992 -1.2695,0.5273 -2.8711,0.5273 -1.1719,0 -2.2461,-0.2734 -1.0547,-0.2734 -3.3789,-1.3086 -2.3047,-1.0352 -3.8477,-1.0352 -1.25,0 -2.3633,0.5469 -1.0938,0.5273 -2.7344,2.1094z" | ||||||
|  |       android:fillColor="#000000"/> | ||||||
|  |   <path | ||||||
|  |       android:pathData="M42.4653,32.2273m-16.7723,0a16.7723,16.7723 0,1 1,33.5446 0a16.7723,16.7723 0,1 1,-33.5446 0" | ||||||
|  |       android:fillColor="#fcea2b"/> | ||||||
|  |   <path | ||||||
|  |       android:pathData="M49.2697,34.097c2.8899,0 5.2344,-2.0871 5.2344,-4.6591 0,-1.2871 0.3267,-2.5742 -0.6213,-3.4164 -0.9473,-0.843 -3.1685,-1.2426 -4.6131,-1.2426 -1.7188,0 -3.7504,0.1043 -4.7043,1.2426 -0.6519,0.7766 -0.5302,2.3722 -0.5302,3.4164 0,2.572 2.343,4.6591 5.2344,4.6591zM34.9819,34.097c2.8899,0 5.2351,-2.0871 5.2351,-4.6591 0,-1.2871 0.326,-2.5742 -0.6213,-3.4164 -0.948,-0.843 -3.1685,-1.2426 -4.6138,-1.2426 -1.7181,0 -3.7497,0.1043 -4.7043,1.2426 -0.6512,0.7766 -0.5302,2.3722 -0.5302,3.4164 0,2.572 2.343,4.6591 5.2344,4.6591z" | ||||||
|  |       android:fillColor="#3f3f3f"/> | ||||||
|  |   <path | ||||||
|  |       android:pathData="M42.3829,32.2681m-16.7723,0a16.7723,16.7723 0,1 1,33.5446 0a16.7723,16.7723 0,1 1,-33.5446 0" | ||||||
|  |       android:strokeLineJoin="round" | ||||||
|  |       android:strokeWidth="2" | ||||||
|  |       android:fillColor="#00000000" | ||||||
|  |       android:strokeColor="#000" | ||||||
|  |       android:strokeLineCap="round"/> | ||||||
|  |   <path | ||||||
|  |       android:pathData="M49.5403,38.6897c-4.794,2.5705 -10.242,2.6675 -14.3148,0M29.983,28.1903s-0.695,6.2349 5.0025,5.774c1.9106,-0.1546 5.7004,-0.474 5.7369,-6.0832 0.0036,-0.509 -0.0051,-1.1668 -0.5907,-1.9179 -0.7766,-0.9969 -2.6048,-1.4373 -7.2522,-1.037 0,0 -2.5129,-0.0729 -2.8965,3.264z" | ||||||
|  |       android:strokeLineJoin="round" | ||||||
|  |       android:strokeWidth="2" | ||||||
|  |       android:fillColor="#00000000" | ||||||
|  |       android:strokeColor="#000" | ||||||
|  |       android:strokeLineCap="round"/> | ||||||
|  |   <path | ||||||
|  |       android:pathData="m30.0341,27.8016 l-0.3158,-2.459 2.7951,-0.3843M54.6733,28.1903s0.695,6.2349 -5.0025,5.774c-1.9106,-0.1546 -5.7004,-0.474 -5.7376,-6.0832 -0.0029,-0.509 0.0058,-1.1668 0.5914,-1.9179 0.7766,-0.9969 2.6048,-1.4373 7.2522,-1.037 0,0 2.5129,-0.0729 2.8965,3.264z" | ||||||
|  |       android:strokeLineJoin="round" | ||||||
|  |       android:strokeWidth="2" | ||||||
|  |       android:fillColor="#00000000" | ||||||
|  |       android:strokeColor="#000" | ||||||
|  |       android:strokeLineCap="round"/> | ||||||
|  |   <path | ||||||
|  |       android:pathData="M39.1874,25.2383s3.0073,1.8479 6.3129,0M40.6685,28.813s1.6058,-2.7353 3.3078,0M54.6172,27.803l0.3158,-2.4582 -2.7951,-0.385" | ||||||
|  |       android:strokeLineJoin="round" | ||||||
|  |       android:strokeWidth="2" | ||||||
|  |       android:fillColor="#00000000" | ||||||
|  |       android:strokeColor="#000" | ||||||
|  |       android:strokeLineCap="round"/> | ||||||
|  |   <path | ||||||
|  |       android:pathData="M40.974,27.8716s1.309,-2.7346 2.6974,0" | ||||||
|  |       android:strokeLineJoin="round" | ||||||
|  |       android:strokeWidth="2" | ||||||
|  |       android:fillColor="#00000000" | ||||||
|  |       android:strokeColor="#000" | ||||||
|  |       android:strokeLineCap="round"/> | ||||||
|  | </vector> | ||||||
							
								
								
									
										198
									
								
								src/file.js.c
									
									
									
									
									
								
							
							
						
						| @@ -18,20 +18,6 @@ | |||||||
|  |  | ||||||
| static JSValue _file_read_file(JSContext* context, JSValueConst this_val, int argc, JSValueConst* argv); | static JSValue _file_read_file(JSContext* context, JSValueConst this_val, int argc, JSValueConst* argv); | ||||||
| static JSValue _file_read_file_zip(JSContext* context, JSValueConst this_val, int argc, JSValueConst* argv); | static JSValue _file_read_file_zip(JSContext* context, JSValueConst this_val, int argc, JSValueConst* argv); | ||||||
| static JSValue _file_stat(JSContext* context, JSValueConst this_val, int argc, JSValueConst* argv); |  | ||||||
| static JSValue _file_stat_zip(JSContext* context, JSValueConst this_val, int argc, JSValueConst* argv); |  | ||||||
| static JSValue _file_write_file(JSContext* context, JSValueConst this_val, int argc, JSValueConst* argv); |  | ||||||
|  |  | ||||||
| static double _time_spec_to_double(const uv_timespec_t* time_spec); |  | ||||||
| static void _file_on_stat_complete(uv_fs_t* request); |  | ||||||
|  |  | ||||||
| typedef struct file_stat_t |  | ||||||
| { |  | ||||||
| 	void* _task; |  | ||||||
| 	JSContext* _context; |  | ||||||
| 	promiseid_t _promise; |  | ||||||
| 	uv_fs_t _request; |  | ||||||
| } file_stat_t; |  | ||||||
|  |  | ||||||
| typedef struct fs_req_t | typedef struct fs_req_t | ||||||
| { | { | ||||||
| @@ -45,12 +31,10 @@ void tf_file_register(JSContext* context) | |||||||
| { | { | ||||||
| 	JSValue global = JS_GetGlobalObject(context); | 	JSValue global = JS_GetGlobalObject(context); | ||||||
| 	JSValue file = JS_NewObject(context); | 	JSValue file = JS_NewObject(context); | ||||||
| 	void* task = JS_GetContextOpaque(context); | 	tf_task_t* task = JS_GetContextOpaque(context); | ||||||
| 	const char* zip = tf_task_get_zip_path(task); | 	const char* zip = tf_task_get_zip_path(task); | ||||||
| 	JS_SetPropertyStr(context, global, "File", file); | 	JS_SetPropertyStr(context, global, "File", file); | ||||||
| 	JS_SetPropertyStr(context, file, "readFile", JS_NewCFunction(context, zip ? _file_read_file_zip : _file_read_file, "readFile", 1)); | 	JS_SetPropertyStr(context, file, "readFile", JS_NewCFunction(context, zip ? _file_read_file_zip : _file_read_file, "readFile", 1)); | ||||||
| 	JS_SetPropertyStr(context, file, "writeFile", JS_NewCFunction(context, _file_write_file, "writeFile", 2)); |  | ||||||
| 	JS_SetPropertyStr(context, file, "stat", JS_NewCFunction(context, zip ? _file_stat_zip : _file_stat, "stat", 1)); |  | ||||||
| 	JS_FreeValue(context, global); | 	JS_FreeValue(context, global); | ||||||
| } | } | ||||||
|  |  | ||||||
| @@ -65,105 +49,6 @@ static void _file_async_close_callback(uv_fs_t* req) | |||||||
| 	tf_free(req); | 	tf_free(req); | ||||||
| } | } | ||||||
|  |  | ||||||
| static void _file_write_write_callback(uv_fs_t* req) |  | ||||||
| { |  | ||||||
| 	uv_fs_req_cleanup(req); |  | ||||||
| 	fs_req_t* fsreq = (fs_req_t*)req; |  | ||||||
| 	tf_task_t* task = req->loop->data; |  | ||||||
| 	JSContext* context = tf_task_get_context(task); |  | ||||||
| 	promiseid_t promise = (promiseid_t)(intptr_t)req->data; |  | ||||||
| 	if (req->result >= 0) |  | ||||||
| 	{ |  | ||||||
| 		tf_task_resolve_promise(task, promise, JS_NewInt64(context, req->result)); |  | ||||||
| 	} |  | ||||||
| 	else |  | ||||||
| 	{ |  | ||||||
| 		tf_task_reject_promise(task, promise, JS_ThrowInternalError(context, "Failed to write %s: %s", req->path, uv_strerror(req->result))); |  | ||||||
| 	} |  | ||||||
| 	int result = uv_fs_close(req->loop, req, fsreq->file, _file_async_close_callback); |  | ||||||
| 	if (result < 0) |  | ||||||
| 	{ |  | ||||||
| 		uv_fs_req_cleanup(req); |  | ||||||
| 		tf_free(fsreq); |  | ||||||
| 	} |  | ||||||
| } |  | ||||||
|  |  | ||||||
| static void _file_write_open_callback(uv_fs_t* req) |  | ||||||
| { |  | ||||||
| 	fs_req_t* fsreq = (fs_req_t*)req; |  | ||||||
| 	const char* path = tf_strdup(req->path); |  | ||||||
| 	uv_fs_req_cleanup(req); |  | ||||||
| 	tf_task_t* task = req->loop->data; |  | ||||||
| 	JSContext* context = tf_task_get_context(task); |  | ||||||
| 	promiseid_t promise = (promiseid_t)(intptr_t)req->data; |  | ||||||
| 	if (req->result >= 0) |  | ||||||
| 	{ |  | ||||||
| 		uv_buf_t buf = { .base = fsreq->buffer, .len = fsreq->size }; |  | ||||||
| 		fsreq->file = req->result; |  | ||||||
| 		int result = uv_fs_write(req->loop, req, fsreq->file, &buf, 1, 0, _file_write_write_callback); |  | ||||||
| 		if (result < 0) |  | ||||||
| 		{ |  | ||||||
| 			uv_fs_req_cleanup(req); |  | ||||||
| 			tf_task_reject_promise(task, promise, JS_ThrowInternalError(context, "Failed to write %s: %s", path, uv_strerror(result))); |  | ||||||
| 			result = uv_fs_close(req->loop, req, fsreq->file, _file_async_close_callback); |  | ||||||
| 			if (result < 0) |  | ||||||
| 			{ |  | ||||||
| 				uv_fs_req_cleanup(req); |  | ||||||
| 				tf_free(fsreq); |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
| 	else |  | ||||||
| 	{ |  | ||||||
| 		tf_task_reject_promise(task, promise, JS_ThrowInternalError(context, "%s", uv_strerror(req->result))); |  | ||||||
| 		tf_free(req); |  | ||||||
| 	} |  | ||||||
| 	tf_free((void*)path); |  | ||||||
| } |  | ||||||
|  |  | ||||||
| static JSValue _file_write_file(JSContext* context, JSValueConst this_val, int argc, JSValueConst* argv) |  | ||||||
| { |  | ||||||
| 	void* task = JS_GetContextOpaque(context); |  | ||||||
| 	const char* file_name = JS_ToCString(context, argv[0]); |  | ||||||
|  |  | ||||||
| 	size_t size; |  | ||||||
| 	uint8_t* buffer = tf_util_try_get_array_buffer(context, &size, argv[1]); |  | ||||||
| 	bool is_array_buffer = false; |  | ||||||
| 	if (buffer) |  | ||||||
| 	{ |  | ||||||
| 		is_array_buffer = true; |  | ||||||
| 	} |  | ||||||
| 	else |  | ||||||
| 	{ |  | ||||||
| 		buffer = (uint8_t*)JS_ToCStringLen(context, &size, argv[1]); |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	promiseid_t promise = -1; |  | ||||||
| 	JSValue promise_value = tf_task_allocate_promise(task, &promise); |  | ||||||
| 	fs_req_t* req = tf_malloc(sizeof(fs_req_t) + size); |  | ||||||
| 	*req = (fs_req_t) |  | ||||||
| 	{ |  | ||||||
| 		.fs = |  | ||||||
| 		{ |  | ||||||
| 			.data = (void*)(intptr_t)promise, |  | ||||||
| 		}, |  | ||||||
| 		.size = size, |  | ||||||
| 	}; |  | ||||||
| 	memcpy(req->buffer, buffer, size); |  | ||||||
| 	if (!is_array_buffer) |  | ||||||
| 	{ |  | ||||||
| 		JS_FreeCString(context, (const char*)buffer); |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	int result = uv_fs_open(tf_task_get_loop(task), &req->fs, file_name, UV_FS_O_CREAT | UV_FS_O_WRONLY, 0644, _file_write_open_callback); |  | ||||||
| 	if (result < 0) |  | ||||||
| 	{ |  | ||||||
| 		tf_task_reject_promise(task, promise, JS_ThrowInternalError(context, "Failed to open %s for write: %s", file_name, uv_strerror(result))); |  | ||||||
| 	} |  | ||||||
| 	JS_FreeCString(context, file_name); |  | ||||||
| 	return promise_value; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| static void _file_read_read_callback(uv_fs_t* req) | static void _file_read_read_callback(uv_fs_t* req) | ||||||
| { | { | ||||||
| 	fs_req_t* fsreq = (fs_req_t*)req; | 	fs_req_t* fsreq = (fs_req_t*)req; | ||||||
| @@ -224,8 +109,9 @@ static void _file_read_open_callback(uv_fs_t* req) | |||||||
|  |  | ||||||
| static JSValue _file_read_file(JSContext* context, JSValueConst this_val, int argc, JSValueConst* argv) | static JSValue _file_read_file(JSContext* context, JSValueConst this_val, int argc, JSValueConst* argv) | ||||||
| { | { | ||||||
| 	void* task = JS_GetContextOpaque(context); | 	tf_task_t* task = JS_GetContextOpaque(context); | ||||||
| 	const char* file_name = JS_ToCString(context, argv[0]); | 	const char* file_name = JS_ToCString(context, argv[0]); | ||||||
|  | 	const char* actual = tf_task_get_path_with_root(task, file_name); | ||||||
|  |  | ||||||
| 	promiseid_t promise = -1; | 	promiseid_t promise = -1; | ||||||
| 	JSValue promise_value = tf_task_allocate_promise(task, &promise); | 	JSValue promise_value = tf_task_allocate_promise(task, &promise); | ||||||
| @@ -239,7 +125,7 @@ static JSValue _file_read_file(JSContext* context, JSValueConst this_val, int ar | |||||||
| 		.size = k_file_read_max, | 		.size = k_file_read_max, | ||||||
| 	}; | 	}; | ||||||
| 	memset(req + 1, 0, k_file_read_max); | 	memset(req + 1, 0, k_file_read_max); | ||||||
| 	int result = uv_fs_open(tf_task_get_loop(task), &req->fs, file_name, UV_FS_O_RDONLY, 0, _file_read_open_callback); | 	int result = uv_fs_open(tf_task_get_loop(task), &req->fs, actual, UV_FS_O_RDONLY, 0, _file_read_open_callback); | ||||||
| 	if (result < 0) | 	if (result < 0) | ||||||
| 	{ | 	{ | ||||||
| 		tf_task_reject_promise(task, promise, JS_ThrowInternalError(context, "Failed to open %s for read: %s", file_name, uv_strerror(result))); | 		tf_task_reject_promise(task, promise, JS_ThrowInternalError(context, "Failed to open %s for read: %s", file_name, uv_strerror(result))); | ||||||
| @@ -247,6 +133,7 @@ static JSValue _file_read_file(JSContext* context, JSValueConst this_val, int ar | |||||||
| 		tf_free(req); | 		tf_free(req); | ||||||
| 	} | 	} | ||||||
| 	JS_FreeCString(context, file_name); | 	JS_FreeCString(context, file_name); | ||||||
|  | 	tf_free((char*)actual); | ||||||
| 	return promise_value; | 	return promise_value; | ||||||
| } | } | ||||||
|  |  | ||||||
| @@ -365,81 +252,6 @@ static JSValue _file_read_file_zip(JSContext* context, JSValueConst this_val, in | |||||||
| 	return promise_value; | 	return promise_value; | ||||||
| } | } | ||||||
|  |  | ||||||
| static JSValue _file_stat(JSContext* context, JSValueConst this_val, int argc, JSValueConst* argv) |  | ||||||
| { |  | ||||||
| 	void* task = JS_GetContextOpaque(context); |  | ||||||
| 	const char* path = JS_ToCString(context, argv[0]); |  | ||||||
| 	promiseid_t promise = -1; |  | ||||||
| 	JSValue promise_value = tf_task_allocate_promise(task, &promise); |  | ||||||
|  |  | ||||||
| 	file_stat_t* data = tf_malloc(sizeof(file_stat_t)); |  | ||||||
| 	data->_task = task; |  | ||||||
| 	data->_promise = promise; |  | ||||||
| 	data->_request.data = data; |  | ||||||
| 	data->_context = context; |  | ||||||
|  |  | ||||||
| 	int result = uv_fs_stat(tf_task_get_loop(task), &data->_request, path, _file_on_stat_complete); |  | ||||||
| 	if (result) |  | ||||||
| 	{ |  | ||||||
| 		tf_task_reject_promise(task, promise, JS_NewInt32(context, result)); |  | ||||||
| 		uv_fs_req_cleanup(&data->_request); |  | ||||||
| 		tf_free(data); |  | ||||||
| 	} |  | ||||||
| 	JS_FreeCString(context, path); |  | ||||||
| 	return promise_value; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| static JSValue _file_stat_zip(JSContext* context, JSValueConst this_val, int argc, JSValueConst* argv) |  | ||||||
| { |  | ||||||
| 	void* task = JS_GetContextOpaque(context); |  | ||||||
| 	promiseid_t promise = -1; |  | ||||||
| 	JSValue promise_value = tf_task_allocate_promise(task, &promise); |  | ||||||
|  |  | ||||||
| 	file_stat_t* data = tf_malloc(sizeof(file_stat_t)); |  | ||||||
| 	data->_task = task; |  | ||||||
| 	data->_promise = promise; |  | ||||||
| 	data->_request.data = data; |  | ||||||
| 	data->_context = context; |  | ||||||
|  |  | ||||||
| 	/* Ignore the requested path and stat the zip itself. */ |  | ||||||
| 	int result = uv_fs_stat(tf_task_get_loop(task), &data->_request, tf_task_get_zip_path(task), _file_on_stat_complete); |  | ||||||
| 	if (result) |  | ||||||
| 	{ |  | ||||||
| 		tf_task_reject_promise(task, promise, JS_NewInt32(context, result)); |  | ||||||
| 		uv_fs_req_cleanup(&data->_request); |  | ||||||
| 		tf_free(data); |  | ||||||
| 	} |  | ||||||
| 	return promise_value; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| static double _time_spec_to_double(const uv_timespec_t* time_spec) |  | ||||||
| { |  | ||||||
| 	return time_spec->tv_sec + (double)(time_spec->tv_nsec) / 1e9; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| static void _file_on_stat_complete(uv_fs_t* request) |  | ||||||
| { |  | ||||||
| 	file_stat_t* data = (file_stat_t*)(request->data); |  | ||||||
| 	JSContext* context = data->_context; |  | ||||||
|  |  | ||||||
| 	if (request->result) |  | ||||||
| 	{ |  | ||||||
| 		tf_task_reject_promise(data->_task, data->_promise, JS_NewInt32(context, request->result)); |  | ||||||
| 	} |  | ||||||
| 	else |  | ||||||
| 	{ |  | ||||||
| 		JSValue result = JS_NewObject(context); |  | ||||||
| 		JS_SetPropertyStr(context, result, "mtime", JS_NewFloat64(context, _time_spec_to_double(&request->statbuf.st_mtim))); |  | ||||||
| 		JS_SetPropertyStr(context, result, "ctime", JS_NewFloat64(context, _time_spec_to_double(&request->statbuf.st_ctim))); |  | ||||||
| 		JS_SetPropertyStr(context, result, "atime", JS_NewFloat64(context, _time_spec_to_double(&request->statbuf.st_atim))); |  | ||||||
| 		JS_SetPropertyStr(context, result, "size", JS_NewFloat64(context, request->statbuf.st_size)); |  | ||||||
| 		tf_task_resolve_promise(data->_task, data->_promise, result); |  | ||||||
| 		JS_FreeValue(context, result); |  | ||||||
| 	} |  | ||||||
| 	uv_fs_req_cleanup(request); |  | ||||||
| 	tf_free(data); |  | ||||||
| } |  | ||||||
|  |  | ||||||
| typedef struct _stat_t | typedef struct _stat_t | ||||||
| { | { | ||||||
| 	uv_fs_t request; | 	uv_fs_t request; | ||||||
|   | |||||||
							
								
								
									
										381
									
								
								src/httpd.js.c
									
									
									
									
									
								
							
							
						
						| @@ -7,6 +7,7 @@ | |||||||
| #include "ssb.db.h" | #include "ssb.db.h" | ||||||
| #include "ssb.h" | #include "ssb.h" | ||||||
| #include "task.h" | #include "task.h" | ||||||
|  | #include "tls.h" | ||||||
| #include "tlscontext.js.h" | #include "tlscontext.js.h" | ||||||
| #include "trace.h" | #include "trace.h" | ||||||
| #include "util.js.h" | #include "util.js.h" | ||||||
| @@ -233,46 +234,6 @@ static JSValue _httpd_make_response_object(JSContext* context, tf_http_request_t | |||||||
| 	return response_object; | 	return response_object; | ||||||
| } | } | ||||||
|  |  | ||||||
| static void _httpd_callback_internal(tf_http_request_t* request, bool is_websocket) |  | ||||||
| { |  | ||||||
| 	http_handler_data_t* data = request->user_data; |  | ||||||
| 	JSContext* context = data->context; |  | ||||||
| 	JSValue request_object = JS_NewObject(context); |  | ||||||
| 	JS_SetPropertyStr(context, request_object, "method", JS_NewString(context, request->method)); |  | ||||||
| 	JS_SetPropertyStr(context, request_object, "uri", JS_NewString(context, request->path)); |  | ||||||
| 	JSValue headers = JS_NewObject(context); |  | ||||||
| 	for (int i = 0; i < request->headers_count; i++) |  | ||||||
| 	{ |  | ||||||
| 		JS_SetPropertyStr(context, headers, request->headers[i].name, JS_NewString(context, request->headers[i].value)); |  | ||||||
| 	} |  | ||||||
| 	JS_SetPropertyStr(context, request_object, "headers", headers); |  | ||||||
| 	if (request->query) |  | ||||||
| 	{ |  | ||||||
| 		JS_SetPropertyStr(context, request_object, "query", JS_NewString(context, request->query)); |  | ||||||
| 	} |  | ||||||
| 	if (request->body) |  | ||||||
| 	{ |  | ||||||
| 		JS_SetPropertyStr(context, request_object, "body", tf_util_new_uint8_array(context, request->body, request->content_length)); |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	JSValue client = JS_NewObject(context); |  | ||||||
| 	JS_SetPropertyStr(context, client, "tls", request->is_tls ? JS_TRUE : JS_FALSE); |  | ||||||
| 	JS_SetPropertyStr(context, request_object, "client", client); |  | ||||||
|  |  | ||||||
| 	JSValue response_object = _httpd_make_response_object(context, request); |  | ||||||
| 	/* The ref is owned by the JS object and will be released by the finalizer. */ |  | ||||||
| 	tf_http_request_ref(request); |  | ||||||
| 	JSValue args[] = { |  | ||||||
| 		request_object, |  | ||||||
| 		response_object, |  | ||||||
| 	}; |  | ||||||
| 	JSValue response = JS_Call(context, data->callback, JS_UNDEFINED, 2, args); |  | ||||||
| 	tf_util_report_error(context, response); |  | ||||||
| 	JS_FreeValue(context, request_object); |  | ||||||
| 	JS_FreeValue(context, response); |  | ||||||
| 	JS_FreeValue(context, response_object); |  | ||||||
| } |  | ||||||
|  |  | ||||||
| static bool _httpd_redirect(tf_http_request_t* request) | static bool _httpd_redirect(tf_http_request_t* request) | ||||||
| { | { | ||||||
| 	if (request->is_tls) | 	if (request->is_tls) | ||||||
| @@ -292,16 +253,6 @@ static bool _httpd_redirect(tf_http_request_t* request) | |||||||
| 	return true; | 	return true; | ||||||
| } | } | ||||||
|  |  | ||||||
| static void _httpd_callback(tf_http_request_t* request) |  | ||||||
| { |  | ||||||
| 	if (_httpd_redirect(request)) |  | ||||||
| 	{ |  | ||||||
| 		return; |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	_httpd_callback_internal(request, false); |  | ||||||
| } |  | ||||||
|  |  | ||||||
| static JSValue _httpd_websocket_upgrade(JSContext* context, JSValueConst this_val, int argc, JSValueConst* argv) | static JSValue _httpd_websocket_upgrade(JSContext* context, JSValueConst this_val, int argc, JSValueConst* argv) | ||||||
| { | { | ||||||
| 	tf_http_request_t* request = JS_GetOpaque(this_val, _httpd_request_class_id); | 	tf_http_request_t* request = JS_GetOpaque(this_val, _httpd_request_class_id); | ||||||
| @@ -388,71 +339,19 @@ static JSValue _httpd_websocket_upgrade(JSContext* context, JSValueConst this_va | |||||||
| 	return JS_UNDEFINED; | 	return JS_UNDEFINED; | ||||||
| } | } | ||||||
|  |  | ||||||
| static void _httpd_cleanup_callback(void* user_data) |  | ||||||
| { |  | ||||||
| 	http_handler_data_t* data = user_data; |  | ||||||
| 	JS_FreeValue(data->context, data->callback); |  | ||||||
| 	tf_free(data); |  | ||||||
| } |  | ||||||
|  |  | ||||||
| static JSValue _httpd_endpoint_all(JSContext* context, JSValueConst this_val, int argc, JSValueConst* argv) |  | ||||||
| { |  | ||||||
| 	tf_http_t* http = JS_GetOpaque(this_val, _httpd_class_id); |  | ||||||
| 	const char* pattern = JS_ToCString(context, argv[0]); |  | ||||||
| 	http_handler_data_t* data = tf_malloc(sizeof(http_handler_data_t)); |  | ||||||
| 	*data = (http_handler_data_t) { .context = context, .callback = JS_DupValue(context, argv[1]) }; |  | ||||||
| 	tf_http_add_handler(http, pattern, _httpd_callback, _httpd_cleanup_callback, data); |  | ||||||
| 	JS_FreeCString(context, pattern); |  | ||||||
| 	return JS_UNDEFINED; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| typedef struct _httpd_listener_t | typedef struct _httpd_listener_t | ||||||
| { | { | ||||||
| 	JSContext* context; | 	tf_tls_context_t* tls; | ||||||
| 	JSValue tls; |  | ||||||
| } httpd_listener_t; | } httpd_listener_t; | ||||||
|  |  | ||||||
| static void _httpd_listener_cleanup(void* user_data) | static void _httpd_listener_cleanup(void* user_data) | ||||||
| { | { | ||||||
| 	httpd_listener_t* listener = user_data; | 	httpd_listener_t* listener = user_data; | ||||||
| 	JS_FreeValue(listener->context, listener->tls); | 	if (listener->tls) | ||||||
| 	tf_free(listener); |  | ||||||
| } |  | ||||||
|  |  | ||||||
| static JSValue _httpd_endpoint_start(JSContext* context, JSValueConst this_val, int argc, JSValueConst* argv) |  | ||||||
| { |  | ||||||
| 	tf_http_t* http = JS_GetOpaque(this_val, _httpd_class_id); |  | ||||||
| 	int port = 0; |  | ||||||
| 	JS_ToInt32(context, &port, argv[0]); |  | ||||||
|  |  | ||||||
| 	httpd_listener_t* listener = tf_malloc(sizeof(httpd_listener_t)); |  | ||||||
| 	*listener = (httpd_listener_t) { .context = context, .tls = JS_DupValue(context, argv[1]) }; |  | ||||||
| 	tf_tls_context_t* tls = tf_tls_context_get(listener->tls); |  | ||||||
| 	int assigned_port = tf_http_listen(http, port, tls, _httpd_listener_cleanup, listener); |  | ||||||
| 	tf_printf(CYAN "~😎 Tilde Friends" RESET " " YELLOW VERSION_NUMBER RESET " is now up at " MAGENTA "http%s://127.0.0.1:%d/" RESET ".\n", tls ? "s" : "", assigned_port); |  | ||||||
| 	return JS_NewInt32(context, assigned_port); |  | ||||||
| } |  | ||||||
|  |  | ||||||
| static void _httpd_free_user_data(void* user_data) |  | ||||||
| { |  | ||||||
| 	tf_free(user_data); |  | ||||||
| } |  | ||||||
|  |  | ||||||
| static JSValue _httpd_set_http_redirect(JSContext* context, JSValueConst this_val, int argc, JSValueConst* argv) |  | ||||||
| { |  | ||||||
| 	tf_http_t* http = JS_GetOpaque(this_val, _httpd_class_id); |  | ||||||
| 	http_user_data_t* user_data = tf_http_get_user_data(http); |  | ||||||
| 	if (!user_data) |  | ||||||
| 	{ | 	{ | ||||||
| 		user_data = tf_malloc(sizeof(http_user_data_t)); | 		tf_tls_context_destroy(listener->tls); | ||||||
| 		memset(user_data, 0, sizeof(http_user_data_t)); |  | ||||||
| 		tf_http_set_user_data(http, user_data, _httpd_free_user_data); |  | ||||||
| 	} | 	} | ||||||
|  | 	tf_free(listener); | ||||||
| 	const char* redirect = JS_ToCString(context, argv[0]); |  | ||||||
| 	snprintf(user_data->redirect, sizeof(user_data->redirect), "%s", redirect ? redirect : ""); |  | ||||||
| 	JS_FreeCString(context, redirect); |  | ||||||
| 	return JS_UNDEFINED; |  | ||||||
| } | } | ||||||
|  |  | ||||||
| typedef struct _auth_query_work_t | typedef struct _auth_query_work_t | ||||||
| @@ -783,13 +682,24 @@ typedef struct _http_file_t | |||||||
| 	char etag[512]; | 	char etag[512]; | ||||||
| } http_file_t; | } http_file_t; | ||||||
|  |  | ||||||
|  | static bool _ends_with(const char* a, const char* suffix) | ||||||
|  | { | ||||||
|  | 	if (!a || !suffix) | ||||||
|  | 	{ | ||||||
|  | 		return false; | ||||||
|  | 	} | ||||||
|  | 	size_t alen = strlen(a); | ||||||
|  | 	size_t suffixlen = strlen(suffix); | ||||||
|  | 	return alen >= suffixlen && strcmp(a + alen - suffixlen, suffix) == 0; | ||||||
|  | } | ||||||
|  |  | ||||||
| static void _httpd_endpoint_static_read(tf_task_t* task, const char* path, int result, const void* data, void* user_data) | static void _httpd_endpoint_static_read(tf_task_t* task, const char* path, int result, const void* data, void* user_data) | ||||||
| { | { | ||||||
| 	http_file_t* file = user_data; | 	http_file_t* file = user_data; | ||||||
| 	tf_http_request_t* request = file->request; | 	tf_http_request_t* request = file->request; | ||||||
| 	if (result >= 0) | 	if (result >= 0) | ||||||
| 	{ | 	{ | ||||||
| 		if (strcmp(path, "core/tfrpc.js") == 0) | 		if (strcmp(path, "core/tfrpc.js") == 0 || _ends_with(path, "core/tfrpc.js")) | ||||||
| 		{ | 		{ | ||||||
| 			const char* content_type = _ext_to_content_type(strrchr(path, '.'), true); | 			const char* content_type = _ext_to_content_type(strrchr(path, '.'), true); | ||||||
| 			const char* headers[] = { | 			const char* headers[] = { | ||||||
| @@ -931,9 +841,10 @@ static void _httpd_endpoint_static(tf_http_request_t* request) | |||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	tf_task_t* task = request->user_data; | 	tf_task_t* task = request->user_data; | ||||||
| 	size_t size = strlen(file_path) + strlen(after) + 1; | 	const char* root_path = tf_task_get_root_path(task); | ||||||
|  | 	size_t size = (root_path ? strlen(root_path) + 1 : 0) + strlen(file_path) + strlen(after) + 1; | ||||||
| 	char* path = alloca(size); | 	char* path = alloca(size); | ||||||
| 	snprintf(path, size, "%s%s", file_path, after); | 	snprintf(path, size, "%s%s%s%s", root_path ? root_path : "", root_path ? "/" : "", file_path, after); | ||||||
| 	tf_http_request_ref(request); | 	tf_http_request_ref(request); | ||||||
| 	tf_file_stat(task, path, _httpd_endpoint_static_stat, request); | 	tf_file_stat(task, path, _httpd_endpoint_static_stat, request); | ||||||
| } | } | ||||||
| @@ -2255,6 +2166,176 @@ static void _httpd_endpoint_logout(tf_http_request_t* request) | |||||||
| 	tf_http_respond(request, 303, headers, tf_countof(headers) / 2, NULL, 0); | 	tf_http_respond(request, 303, headers, tf_countof(headers) / 2, NULL, 0); | ||||||
| } | } | ||||||
|  |  | ||||||
|  | typedef struct _auto_login_t | ||||||
|  | { | ||||||
|  | 	tf_http_request_t* request; | ||||||
|  | 	bool autologin; | ||||||
|  | 	const char* users; | ||||||
|  | } auto_login_t; | ||||||
|  |  | ||||||
|  | static void _httpd_auto_login_work(tf_ssb_t* ssb, void* user_data) | ||||||
|  | { | ||||||
|  | 	auto_login_t* request = user_data; | ||||||
|  | 	sqlite3* db = tf_ssb_acquire_db_reader(ssb); | ||||||
|  | 	tf_ssb_db_get_global_setting_bool(db, "autologin", &request->autologin); | ||||||
|  | 	tf_ssb_release_db_reader(ssb, db); | ||||||
|  |  | ||||||
|  | 	if (request->autologin) | ||||||
|  | 	{ | ||||||
|  | 		request->users = tf_ssb_db_get_property(ssb, "auth", "users"); | ||||||
|  | 		if (request->users && strcmp(request->users, "[]") == 0) | ||||||
|  | 		{ | ||||||
|  | 			tf_free((void*)request->users); | ||||||
|  | 			request->users = NULL; | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		if (!request->users) | ||||||
|  | 		{ | ||||||
|  | 			JSMallocFunctions funcs = { 0 }; | ||||||
|  | 			tf_get_js_malloc_functions(&funcs); | ||||||
|  | 			JSRuntime* runtime = JS_NewRuntime2(&funcs, NULL); | ||||||
|  | 			JSContext* context = JS_NewContext(runtime); | ||||||
|  | 			static const char* k_account_name = "mobile"; | ||||||
|  | 			sqlite3* db = tf_ssb_acquire_db_writer(ssb); | ||||||
|  | 			bool registered = tf_ssb_db_register_account(tf_ssb_get_loop(ssb), db, context, k_account_name, k_account_name); | ||||||
|  | 			tf_ssb_release_db_writer(ssb, db); | ||||||
|  | 			if (registered) | ||||||
|  | 			{ | ||||||
|  | 				_make_administrator_if_first(ssb, context, k_account_name, true); | ||||||
|  | 			} | ||||||
|  | 			JS_FreeContext(context); | ||||||
|  | 			JS_FreeRuntime(runtime); | ||||||
|  |  | ||||||
|  | 			request->users = tf_ssb_db_get_property(ssb, "auth", "users"); | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | static void _httpd_auto_login_after_work(tf_ssb_t* ssb, int status, void* user_data) | ||||||
|  | { | ||||||
|  | 	auto_login_t* work = user_data; | ||||||
|  | 	JSContext* context = tf_ssb_get_context(ssb); | ||||||
|  | 	const char* session_token = NULL; | ||||||
|  | 	if (!work->autologin) | ||||||
|  | 	{ | ||||||
|  | 		const char* k_payload = tf_http_status_text(404); | ||||||
|  | 		tf_http_respond(work->request, 404, NULL, 0, k_payload, strlen(k_payload)); | ||||||
|  | 	} | ||||||
|  | 	else | ||||||
|  | 	{ | ||||||
|  | 		if (work->users) | ||||||
|  | 		{ | ||||||
|  | 			JSValue json = JS_ParseJSON(context, work->users, strlen(work->users), NULL); | ||||||
|  | 			JSValue user = JS_GetPropertyUint32(context, json, 0); | ||||||
|  | 			const char* user_string = JS_ToCString(context, user); | ||||||
|  | 			session_token = _make_session_jwt(context, ssb, user_string); | ||||||
|  | 			JS_FreeCString(context, user_string); | ||||||
|  | 			JS_FreeValue(context, user); | ||||||
|  | 			JS_FreeValue(context, json); | ||||||
|  | 		} | ||||||
|  | 		if (session_token) | ||||||
|  | 		{ | ||||||
|  | 			const char* cookie = _make_set_session_cookie_header(work->request, session_token); | ||||||
|  | 			tf_free((void*)session_token); | ||||||
|  | 			const char* headers[] = { | ||||||
|  | 				"Set-Cookie", | ||||||
|  | 				cookie, | ||||||
|  | 				"Location", | ||||||
|  | 				"/", | ||||||
|  | 			}; | ||||||
|  | 			tf_http_respond(work->request, 303, headers, tf_countof(headers) / 2, NULL, 0); | ||||||
|  | 			tf_free((void*)cookie); | ||||||
|  | 		} | ||||||
|  | 		else | ||||||
|  | 		{ | ||||||
|  | 			const char* headers[] = { | ||||||
|  | 				"Location", | ||||||
|  | 				"/", | ||||||
|  | 			}; | ||||||
|  | 			tf_http_respond(work->request, 303, headers, tf_countof(headers) / 2, NULL, 0); | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	tf_http_request_unref(work->request); | ||||||
|  | 	tf_free((void*)work->users); | ||||||
|  | 	tf_free(work); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | static void _httpd_endpoint_login_auto(tf_http_request_t* request) | ||||||
|  | { | ||||||
|  | 	tf_task_t* task = request->user_data; | ||||||
|  | 	tf_http_request_ref(request); | ||||||
|  | 	tf_ssb_t* ssb = tf_task_get_ssb(task); | ||||||
|  |  | ||||||
|  | 	auto_login_t* work = tf_malloc(sizeof(auto_login_t)); | ||||||
|  | 	*work = (auto_login_t) { .request = request }; | ||||||
|  | 	tf_ssb_run_work(ssb, _httpd_auto_login_work, _httpd_auto_login_after_work, work); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | static void _httpd_endpoint_app_socket(tf_http_request_t* request) | ||||||
|  | { | ||||||
|  | 	tf_task_t* task = request->user_data; | ||||||
|  | 	tf_ssb_t* ssb = tf_task_get_ssb(task); | ||||||
|  |  | ||||||
|  | 	JSContext* context = tf_ssb_get_context(ssb); | ||||||
|  | 	JSValue global = JS_GetGlobalObject(context); | ||||||
|  | 	JSValue exports = JS_GetPropertyStr(context, global, "exports"); | ||||||
|  | 	JSValue app_socket = JS_GetPropertyStr(context, exports, "app_socket"); | ||||||
|  |  | ||||||
|  | 	JSValue request_object = JS_NewObject(context); | ||||||
|  | 	JSValue headers = JS_NewObject(context); | ||||||
|  | 	for (int i = 0; i < request->headers_count; i++) | ||||||
|  | 	{ | ||||||
|  | 		JS_SetPropertyStr(context, headers, request->headers[i].name, JS_NewString(context, request->headers[i].value)); | ||||||
|  | 	} | ||||||
|  | 	JS_SetPropertyStr(context, request_object, "headers", headers); | ||||||
|  |  | ||||||
|  | 	JSValue response = _httpd_make_response_object(context, request); | ||||||
|  | 	tf_http_request_ref(request); | ||||||
|  |  | ||||||
|  | 	JSValue args[] = { | ||||||
|  | 		request_object, | ||||||
|  | 		response, | ||||||
|  | 	}; | ||||||
|  |  | ||||||
|  | 	JSValue result = JS_Call(context, app_socket, JS_NULL, tf_countof(args), args); | ||||||
|  | 	tf_util_report_error(context, result); | ||||||
|  | 	JS_FreeValue(context, result); | ||||||
|  |  | ||||||
|  | 	for (int i = 0; i < tf_countof(args); i++) | ||||||
|  | 	{ | ||||||
|  | 		JS_FreeValue(context, args[i]); | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	JS_FreeValue(context, app_socket); | ||||||
|  | 	JS_FreeValue(context, exports); | ||||||
|  | 	JS_FreeValue(context, global); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | static void _httpd_free_user_data(void* user_data) | ||||||
|  | { | ||||||
|  | 	tf_free(user_data); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | static const char* _httpd_read_file(tf_task_t* task, const char* path) | ||||||
|  | { | ||||||
|  | 	const char* actual = tf_task_get_path_with_root(task, path); | ||||||
|  | 	const size_t k_max_read = 8 * 1024 * 1024; | ||||||
|  | 	char* result = NULL; | ||||||
|  | 	char* buffer = tf_malloc(k_max_read); | ||||||
|  | 	FILE* file = fopen(actual, "rb"); | ||||||
|  | 	if (file) | ||||||
|  | 	{ | ||||||
|  | 		size_t size = fread(buffer, 1, k_max_read, file); | ||||||
|  | 		result = tf_malloc(size + 1); | ||||||
|  | 		memcpy(result, buffer, size); | ||||||
|  | 		result[size] = '\0'; | ||||||
|  | 		fclose(file); | ||||||
|  | 	} | ||||||
|  | 	tf_free(buffer); | ||||||
|  | 	tf_free((char*)actual); | ||||||
|  | 	return result; | ||||||
|  | } | ||||||
|  |  | ||||||
| void tf_httpd_register(JSContext* context) | void tf_httpd_register(JSContext* context) | ||||||
| { | { | ||||||
| 	JS_NewClassID(&_httpd_class_id); | 	JS_NewClassID(&_httpd_class_id); | ||||||
| @@ -2275,15 +2356,46 @@ void tf_httpd_register(JSContext* context) | |||||||
| 	{ | 	{ | ||||||
| 		fprintf(stderr, "Failed to register Request.\n"); | 		fprintf(stderr, "Failed to register Request.\n"); | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	JSValue global = JS_GetGlobalObject(context); | 	JSValue global = JS_GetGlobalObject(context); | ||||||
| 	JSValue httpd = JS_NewObjectClass(context, _httpd_class_id); | 	JSValue httpd = JS_NewObjectClass(context, _httpd_class_id); | ||||||
|  |  | ||||||
| 	tf_task_t* task = tf_task_get(context); | 	tf_task_t* task = tf_task_get(context); | ||||||
|  | 	tf_ssb_t* ssb = tf_task_get_ssb(task); | ||||||
| 	uv_loop_t* loop = tf_task_get_loop(task); | 	uv_loop_t* loop = tf_task_get_loop(task); | ||||||
| 	tf_http_t* http = tf_http_create(loop); | 	tf_http_t* http = tf_http_create(loop); | ||||||
| 	tf_http_set_trace(http, tf_task_get_trace(task)); | 	tf_http_set_trace(http, tf_task_get_trace(task)); | ||||||
| 	JS_SetOpaque(httpd, http); | 	JS_SetOpaque(httpd, http); | ||||||
|  |  | ||||||
|  | 	int64_t http_port = 0; | ||||||
|  | 	int64_t https_port = 0; | ||||||
|  | 	char out_http_port_file[512] = ""; | ||||||
|  | 	sqlite3* db = tf_ssb_acquire_db_reader(ssb); | ||||||
|  | 	tf_ssb_db_get_global_setting_int64(db, "http_port", &http_port); | ||||||
|  | 	tf_ssb_db_get_global_setting_int64(db, "https_port", &https_port); | ||||||
|  | 	tf_ssb_db_get_global_setting_string(db, "out_http_port_file", out_http_port_file, sizeof(out_http_port_file)); | ||||||
|  | 	tf_ssb_release_db_reader(ssb, db); | ||||||
|  |  | ||||||
|  | 	if (https_port) | ||||||
|  | 	{ | ||||||
|  | 		http_user_data_t* user_data = tf_http_get_user_data(http); | ||||||
|  | 		if (!user_data) | ||||||
|  | 		{ | ||||||
|  | 			user_data = tf_malloc(sizeof(http_user_data_t)); | ||||||
|  | 			memset(user_data, 0, sizeof(http_user_data_t)); | ||||||
|  | 			tf_http_set_user_data(http, user_data, _httpd_free_user_data); | ||||||
|  | 		} | ||||||
|  | 		sqlite3* db = tf_ssb_acquire_db_reader(ssb); | ||||||
|  | 		tf_ssb_db_get_global_setting_string(db, "http_redirect", user_data->redirect, sizeof(user_data->redirect)); | ||||||
|  | 		tf_ssb_release_db_reader(ssb, db); | ||||||
|  |  | ||||||
|  | 		/* Workaround. */ | ||||||
|  | 		if (strcmp(user_data->redirect, "0") == 0) | ||||||
|  | 		{ | ||||||
|  | 			*user_data->redirect = '\0'; | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
| 	tf_http_add_handler(http, "/", _httpd_endpoint_root, NULL, task); | 	tf_http_add_handler(http, "/", _httpd_endpoint_root, NULL, task); | ||||||
| 	tf_http_add_handler(http, "/codemirror/*", _httpd_endpoint_static, NULL, task); | 	tf_http_add_handler(http, "/codemirror/*", _httpd_endpoint_static, NULL, task); | ||||||
| 	tf_http_add_handler(http, "/lit/*", _httpd_endpoint_static, NULL, task); | 	tf_http_add_handler(http, "/lit/*", _httpd_endpoint_static, NULL, task); | ||||||
| @@ -2310,12 +2422,57 @@ void tf_httpd_register(JSContext* context) | |||||||
| 	tf_http_add_handler(http, "/trace", _httpd_endpoint_trace, NULL, task); | 	tf_http_add_handler(http, "/trace", _httpd_endpoint_trace, NULL, task); | ||||||
|  |  | ||||||
| 	tf_http_add_handler(http, "/login/logout", _httpd_endpoint_logout, NULL, task); | 	tf_http_add_handler(http, "/login/logout", _httpd_endpoint_logout, NULL, task); | ||||||
|  | 	tf_http_add_handler(http, "/login/auto", _httpd_endpoint_login_auto, NULL, task); | ||||||
| 	tf_http_add_handler(http, "/login", _httpd_endpoint_login, NULL, task); | 	tf_http_add_handler(http, "/login", _httpd_endpoint_login, NULL, task); | ||||||
|  |  | ||||||
| 	JS_SetPropertyStr(context, httpd, "all", JS_NewCFunction(context, _httpd_endpoint_all, "all", 2)); | 	tf_http_add_handler(http, "/app/socket", _httpd_endpoint_app_socket, NULL, task); | ||||||
| 	JS_SetPropertyStr(context, httpd, "start", JS_NewCFunction(context, _httpd_endpoint_start, "start", 2)); |  | ||||||
| 	JS_SetPropertyStr(context, httpd, "set_http_redirect", JS_NewCFunction(context, _httpd_set_http_redirect, "set_http_redirect", 1)); |  | ||||||
| 	JS_SetPropertyStr(context, httpd, "auth_query", JS_NewCFunction(context, _httpd_auth_query, "auth_query", 1)); | 	JS_SetPropertyStr(context, httpd, "auth_query", JS_NewCFunction(context, _httpd_auth_query, "auth_query", 1)); | ||||||
| 	JS_SetPropertyStr(context, global, "httpd", httpd); | 	JS_SetPropertyStr(context, global, "httpd", httpd); | ||||||
| 	JS_FreeValue(context, global); | 	JS_FreeValue(context, global); | ||||||
|  |  | ||||||
|  | 	if (http_port > 0 || *out_http_port_file) | ||||||
|  | 	{ | ||||||
|  | 		httpd_listener_t* listener = tf_malloc(sizeof(httpd_listener_t)); | ||||||
|  | 		*listener = (httpd_listener_t) { 0 }; | ||||||
|  | 		int assigned_port = tf_http_listen(http, http_port, NULL, _httpd_listener_cleanup, listener); | ||||||
|  | 		tf_printf(CYAN "~😎 Tilde Friends" RESET " " YELLOW VERSION_NUMBER RESET " is now up at " MAGENTA "http://127.0.0.1:%d/" RESET ".\n", assigned_port); | ||||||
|  |  | ||||||
|  | 		if (*out_http_port_file) | ||||||
|  | 		{ | ||||||
|  | 			const char* actual_http_port_file = tf_task_get_path_with_root(task, out_http_port_file); | ||||||
|  | 			FILE* file = fopen(actual_http_port_file, "wb"); | ||||||
|  | 			if (file) | ||||||
|  | 			{ | ||||||
|  | 				fprintf(file, "%d", assigned_port); | ||||||
|  | 				fclose(file); | ||||||
|  | 				tf_printf("Wrote the port file: %s.\n", out_http_port_file); | ||||||
|  | 			} | ||||||
|  | 			else | ||||||
|  | 			{ | ||||||
|  | 				tf_printf("Failed to open %s for write: %s.\n", out_http_port_file, strerror(errno)); | ||||||
|  | 			} | ||||||
|  | 			tf_free((char*)actual_http_port_file); | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		if (https_port) | ||||||
|  | 		{ | ||||||
|  | 			const char* k_certificate = "data/httpd/certificate.pem"; | ||||||
|  | 			const char* k_private_key = "data/httpd/privatekey.pem"; | ||||||
|  | 			const char* certificate = _httpd_read_file(task, k_certificate); | ||||||
|  | 			const char* private_key = _httpd_read_file(task, k_private_key); | ||||||
|  | 			if (certificate && private_key) | ||||||
|  | 			{ | ||||||
|  | 				tf_tls_context_t* tls = tf_tls_context_create(); | ||||||
|  | 				tf_tls_context_set_certificate(tls, certificate); | ||||||
|  | 				tf_tls_context_set_private_key(tls, private_key); | ||||||
|  | 				httpd_listener_t* listener = tf_malloc(sizeof(httpd_listener_t)); | ||||||
|  | 				*listener = (httpd_listener_t) { .tls = tls }; | ||||||
|  | 				int assigned_port = tf_http_listen(http, https_port, tls, _httpd_listener_cleanup, listener); | ||||||
|  | 				tf_printf(CYAN "~😎 Tilde Friends" RESET " " YELLOW VERSION_NUMBER RESET " is now up at " MAGENTA "https://127.0.0.1:%d/" RESET ".\n", assigned_port); | ||||||
|  | 			} | ||||||
|  | 			tf_free((char*)certificate); | ||||||
|  | 			tf_free((char*)private_key); | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
| } | } | ||||||
|   | |||||||
| @@ -1 +0,0 @@ | |||||||
| \mainpage Tilde Friends Source Documentation |  | ||||||
| @@ -16,7 +16,7 @@ void tf_run_thread_start(const char* zip_path); | |||||||
|  |  | ||||||
| static void _start_initial_load(WKWebView* web_view) | static void _start_initial_load(WKWebView* web_view) | ||||||
| { | { | ||||||
| 	[web_view loadRequest:[NSURLRequest requestWithURL:[NSURL URLWithString:@"http://localhost:12345/"]]]; | 	[web_view loadRequest:[NSURLRequest requestWithURL:[NSURL URLWithString:@"http://localhost:12345/login/auto"]]]; | ||||||
| } | } | ||||||
|  |  | ||||||
| @implementation ViewController : UIViewController | @implementation ViewController : UIViewController | ||||||
|   | |||||||
| @@ -7,6 +7,6 @@ | |||||||
| 	<key>com.apple.developer.team-identifier</key> | 	<key>com.apple.developer.team-identifier</key> | ||||||
| 	<string>EDVXQ27EB5</string> | 	<string>EDVXQ27EB5</string> | ||||||
| 	<key>get-task-allow</key> | 	<key>get-task-allow</key> | ||||||
| 	<true/> | 	<false/> | ||||||
| </dict> | </dict> | ||||||
| </plist> | </plist> | ||||||
|   | |||||||
| @@ -2,34 +2,86 @@ | |||||||
| <!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd"> | <!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd"> | ||||||
| <plist version="1.0"> | <plist version="1.0"> | ||||||
| <dict> | <dict> | ||||||
| 	<key>CFBundleName</key> | 	<key>CFBundleDisplayName</key> | ||||||
| 	<string>tildefriends</string> | 	<string>Tilde Friends</string> | ||||||
| 	<key>CFBundleSupportedPlatforms</key> |  | ||||||
| 	<array> |  | ||||||
| 		<string>iPhoneOS</string> |  | ||||||
| 	</array> |  | ||||||
| 	<key>CFBundleExecutable</key> | 	<key>CFBundleExecutable</key> | ||||||
| 	<string>tildefriends</string> | 	<string>tildefriends</string> | ||||||
| 	<key>CFBundleIdentifier</key> | 	<key>CFBundleIdentifier</key> | ||||||
| 	<string>com.unprompted.tildefriends</string> | 	<string>com.unprompted.tildefriends</string> | ||||||
| 	<key>CFBundleResourceSpecification</key> | 	<key>CFBundleName</key> | ||||||
| 	<string>ResourceRules.plist</string> | 	<string>tildefriends</string> | ||||||
|  | 	<key>CFBundlePackageType</key> | ||||||
|  | 	<string>APPL</string> | ||||||
|  | 	<key>CFBundleShortVersionString</key> | ||||||
|  | 	<string>0.0.28</string> | ||||||
|  | 	<key>CFBundleSupportedPlatforms</key> | ||||||
|  | 	<array> | ||||||
|  | 		<string>iPhoneOS</string> | ||||||
|  | 	</array> | ||||||
|  | 	<key>CFBundleVersion</key> | ||||||
|  | 	<string>8</string> | ||||||
|  | 	<key>DTPlatformName</key> | ||||||
|  | 	<string>iphoneos</string> | ||||||
| 	<key>LSRequiresIPhoneOS</key> | 	<key>LSRequiresIPhoneOS</key> | ||||||
| 	<true/> | 	<true/> | ||||||
| 	<key>CFBundleDisplayName</key> | 	<key>MinimumOSVersion</key> | ||||||
| 	<string>Tilde Friends</string> | 	<string>14.0</string> | ||||||
| 	<key>CFBundleVersion</key> | 	<key>UIDeviceFamily</key> | ||||||
| 	<string>1.0</string> | 	<array> | ||||||
| 	<key>CFBundleShortVersionString</key> | 		<integer>1</integer> | ||||||
| 	<string>1.0</string> | 		<integer>2</integer> | ||||||
| 	<key>CFBundleIconName</key> | 	</array> | ||||||
| 	<string>AppIcon</string> |  | ||||||
| 	<key>CFBundleIconFile</key> |  | ||||||
| 	<string>tildefriends.png</string> |  | ||||||
| 	<key>UILaunchScreen</key> | 	<key>UILaunchScreen</key> | ||||||
| 	<dict> | 	<dict> | ||||||
| 		<key>UIImageName</key> | 		<key>UIImageName</key> | ||||||
| 		<string>tildefriends.png</string> | 		<string>tildefriends.png</string> | ||||||
| 	</dict> | 	</dict> | ||||||
|  | 	<key>UIRequiredDeviceCapabilities</key> | ||||||
|  | 	<array> | ||||||
|  | 		<string>arm64</string> | ||||||
|  | 	</array> | ||||||
|  | 	<key>BuildMachineOSBuild</key> | ||||||
|  | 	<string>24D70</string> | ||||||
|  | 	<key>DTPlatformBuild</key> | ||||||
|  | 	<string>22C146</string> | ||||||
|  | 	<key>DTPlatformVersion</key> | ||||||
|  | 	<string>18.2</string> | ||||||
|  | 	<key>DTSDKBuild</key> | ||||||
|  | 	<string>22C146</string> | ||||||
|  | 	<key>DTSDKName</key> | ||||||
|  | 	<string>iphoneos18.2</string> | ||||||
|  | 	<key>DTXcode</key> | ||||||
|  | 	<string>1620</string> | ||||||
|  | 	<key>DTXcodeBuild</key> | ||||||
|  | 	<string>16C5032a</string> | ||||||
|  | 	<key>UISupportedInterfaceOrientations~ipad</key> | ||||||
|  | 	<array> | ||||||
|  | 		<string>UIInterfaceOrientationPortrait</string> | ||||||
|  | 		<string>UIInterfaceOrientationPortraitUpsideDown</string> | ||||||
|  | 		<string>UIInterfaceOrientationLandscapeLeft</string> | ||||||
|  | 		<string>UIInterfaceOrientationLandscapeRight</string> | ||||||
|  | 	</array> | ||||||
|  | 	<key>UISupportedInterfaceOrientations~iphone</key> | ||||||
|  | 	<array> | ||||||
|  | 		<string>UIInterfaceOrientationPortrait</string> | ||||||
|  | 		<string>UIInterfaceOrientationLandscapeLeft</string> | ||||||
|  | 		<string>UIInterfaceOrientationLandscapeRight</string> | ||||||
|  | 	</array> | ||||||
|  | 	<key>UILaunchStoryboardName</key> | ||||||
|  | 	<string>LaunchScreen</string> | ||||||
|  | 	<key>CFBundleIcons</key> | ||||||
|  | 	<dict> | ||||||
|  | 		<key>CFBundlePrimaryIcon</key> | ||||||
|  | 		<dict> | ||||||
|  | 			<key>CFBundleIconFiles</key> | ||||||
|  | 			<array> | ||||||
|  | 				<string>AppIcon60x60</string> | ||||||
|  | 			</array> | ||||||
|  | 			<key>CFBundleIconName</key> | ||||||
|  | 			<string>AppIcon</string> | ||||||
|  | 		</dict> | ||||||
|  | 	</dict> | ||||||
|  | 	<key>ITSAppUsesNonExemptEncryption</key> | ||||||
|  | 	<false/> | ||||||
| </dict> | </dict> | ||||||
| </plist> | </plist> | ||||||
|   | |||||||
							
								
								
									
										
											BIN
										
									
								
								src/ios/distribution.mobileprovision
									
									
									
									
									
										Normal file
									
								
							
							
						
						
							
								
								
									
										
											BIN
										
									
								
								src/ios/icons/Assets.car
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1 @@ | |||||||
|  | {"images":[{"idiom":"ios-marketing","scale":"1x","size":"1024x1024","filename":"icon-ios-marketing-1-1024-1024.png"}],"info":{"author":"xcode","version":1}} | ||||||
| After Width: | Height: | Size: 101 KiB | 
							
								
								
									
										1
									
								
								src/ios/icons/Assets.xcassets/Contents.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1 @@ | |||||||
|  | {"info": {"version": 1, "author": "xcode"}} | ||||||
							
								
								
									
										
											BIN
										
									
								
								src/ios/icons/android/mipmap-hdpi.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 4.2 KiB | 
							
								
								
									
										
											BIN
										
									
								
								src/ios/icons/android/mipmap-ldpi.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 1.8 KiB | 
							
								
								
									
										
											BIN
										
									
								
								src/ios/icons/android/mipmap-mdpi.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 2.6 KiB | 
							
								
								
									
										
											BIN
										
									
								
								src/ios/icons/android/mipmap-xhdpi.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 5.9 KiB | 
							
								
								
									
										
											BIN
										
									
								
								src/ios/icons/android/mipmap-xxhdpi.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 9.6 KiB | 
							
								
								
									
										
											BIN
										
									
								
								src/ios/icons/android/mipmap-xxxhdpi.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 13 KiB | 
							
								
								
									
										
											BIN
										
									
								
								src/ios/icons/ios/100.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 6.2 KiB | 
							
								
								
									
										
											BIN
										
									
								
								src/ios/icons/ios/114.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 7.2 KiB | 
							
								
								
									
										
											BIN
										
									
								
								src/ios/icons/ios/128.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 8.3 KiB | 
							
								
								
									
										
											BIN
										
									
								
								src/ios/icons/ios/16.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 639 B | 
							
								
								
									
										
											BIN
										
									
								
								src/ios/icons/ios/172.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 12 KiB | 
							
								
								
									
										
											BIN
										
									
								
								src/ios/icons/ios/180.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 12 KiB | 
							
								
								
									
										
											BIN
										
									
								
								src/ios/icons/ios/196.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 14 KiB | 
							
								
								
									
										
											BIN
										
									
								
								src/ios/icons/ios/216.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 15 KiB | 
							
								
								
									
										
											BIN
										
									
								
								src/ios/icons/ios/256.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 17 KiB | 
							
								
								
									
										
											BIN
										
									
								
								src/ios/icons/ios/32.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 1.5 KiB | 
							
								
								
									
										
											BIN
										
									
								
								src/ios/icons/ios/48.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 2.6 KiB | 
							
								
								
									
										
											BIN
										
									
								
								src/ios/icons/ios/50.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 2.7 KiB | 
							
								
								
									
										
											BIN
										
									
								
								src/ios/icons/ios/512.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 38 KiB | 
							
								
								
									
										
											BIN
										
									
								
								src/ios/icons/ios/55.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 3.1 KiB | 
							
								
								
									
										
											BIN
										
									
								
								src/ios/icons/ios/64.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 3.7 KiB | 
							
								
								
									
										
											BIN
										
									
								
								src/ios/icons/ios/88.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 5.3 KiB | 
							
								
								
									
										
											BIN
										
									
								
								src/ios/icons/ios/icon-120.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 7.6 KiB | 
							
								
								
									
										
											BIN
										
									
								
								src/ios/icons/ios/icon-144.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 9.6 KiB | 
							
								
								
									
										
											BIN
										
									
								
								src/ios/icons/ios/icon-152.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 10 KiB | 
							
								
								
									
										
											BIN
										
									
								
								src/ios/icons/ios/icon-167.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 11 KiB | 
							
								
								
									
										
											BIN
										
									
								
								src/ios/icons/ios/icon-72.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 4.2 KiB | 
							
								
								
									
										
											BIN
										
									
								
								src/ios/icons/ios/icon-76.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 4.5 KiB | 
							
								
								
									
										
											BIN
										
									
								
								src/ios/icons/ios/icon-notification.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 858 B | 
							
								
								
									
										
											BIN
										
									
								
								src/ios/icons/ios/icon-notification@2x.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 2.1 KiB | 
							
								
								
									
										
											BIN
										
									
								
								src/ios/icons/ios/icon-notification@3x.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 3.4 KiB | 
							
								
								
									
										
											BIN
										
									
								
								src/ios/icons/ios/icon-small-120.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 7.6 KiB | 
							
								
								
									
										
											BIN
										
									
								
								src/ios/icons/ios/icon-small-40.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 2.1 KiB | 
							
								
								
									
										
											BIN
										
									
								
								src/ios/icons/ios/icon-small-80.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 4.8 KiB | 
							
								
								
									
										
											BIN
										
									
								
								src/ios/icons/ios/icon-small.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 1.4 KiB | 
							
								
								
									
										
											BIN
										
									
								
								src/ios/icons/ios/icon-small@2x.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 3.2 KiB | 
							
								
								
									
										
											BIN
										
									
								
								src/ios/icons/ios/icon-small@3x.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 5.2 KiB | 
							
								
								
									
										
											BIN
										
									
								
								src/ios/icons/ios/icon-store.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 101 KiB | 
							
								
								
									
										
											BIN
										
									
								
								src/ios/icons/ios/icon.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 3.2 KiB | 
							
								
								
									
										
											BIN
										
									
								
								src/ios/icons/ios/icon@2x.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 7.2 KiB |