Compare commits
	
		
			47 Commits
		
	
	
		
			3746622a11
			...
			main
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 09a0cfd349 | |||
| 5bf7346321 | |||
| dd558c57e0 | |||
| 5647196924 | |||
| 49b1834bc6 | |||
| d111647ea8 | |||
| d7580dab9b | |||
| 28db8a8d5f | |||
| e64d5617e7 | |||
| acd114650a | |||
| 7a47ffaa61 | |||
| 42f7f66f35 | |||
| b2b4ffeeae | |||
| 26de1f7daa | |||
| 07605933dc | |||
| 4bdc7ec616 | |||
| 8ca64550e5 | |||
| 25dbac804c | |||
| 6ab3fd168b | |||
| 94858e2371 | |||
| 6d13502e94 | |||
| 77001e595c | |||
| 6fad20ffa3 | |||
| 00fb6c9839 | |||
| 97fcf72d63 | |||
| 5d8d02515d | |||
| 859fe1feb0 | |||
| 8f61d83f41 | |||
| 6423b3e479 | |||
| 2bc8cec8a2 | |||
| b49a6cd685 | |||
| 2885380f40 | |||
| 2ec3b6a249 | |||
| 3ef795452d | |||
| 479d87c8b8 | |||
| a56077dcc7 | |||
| d3f4587c3b | |||
| 623705b7a1 | |||
| 8f87f4751d | |||
| 2ac6dfde9d | |||
| 81ade7a400 | |||
| 63f7ff9f27 | |||
| 8a0fa17a79 | |||
| 0ead5ed967 | |||
| 53261a6fbc | |||
| c60ff86a4d | |||
| 83a0b017c5 | 
							
								
								
									
										4
									
								
								.gitmodules
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										4
									
								
								.gitmodules
									
									
									
									
										vendored
									
									
								
							| @@ -19,10 +19,6 @@ | ||||
| [submodule "deps/picohttpparser"] | ||||
| 	path = deps/picohttpparser | ||||
| 	url = https://github.com/h2o/picohttpparser.git | ||||
| [submodule "deps/openssl_src"] | ||||
| 	path = deps/openssl_src | ||||
| 	url = https://github.com/openssl/openssl.git | ||||
| 	shallow = true | ||||
| [submodule "deps/c-ares"] | ||||
| 	path = deps/c-ares | ||||
| 	url = https://github.com/c-ares/c-ares.git | ||||
|   | ||||
| @@ -4,7 +4,6 @@ RUN apt-get update && \ | ||||
| 	apt-get install -y --no-install-recommends \ | ||||
| 		gcc \ | ||||
| 		libc6-dev \ | ||||
| 		perl \ | ||||
| 		make | ||||
|  | ||||
| COPY . /app | ||||
|   | ||||
							
								
								
									
										1
									
								
								Doxyfile
									
									
									
									
									
								
							
							
						
						
									
										1
									
								
								Doxyfile
									
									
									
									
									
								
							| @@ -910,7 +910,6 @@ INPUT                  = README.md \ | ||||
|                          core/app.js \ | ||||
|                          core/client.js \ | ||||
|                          core/core.js \ | ||||
|                          core/http.js \ | ||||
|                          core/tfrpc.js \ | ||||
|                          docs/ \ | ||||
|                          src/ | ||||
|   | ||||
							
								
								
									
										138
									
								
								GNUmakefile
									
									
									
									
									
								
							
							
						
						
									
										138
									
								
								GNUmakefile
									
									
									
									
									
								
							| @@ -38,7 +38,6 @@ BUNDLETOOL = out/bundletool.jar | ||||
|  | ||||
| HAVE_WIN := | ||||
| HAVE_CROSS_AARCH64 := | ||||
| USE_SYSTEM_SSL := | ||||
|  | ||||
| export SOURCE_DATE_EPOCH=1 | ||||
| export TZ=UTC | ||||
| @@ -65,7 +64,6 @@ LDFLAGS += \ | ||||
| 	-lbsd \ | ||||
| 	-lnetwork \ | ||||
| 	-Wno-stringop-overflow | ||||
| USE_SYSTEM_SSL := 1 | ||||
| HAVE_ANDROID = 0 | ||||
| HAVE_LINUX_IOS = 0 | ||||
| HAVE_LINUX_MACOS = 0 | ||||
| @@ -80,13 +78,12 @@ LDFLAGS += \ | ||||
| HAVE_ANDROID := | ||||
| HAVE_LINUX_IOS := | ||||
| HAVE_LINUX_MACOS := | ||||
| USE_SYSTEM_SSL := 1 | ||||
| else | ||||
| $(error Unexpected host platform $(UNAME_S).) | ||||
| 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)) | ||||
| $(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)) | ||||
|  | ||||
| CFLAGS += \ | ||||
| 	-std=gnu11 \ | ||||
| @@ -270,16 +267,12 @@ $(WINDOWS_TARGETS): AS = $(CC) | ||||
| $(WINDOWS_TARGETS): CFLAGS += \ | ||||
| 	-D_WIN32_WINNT=0x0A00 \ | ||||
| 	-DWINVER=0x0A00 \ | ||||
| 	-DNTDDI_VERSION=NTDDI_WIN10 \ | ||||
| 	-Iout/openssl/$(UNAME_S)/mingw64/usr/local/include | ||||
| 	-DNTDDI_VERSION=NTDDI_WIN10 | ||||
| $(WINDOWS_TARGETS): LDFLAGS += \ | ||||
| 	-static \ | ||||
| 	-lm \ | ||||
| 	-Lout/openssl/$(UNAME_S)/mingw64/usr/local/lib | ||||
| 	-lm | ||||
| $(AARCH64_TARGETS): CC = aarch64-linux-gnu-gcc | ||||
| $(AARCH64_TARGETS): AS = $(CC) | ||||
| $(AARCH64_TARGETS): CFLAGS += -Iout/openssl/Linux/aarch64/usr/local/include | ||||
| $(AARCH64_TARGETS): LDFLAGS += -Lout/openssl/Linux/aarch64/usr/local/lib | ||||
| ifeq ($(UNAME_S),Darwin) | ||||
| $(HOST_TARGETS): CC = xcrun clang | ||||
| $(IOS_TARGETS): IOS_SYSROOT := $(shell xcrun --sdk iphoneos --show-sdk-path) | ||||
| @@ -304,39 +297,12 @@ $(ANDROID_TARGETS): AS = $(CC) | ||||
| $(ANDROID_TARGETS): CFLAGS += \ | ||||
| 	-target $(ANDROID_NDK_TARGET_TRIPLE)$(ANDROID_MIN_SDK_VERSION) \ | ||||
| 	-Wno-unknown-warning-option | ||||
| $(ANDROID_ARMV7A_TARGETS): CFLAGS += -Iout/openssl/android/armeabi-v7a/usr/local/include | ||||
| $(ANDROID_ARMV7A_TARGETS): LDFLAGS += -Lout/openssl/android/armeabi-v7a/usr/local/lib | ||||
| $(ANDROID_ARM64_TARGETS): CFLAGS += -Iout/openssl/android/arm64-v8a/usr/local/include | ||||
| $(ANDROID_ARM64_TARGETS): LDFLAGS += -Lout/openssl/android/arm64-v8a/usr/local/lib | ||||
| $(ANDROID_X86_TARGETS): CFLAGS += -Iout/openssl/android/x86/usr/local/include | ||||
| $(ANDROID_X86_TARGETS): CFLAGS += -Wno-atomic-alignment | ||||
| $(ANDROID_X86_TARGETS): LDFLAGS += -Lout/openssl/android/x86/usr/local/lib | ||||
| $(ANDROID_X86_64_TARGETS): CFLAGS += -Iout/openssl/android/x86_64/usr/local/include | ||||
| $(ANDROID_X86_64_TARGETS): LDFLAGS += -Lout/openssl/android/x86_64/usr/local/lib | ||||
| $(NONMACOS_TARGETS): CFLAGS += -Wno-cast-function-type | ||||
| $(MACOS_TARGETS): LDFLAGS += -Wl,-dead_strip | ||||
| $(NONMACOS_TARGETS): LDFLAGS += -Wl,--gc-sections -Wl,--as-needed | ||||
| $(IOS_TARGETS): CFLAGS += -miphoneos-version-min=$(IPHONEOS_VERSION_MIN) | ||||
| $(IOS_TARGETS): LDFLAGS += -miphoneos-version-min=$(IPHONEOS_VERSION_MIN) | ||||
| ifeq ($(UNAME_S),Darwin) | ||||
| $(IOS_TARGETS): CFLAGS += -Iout/openssl/ios/ios64-xcrun/usr/local/include | ||||
| $(IOS_TARGETS): LDFLAGS += -Lout/openssl/ios/ios64-xcrun/usr/local/lib | ||||
| else | ||||
| $(IOS_TARGETS): CFLAGS += -Iout/openssl/$(UNAME_S)/ios64-cross/usr/local/include | ||||
| $(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 | ||||
| $(IOSSIM_TARGETS): CFLAGS += -Iout/openssl/ios/iossimulator-xcrun/usr/local/include | ||||
| $(IOSSIM_TARGETS): LDFLAGS += -Lout/openssl/ios/iossimulator-xcrun/usr/local/lib | ||||
| $(HOST_TARGETS): CFLAGS += -Iout/openssl/$(UNAME_S)/$(UNAME_M)/usr/local/include | ||||
| $(HOST_TARGETS): LDFLAGS += -Lout/openssl/$(UNAME_S)/$(UNAME_M)/usr/local/lib | ||||
|  | ||||
| ifeq ($(UNAME_M),x86_64) | ||||
| ifeq ($(UNAME_S),Linux) | ||||
| @@ -824,9 +790,6 @@ $(MINIUNZIP_OBJS): CFLAGS += \ | ||||
| LDFLAGS += \ | ||||
| 	-pthread \ | ||||
| 	-lm | ||||
| $(HOST_TARGETS) $(IOS_TARGETS) $(IOSSIM_TARGETS) $(AARCH64_TARGETS) $(filter-out $(HOST_TARGETS),$(MACOS_TARGETS)): LDFLAGS += \ | ||||
| 	-lssl \ | ||||
| 	-lcrypto | ||||
| ifneq ($(UNAME_S),Haiku) | ||||
| ifneq ($(UNAME_S),OpenBSD) | ||||
| $(HOST_TARGETS) $(IOS_TARGETS) $(IOSSIM_TARGETS): LDFLAGS += \ | ||||
| @@ -834,8 +797,6 @@ $(HOST_TARGETS) $(IOS_TARGETS) $(IOSSIM_TARGETS): LDFLAGS += \ | ||||
| endif | ||||
| endif | ||||
| $(WINDOWS_TARGETS): LDFLAGS += \ | ||||
| 	-lssl \ | ||||
| 	-lcrypto \ | ||||
| 	-lcrypt32 \ | ||||
| 	-ldbghelp \ | ||||
| 	-liphlpapi \ | ||||
| @@ -848,9 +809,7 @@ $(WINDOWS_TARGETS): LDFLAGS += \ | ||||
| $(ANDROID_TARGETS): LDFLAGS += \ | ||||
| 	-target $(ANDROID_NDK_TARGET_TRIPLE)$(ANDROID_MIN_SDK_VERSION) \ | ||||
| 	-ldl \ | ||||
| 	-llog \ | ||||
| 	-lssl \ | ||||
| 	-lcrypto | ||||
| 	-llog | ||||
| $(MACOS_TARGETS) $(IOS_TARGETS) $(IOSSIM_TARGETS): CFLAGS += \ | ||||
| 	-Wno-unknown-warning-option | ||||
| $(IOS_TARGETS) $(IOSSIM_TARGETS): LDFLAGS += \ | ||||
| @@ -1193,6 +1152,7 @@ out/tildefriends-%.ipa: out/tildefriends-ios%.app/tildefriends | ||||
| 	@echo "[ipa] $@" | ||||
| 	@rm -rf $@.tmp $@ | ||||
| 	@mkdir -p $@.tmp/Payload/tildefriends.app/ | ||||
| 	@cp src/ios/tildefriends512.png $@.tmp/iTunesArtwork | ||||
| 	@cp -R $(dir $<)/* $@.tmp/Payload/tildefriends.app/ | ||||
| 	@cd $@.tmp/ && zip -u ../../$@ -q -9 -r ./ | ||||
| 	@rm -rf $@.tmp/ | ||||
| @@ -1224,94 +1184,6 @@ iossimdebuggo: out/tildefriends-iossimdebug.app/tildefriends ## Build, install, | ||||
| 	xcrun simctl launch booted com.unprompted.tildefriends | ||||
| .PHONY: iossimdebuggo | ||||
|  | ||||
| ANDROID_DEPS := out/openssl/android/arm64-v8a/usr/local/lib/libssl.a | ||||
| $(ANDROID_DEPS): | ||||
| 	+@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) | ||||
|  | ||||
| ifeq ($(UNAME_S),Linux) | ||||
| ifneq ($(USE_SYSTEM_SSL),1) | ||||
| LOCAL_DEPS := out/openssl/$(UNAME_S)/$(UNAME_M)/usr/local/lib/libssl.a | ||||
| $(LOCAL_DEPS): | ||||
| 	+@tools/ssl-local | ||||
| $(filter $(BUILD_DIR)/debug/%,$(APP_OBJS)) $(filter $(BUILD_DIR)/release/%,$(APP_OBJS)): | $(LOCAL_DEPS) | ||||
| endif | ||||
|  | ||||
| ifeq ($(HAVE_CROSS_AARCH64),1) | ||||
| LOCAL_DEPS := out/openssl/$(UNAME_S)/aarch64/usr/local/lib/libssl.a | ||||
| $(LOCAL_DEPS): | ||||
| 	+@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) | ||||
| endif | ||||
|  | ||||
| ifeq ($(HAVE_LINUX_IOS),1) | ||||
| LOCAL_DEPS := out/openssl/$(UNAME_S)/ios64-cross/usr/local/lib/libssl.a | ||||
| $(LOCAL_DEPS): | ||||
| 	+@PATH=deps/ios_toolchain/target/bin:$$PATH \ | ||||
| 		BUILD_TARGET=ios64-cross \ | ||||
| 		SSL_TARGET=ios64-cross \ | ||||
| 		CROSS_COMPILE=../../deps/ios_toolchain/target/bin/arm-apple-darwin11- \ | ||||
| 		CROSS_TOP=../../deps/ios_toolchain/target \ | ||||
| 		CROSS_SDK=iPhoneOS18.2.sdk \ | ||||
| 		CC=clang \ | ||||
| 		OPTIONS=-miphoneos-version-min=$(IPHONEOS_VERSION_MIN) \ | ||||
| 		tools/ssl-local | ||||
| $(filter $(BUILD_DIR)/ios%,$(APP_OBJS)): | $(LOCAL_DEPS) | ||||
| 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 | ||||
|  | ||||
| ifeq ($(UNAME_S),Darwin) | ||||
| LOCAL_DEPS := out/openssl/$(UNAME_S)/$(UNAME_M)/usr/local/lib/libssl.a | ||||
| $(LOCAL_DEPS): | ||||
| 	+@tools/ssl-local | ||||
| $(filter $(BUILD_DIR)/debug/%,$(APP_OBJS)) $(filter $(BUILD_DIR)/release/%,$(APP_OBJS)): | $(LOCAL_DEPS) | ||||
| endif | ||||
|  | ||||
| ifeq ($(HAVE_WIN),1) | ||||
| WINDOWS_DEPS := out/openssl/$(UNAME_S)/mingw64/usr/local/lib/libssl.a | ||||
| $(WINDOWS_DEPS): | ||||
| 	+@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) | ||||
| endif | ||||
|  | ||||
| ifeq ($(UNAME_S),Darwin) | ||||
| IOS_DEPS := out/openssl/ios/ios64-xcrun/usr/local/lib/libssl.a | ||||
| $(IOS_DEPS): | ||||
| 	+@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) | ||||
| endif | ||||
|  | ||||
| out/macos%/tildefriends: out/macos%-arm/tildefriends out/macos%-x86_64/tildefriends | ||||
| 	@echo [lipo] $@ | ||||
| 	@mkdir -p $(@D) | ||||
|   | ||||
| @@ -38,8 +38,6 @@ dependencies in the right places. | ||||
|  | ||||
| ### Requirements | ||||
|  | ||||
| System OpenSSL libraries are assumed to be available on Haiku and OpenBSD. | ||||
|  | ||||
| On MacOS, Xcode's command-line tools are expected to be available. | ||||
|  | ||||
| ### Build Commands | ||||
|   | ||||
| @@ -1,5 +1,5 @@ | ||||
| { | ||||
| 	"type": "tildefriends-app", | ||||
| 	"emoji": "📜", | ||||
| 	"previous": "&BEf0nraBdHk/+PWqx6tOSu5rheWVaxaL7orAOz3285M=.sha256" | ||||
| 	"previous": "&sJqeyYjHys6Z8IqqtZ2ij2ZC1E2xieu/FU/u2hE+O1U=.sha256" | ||||
| } | ||||
|   | ||||
| @@ -55,6 +55,9 @@ app.setDocument(`<head> | ||||
| </head> | ||||
| <body style="color:#fff"> | ||||
| 	${markdown(docs.docs.global)} | ||||
| 	<!-- | ||||
| 	${Object.keys(docs.docs).filter(x => [...treeify('', globalThis)].indexOf(x) == -1).map(x => `<p>STALE: ${x}</p>`).join('')} | ||||
| 	--> | ||||
| 	${[...treeify('', globalThis)].map(x => document(x)).join('\n')} | ||||
| 	<a id="Database"></a> | ||||
| 	${markdown(docs.docs.database)} | ||||
|   | ||||
| @@ -195,51 +195,6 @@ Call a function after some delay. | ||||
|  * *Number* **timeout** Number of milliseconds to wait before calling the callback function. | ||||
| `; | ||||
|  | ||||
| docs['parseHttpRequest()'] = ` | ||||
| Parses an HTTP request. | ||||
| ### Parameters | ||||
|  * *Uint8Array* **request** The request data.  Maybe be partial or contain extra data.  The return value will | ||||
|     indicate when and where it is complete. | ||||
|  * *Number* **last_length** The length of the data passed on a previous attempt for the same request, or 0 initially. | ||||
| ### Returns | ||||
|  * *Integer* **-2** if the request is incomplete. | ||||
|  * *Integer* **-1** if the request could not be parsed. | ||||
|  * *Object* An object with **bytes_parsed**, **minor_version**, **path**, and **headers** fields on successful parse. | ||||
| `; | ||||
|  | ||||
| docs['parseHttpResponse()'] = ` | ||||
| Parses an HTTP response. | ||||
| ### Parameters | ||||
|  * *Uint8Array* **response** The response data.  Maybe be partial or contain extra data.  The return value will | ||||
|     indicate when and where it is complete. | ||||
|  * *Number* **last_length** The length of the data passed on a previous attempt for the same response, or 0 initially. | ||||
| ### Returns | ||||
|  * *Integer* **-2** if the response is incomplete. | ||||
|  * *Integer* **-1** if the response could not be parsed. | ||||
|  * *Object* An object with **bytes_parsed**, **minor_version**, **status**, **message**, and **headers** fields on successful parse. | ||||
| `; | ||||
|  | ||||
| docs['sha1Digest()'] = ` | ||||
| Calculates a SHA1 digest. | ||||
|  | ||||
| Completes synchronously. | ||||
| ### Parameters | ||||
|  * *String* **value** The value for which to calculate the digest. | ||||
| ### Returns | ||||
| *String* The SHA1 digest of UTF-8 encoded \`value\`. | ||||
| `; | ||||
|  | ||||
| docs['maskBytes()'] = ` | ||||
| Masks bytes for WebSocket communication. | ||||
|  | ||||
| Completes synchronously. | ||||
| ### Parameters | ||||
|  * *Uint8Array* **bytes** The byte array of data to mask. | ||||
|  * *Uint32* **mask** The mask to apply. | ||||
| ### Returns | ||||
| *Uint32Array* The masked bytes. | ||||
| `; | ||||
|  | ||||
| docs['exit()'] = ` | ||||
| Exits the app.  But why would you want to do that? | ||||
|  | ||||
|   | ||||
| @@ -1,5 +1,5 @@ | ||||
| { | ||||
| 	"type": "tildefriends-app", | ||||
| 	"emoji": "🦀", | ||||
| 	"previous": "&01jXxJgs24zTcJk+csXeUWfm/MQ/+94Zy7K0r2OYmWw=.sha256" | ||||
| 	"previous": "&ceivZzkWd2Bvvdzdvdv68bZm2Y9oC0J352C78ZmMMU4=.sha256" | ||||
| } | ||||
|   | ||||
| @@ -1,6 +1,6 @@ | ||||
| import * as tfrpc from '/static/tfrpc.js'; | ||||
| import {html, render} from './lit-all.min.js'; | ||||
| import {styles} from './tf-styles.js'; | ||||
| import {styles, generate_theme} from './tf-styles.js'; | ||||
|  | ||||
| let g_emojis; | ||||
|  | ||||
| @@ -140,6 +140,9 @@ export async function picker(callback, anchor, author, recent) { | ||||
| 		<style> | ||||
| 			${styles} | ||||
| 		</style> | ||||
| 		<style> | ||||
| 			${generate_theme()} | ||||
| 		</style> | ||||
| 		<div | ||||
| 			class="w3-modal" | ||||
| 			style="display: block; box-sizing: border-box; z-index: 10" | ||||
|   | ||||
| @@ -12,12 +12,14 @@ import * as tf_tab_news from './tf-tab-news.js'; | ||||
| import * as tf_tab_news_feed from './tf-tab-news-feed.js'; | ||||
| import * as tf_tab_search from './tf-tab-search.js'; | ||||
| import * as tf_tab_connections from './tf-tab-connections.js'; | ||||
| import * as tf_tab_query from './tf-tab-query.js'; | ||||
| import * as tf_tag from './tf-tag.js'; | ||||
| import * as tf_styles from './tf-styles.js'; | ||||
|  | ||||
| window.addEventListener('load', function () { | ||||
| 	let style = document.createElement('style'); | ||||
| 	style.innerText = tf_styles.styles; | ||||
| 	Promise.resolve(tf_styles.generate_theme()).then(function (x) { | ||||
| 		style.innerText += x; | ||||
| 	}); | ||||
| 	document.body.appendChild(style); | ||||
| }); | ||||
|   | ||||
| @@ -1,6 +1,6 @@ | ||||
| import {LitElement, html, css, guard, until} from './lit-all.min.js'; | ||||
| import * as tfrpc from '/static/tfrpc.js'; | ||||
| import {styles} from './tf-styles.js'; | ||||
| import {styles, generate_theme} from './tf-styles.js'; | ||||
|  | ||||
| class TfElement extends LitElement { | ||||
| 	static get properties() { | ||||
| @@ -206,8 +206,6 @@ class TfElement extends LitElement { | ||||
| 			this.tab = 'search'; | ||||
| 		} else if (this.hash === '#connections') { | ||||
| 			this.tab = 'connections'; | ||||
| 		} else if (this.hash.startsWith('#sql=')) { | ||||
| 			this.tab = 'query'; | ||||
| 		} else { | ||||
| 			this.tab = 'news'; | ||||
| 		} | ||||
| @@ -736,17 +734,6 @@ class TfElement extends LitElement { | ||||
| 						: null} | ||||
| 				></tf-tab-search> | ||||
| 			`; | ||||
| 		} else if (this.tab === 'query') { | ||||
| 			return html` | ||||
| 				<tf-tab-query | ||||
| 					.following=${this.following} | ||||
| 					whoami=${this.whoami} | ||||
| 					.users=${this.users} | ||||
| 					query=${this.hash?.startsWith('#sql=') | ||||
| 						? this.hash.substring(5) | ||||
| 						: null} | ||||
| 				></tf-tab-query> | ||||
| 			`; | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| @@ -757,8 +744,6 @@ class TfElement extends LitElement { | ||||
| 			await tfrpc.rpc.setHash('#'); | ||||
| 		} else if (tab === 'connections') { | ||||
| 			await tfrpc.rpc.setHash('#connections'); | ||||
| 		} else if (tab === 'query') { | ||||
| 			await tfrpc.rpc.setHash('#sql='); | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| @@ -778,6 +763,17 @@ class TfElement extends LitElement { | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	async pick_color() { | ||||
| 		let input = document.createElement('input'); | ||||
| 		input.type = 'color'; | ||||
| 		input.value = (await tfrpc.rpc.localStorageGet('color')) ?? '#ff0000'; | ||||
| 		input.addEventListener('change', async function () { | ||||
| 			await tfrpc.rpc.localStorageSet('color', input.value); | ||||
| 			window.location.reload(); | ||||
| 		}); | ||||
| 		input.click(); | ||||
| 	} | ||||
|  | ||||
| 	render() { | ||||
| 		let self = this; | ||||
|  | ||||
| @@ -792,7 +788,6 @@ class TfElement extends LitElement { | ||||
| 			'📰': 'news', | ||||
| 			'📡': 'connections', | ||||
| 			'🔍': 'search', | ||||
| 			'👩💻': 'query', | ||||
| 		}; | ||||
|  | ||||
| 		let tabs = html` | ||||
| @@ -835,6 +830,12 @@ class TfElement extends LitElement { | ||||
| 						</button> | ||||
| 					` | ||||
| 				)} | ||||
| 				<button | ||||
| 					class="w3-bar-item w3-button w3-right" | ||||
| 					@click=${this.pick_color} | ||||
| 				> | ||||
| 					🎨<span class="w3-hide-small">Color</span> | ||||
| 				</button> | ||||
| 			</div> | ||||
| 		`; | ||||
| 		let contents = this.guest | ||||
| @@ -870,6 +871,9 @@ class TfElement extends LitElement { | ||||
| 					` | ||||
| 				: undefined; | ||||
| 		return html` | ||||
| 			<style> | ||||
| 				${generate_theme()} | ||||
| 			</style> | ||||
| 			<div | ||||
| 				style="width: 100vw; min-height: 100vh; height: 100vh; display: flex; flex-direction: column" | ||||
| 				class="w3-theme-dark" | ||||
|   | ||||
| @@ -1,7 +1,7 @@ | ||||
| import {LitElement, html, unsafeHTML, live} from './lit-all.min.js'; | ||||
| import * as tfutils from './tf-utils.js'; | ||||
| import * as tfrpc from '/static/tfrpc.js'; | ||||
| import {styles} from './tf-styles.js'; | ||||
| import {styles, generate_theme} from './tf-styles.js'; | ||||
| import Tribute from './tribute.esm.js'; | ||||
|  | ||||
| class TfComposeElement extends LitElement { | ||||
| @@ -603,6 +603,9 @@ class TfComposeElement extends LitElement { | ||||
| 						🔐 Encrypt | ||||
| 					</button>`; | ||||
| 		let result = html` | ||||
| 			<style> | ||||
| 				${generate_theme()} | ||||
| 			</style> | ||||
| 			<style> | ||||
| 				.w3-input:empty::before { | ||||
| 					content: attr(placeholder); | ||||
|   | ||||
| @@ -4,12 +4,14 @@ import { | ||||
| 	html, | ||||
| 	repeat, | ||||
| 	render, | ||||
| 	unsafeCSS, | ||||
| 	unsafeHTML, | ||||
| 	until, | ||||
| } from './lit-all.min.js'; | ||||
| import * as tfrpc from '/static/tfrpc.js'; | ||||
| import * as tfutils from './tf-utils.js'; | ||||
| import * as emojis from './emojis.js'; | ||||
| import {styles} from './tf-styles.js'; | ||||
| import {styles, generate_theme} from './tf-styles.js'; | ||||
|  | ||||
| class TfMessageElement extends LitElement { | ||||
| 	static get properties() { | ||||
| @@ -24,6 +26,7 @@ class TfMessageElement extends LitElement { | ||||
| 			channel: {type: String}, | ||||
| 			channel_unread: {type: Number}, | ||||
| 			recent_reactions: {type: Array}, | ||||
| 			depth: {type: Number}, | ||||
| 		}; | ||||
| 	} | ||||
|  | ||||
| @@ -40,6 +43,7 @@ class TfMessageElement extends LitElement { | ||||
| 		this.expanded = {}; | ||||
| 		this.channel_unread = -1; | ||||
| 		this.recent_reactions = []; | ||||
| 		this.depth = 0; | ||||
| 	} | ||||
|  | ||||
| 	connectedCallback() { | ||||
| @@ -324,7 +328,9 @@ class TfMessageElement extends LitElement { | ||||
| 	} | ||||
|  | ||||
| 	expanded_key() { | ||||
| 		return this.message?.id || this.messages?.map((x) => x.id).join(':'); | ||||
| 		return ( | ||||
| 			this.message?.id || this.message?.messages?.map((x) => x.id).join(':') | ||||
| 		); | ||||
| 	} | ||||
|  | ||||
| 	set_expanded(expanded, tag) { | ||||
| @@ -362,12 +368,13 @@ class TfMessageElement extends LitElement { | ||||
| 					</button> | ||||
| 				`; | ||||
| 			} else { | ||||
| 				return html` <div class="w3-container w3-margin-bottom"> | ||||
| 				return html` <ul class="w3-container w3-margin-bottom w3-ul w3-card-4"> | ||||
| 					${repeat( | ||||
| 						this.message.child_messages || [], | ||||
| 						(x) => x.id, | ||||
| 						(x) => | ||||
| 								html`<tf-message | ||||
| 							html`<li style="padding: 0"> | ||||
| 								<tf-message | ||||
| 									.message=${x} | ||||
| 									whoami=${this.whoami} | ||||
| 									.users=${this.users} | ||||
| @@ -376,16 +383,20 @@ class TfMessageElement extends LitElement { | ||||
| 									channel=${this.channel} | ||||
| 									channel_unread=${this.channel_unread} | ||||
| 									.recent_reactions=${this.recent_reactions} | ||||
| 								></tf-message>` | ||||
| 									depth=${this.depth + 1} | ||||
| 								></tf-message> | ||||
| 							</li>` | ||||
| 					)} | ||||
| 					</div> | ||||
| 					<li style="padding: 0" class="w3-margin-bottom"> | ||||
| 						<button | ||||
| 							class="w3-button w3-theme-d1 w3-block w3-bar" | ||||
| 							style="box-sizing: border-box" | ||||
| 							@click=${() => self.set_expanded(false)} | ||||
| 						> | ||||
| 							Collapse | ||||
| 					</button>`; | ||||
| 						</button> | ||||
| 					</li> | ||||
| 				</ul>`; | ||||
| 			} | ||||
| 		} else { | ||||
| 			return undefined; | ||||
| @@ -549,7 +560,10 @@ class TfMessageElement extends LitElement { | ||||
| 				} | ||||
| 			</style> | ||||
| 			<div | ||||
| 				class="w3-card-4 ${this.class_background()} w3-border-theme w3-margin-top" | ||||
| 				class="w3-card-4 ${this.class_background()} w3-border-theme ${this | ||||
| 					.depth == 0 | ||||
| 					? 'w3-margin-top' | ||||
| 					: ''}" | ||||
| 				style="overflow-wrap: anywhere; display: block; max-width: 100%" | ||||
| 			> | ||||
| 				${inner} | ||||
| @@ -576,6 +590,7 @@ class TfMessageElement extends LitElement { | ||||
| 						channel=${self.channel} | ||||
| 						channel_unread=${self.channel_unread} | ||||
| 						.recent_reactions=${self.recent_reactions} | ||||
| 						depth=${self.depth + 1} | ||||
| 					></tf-message> | ||||
| 				` | ||||
| 			)} | ||||
| @@ -611,14 +626,16 @@ class TfMessageElement extends LitElement { | ||||
| 		let sorted = this.message.messages | ||||
| 			.map((x) => [ | ||||
| 				x.author, | ||||
| 				x.content.blocking !== undefined | ||||
| 					? x.content.blocking | ||||
| 						? 'is blocking' | ||||
| 						: 'is no longer blocking' | ||||
| 					: x.content.following !== undefined | ||||
| 						? x.content.following | ||||
| 				x.content.following && x.content.blocking | ||||
| 					? 'is following and blocking' | ||||
| 					: x.content.following | ||||
| 						? 'is following' | ||||
| 							: 'is no longer following' | ||||
| 						: x.content.blocking | ||||
| 							? 'is blocking' | ||||
| 							: x.content.blocking !== undefined | ||||
| 								? 'is no longer blocking' | ||||
| 								: x.content.following !== undefined | ||||
| 									? 'is no longer following' | ||||
| 									: '', | ||||
| 				x.content.contact, | ||||
| 				x, | ||||
| @@ -686,7 +703,7 @@ class TfMessageElement extends LitElement { | ||||
| 			: undefined; | ||||
| 	} | ||||
|  | ||||
| 	render() { | ||||
| 	_render() { | ||||
| 		let content = this.message?.content; | ||||
| 		if (this.message?.decrypted?.type == 'post') { | ||||
| 			content = this.message.decrypted; | ||||
| @@ -707,6 +724,7 @@ class TfMessageElement extends LitElement { | ||||
| 									.expanded=${this.expanded} | ||||
| 									channel=${this.channel} | ||||
| 									channel_unread=${this.channel_unread} | ||||
| 									depth=${this.depth + 1} | ||||
| 								></tf-message>` | ||||
| 						)} | ||||
| 					</div> | ||||
| @@ -762,6 +780,7 @@ class TfMessageElement extends LitElement { | ||||
| 									.expanded=${this.expanded} | ||||
| 									channel=${this.channel} | ||||
| 									channel_unread=${this.channel_unread} | ||||
| 									depth=${this.depth + 1} | ||||
| 								></tf-message>` | ||||
| 						)} | ||||
| 					</div> | ||||
| @@ -842,6 +861,7 @@ class TfMessageElement extends LitElement { | ||||
| 								.expanded=${this.expanded} | ||||
| 								channel=${this.channel} | ||||
| 								channel_unread=${this.channel_unread} | ||||
| 								depth=${this.depth + 1} | ||||
| 							></tf-message> | ||||
| 						` | ||||
| 					)} | ||||
| @@ -1077,6 +1097,15 @@ class TfMessageElement extends LitElement { | ||||
| 			return this.render_small_frame(this.render_raw()); | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	render() { | ||||
| 		return html` | ||||
| 			<style> | ||||
| 				${generate_theme()} | ||||
| 			</style> | ||||
| 			${this._render()} | ||||
| 		`; | ||||
| 	} | ||||
| } | ||||
|  | ||||
| customElements.define('tf-message', TfMessageElement); | ||||
|   | ||||
| @@ -1,6 +1,6 @@ | ||||
| import {LitElement, html, unsafeHTML, repeat, until} from './lit-all.min.js'; | ||||
| import * as tfrpc from '/static/tfrpc.js'; | ||||
| import {styles} from './tf-styles.js'; | ||||
| import {styles, generate_theme} from './tf-styles.js'; | ||||
|  | ||||
| class TfNewsElement extends LitElement { | ||||
| 	static get properties() { | ||||
| @@ -231,6 +231,9 @@ class TfNewsElement extends LitElement { | ||||
| 			} | ||||
| 		} | ||||
| 		return html` | ||||
| 			<style> | ||||
| 				${generate_theme()} | ||||
| 			</style> | ||||
| 			<div> | ||||
| 				${repeat( | ||||
| 					final_messages, | ||||
|   | ||||
| @@ -1,7 +1,7 @@ | ||||
| import {LitElement, html, until, unsafeHTML} from './lit-all.min.js'; | ||||
| import * as tfrpc from '/static/tfrpc.js'; | ||||
| import * as tfutils from './tf-utils.js'; | ||||
| import {styles} from './tf-styles.js'; | ||||
| import {styles, generate_theme} from './tf-styles.js'; | ||||
|  | ||||
| class TfProfileElement extends LitElement { | ||||
| 	static get properties() { | ||||
| @@ -316,7 +316,9 @@ class TfProfileElement extends LitElement { | ||||
| 		} | ||||
| 		image = this.editing?.image ?? image; | ||||
| 		let description = this.editing?.description ?? profile.description; | ||||
| 		return html`<div class="w3-card-4 w3-container w3-theme-d3" style="box-sizing: border-box"> | ||||
| 		return html` | ||||
| 			<style>${generate_theme()}</style> | ||||
| 			<div class="w3-card-4 w3-container w3-theme-d3" style="box-sizing: border-box"> | ||||
| 			<header class="w3-container"> | ||||
| 				<p><tf-user id=${this.id} .users=${this.users}></tf-user> (${tfutils.human_readable_size(this.size)} in ${this.sequence} messages)</p> | ||||
| 			</header> | ||||
|   | ||||
| @@ -1,5 +1,5 @@ | ||||
| import {LitElement, html, unsafeHTML} from './lit-all.min.js'; | ||||
| import {styles} from './tf-styles.js'; | ||||
| import {styles, generate_theme} from './tf-styles.js'; | ||||
|  | ||||
| class TfReactionsModalElement extends LitElement { | ||||
| 	static get properties() { | ||||
| @@ -24,7 +24,10 @@ class TfReactionsModalElement extends LitElement { | ||||
| 	render() { | ||||
| 		let self = this; | ||||
| 		return this.votes?.length | ||||
| 			? html` <div | ||||
| 			? html` <style> | ||||
| 						${generate_theme()} | ||||
| 					</style> | ||||
| 					<div | ||||
| 						class="w3-modal w3-animate-opacity" | ||||
| 						style="display: block; box-sizing: border-box; z-index: 10" | ||||
| 						@click=${this.clear} | ||||
| @@ -36,7 +39,9 @@ class TfReactionsModalElement extends LitElement { | ||||
| 							<div class="w3-container w3-padding"> | ||||
| 								<header class="w3-container"> | ||||
| 									<h2>Reactions</h2> | ||||
| 								<span class="w3-button w3-display-topright" @click=${this.clear} | ||||
| 									<span | ||||
| 										class="w3-button w3-display-topright" | ||||
| 										@click=${this.clear} | ||||
| 										>×</span | ||||
| 									> | ||||
| 								</header> | ||||
| @@ -45,7 +50,9 @@ class TfReactionsModalElement extends LitElement { | ||||
| 										.sort((x, y) => y.timestamp - x.timestamp) | ||||
| 										.map( | ||||
| 											(x) => html` | ||||
| 											<li style="display: flex; flex-direction: row; gap: 4px"> | ||||
| 												<li | ||||
| 													style="display: flex; flex-direction: row; gap: 4px" | ||||
| 												> | ||||
| 													<span style="flex-basis: 3em" | ||||
| 														>${x?.content?.vote?.expression}</span | ||||
| 													> | ||||
|   | ||||
| @@ -1,4 +1,5 @@ | ||||
| import {css, unsafeCSS} from './lit-all.min.js'; | ||||
| import {css, unsafeCSS, until} from './lit-all.min.js'; | ||||
| import * as tfrpc from '/static/tfrpc.js'; | ||||
|  | ||||
| const tf = css` | ||||
| 	img { | ||||
| @@ -408,16 +409,8 @@ function is_dark(hex, value) { | ||||
| 	return (r * 299 + g * 587 + b * 114) / 1000 < value; | ||||
| } | ||||
|  | ||||
| function generated() { | ||||
| 	let now = new Date(); | ||||
| 	let k_color = rgb_to_hex([ | ||||
| 		(now.getDay() * 128) / 6, | ||||
| 		(now.getHours() * 128) / 23, | ||||
| 		(now.getSeconds() * 128) / 59, | ||||
| 	]); | ||||
| 	//let k_color = '#034f84'; | ||||
| 	//let k_color = rgb_to_hex([Math.random() * 256, Math.random() * 256, Math.random() * 256]); | ||||
| 	let [r, g, b] = hex_to_rgb(k_color); | ||||
| export function generate(color) { | ||||
| 	let [r, g, b] = hex_to_rgb(color); | ||||
| 	let [h, s, l] = rgb_to_hsl(r, g, b); | ||||
|  | ||||
| 	let theme1 = { | ||||
| @@ -461,4 +454,28 @@ function generated() { | ||||
| 	return unsafeCSS(result); | ||||
| } | ||||
|  | ||||
| export let styles = [tf, w3, generated()]; | ||||
| let g_theme; | ||||
| export function generate_theme() { | ||||
| 	return g_theme | ||||
| 		? g_theme | ||||
| 		: until( | ||||
| 				tfrpc.rpc.localStorageGet('color').then(function (value) { | ||||
| 					g_theme = generate(value ?? '#034f84'); | ||||
| 					return g_theme; | ||||
| 				}), | ||||
| 				generated_now() | ||||
| 			); | ||||
| } | ||||
|  | ||||
| function generated_now() { | ||||
| 	let now = new Date(); | ||||
| 	return generate( | ||||
| 		rgb_to_hex([ | ||||
| 			(now.getDay() * 128) / 6, | ||||
| 			(now.getHours() * 128) / 23, | ||||
| 			(now.getSeconds() * 128) / 59, | ||||
| 		]) | ||||
| 	); | ||||
| } | ||||
|  | ||||
| export let styles = [tf, w3]; | ||||
|   | ||||
| @@ -1,6 +1,6 @@ | ||||
| import {LitElement, html} from './lit-all.min.js'; | ||||
| import * as tfrpc from '/static/tfrpc.js'; | ||||
| import {styles} from './tf-styles.js'; | ||||
| import {styles, generate_theme} from './tf-styles.js'; | ||||
|  | ||||
| class TfTabConnectionsElement extends LitElement { | ||||
| 	static get properties() { | ||||
| @@ -251,6 +251,9 @@ class TfTabConnectionsElement extends LitElement { | ||||
| 	render() { | ||||
| 		let self = this; | ||||
| 		return html` | ||||
| 			<style> | ||||
| 				${generate_theme()} | ||||
| 			</style> | ||||
| 			<div class="w3-container" style="box-sizing: border-box"> | ||||
| 				<h2>New Connection</h2> | ||||
| 				<textarea class="w3-input w3-theme-d1" id="code"></textarea> | ||||
|   | ||||
| @@ -1,6 +1,6 @@ | ||||
| import {LitElement, cache, html, unsafeHTML, until} from './lit-all.min.js'; | ||||
| import * as tfrpc from '/static/tfrpc.js'; | ||||
| import {styles} from './tf-styles.js'; | ||||
| import {styles, generate_theme} from './tf-styles.js'; | ||||
|  | ||||
| class TfTabNewsFeedElement extends LitElement { | ||||
| 	static get properties() { | ||||
| @@ -256,14 +256,24 @@ class TfTabNewsFeedElement extends LitElement { | ||||
| 			let t0 = new Date(); | ||||
| 			let initial_messages = await tfrpc.rpc.query( | ||||
| 				` | ||||
| 					WITH | ||||
| 						channels AS (SELECT '#' || value AS value FROM json_each(?5)) | ||||
| 					SELECT TRUE AS is_primary, messages.rowid, messages.id, messages.previous, messages.author, messages.sequence, messages.timestamp, messages.hash, json(messages.content) AS content, messages.signature | ||||
| 							FROM messages | ||||
| 					JOIN json_each(?) AS following ON messages.author = following.value | ||||
| 							JOIN json_each(?1) AS following ON messages.author = following.value | ||||
| 							WHERE messages.timestamp < ?3 AND (?2 IS NULL OR messages.timestamp >= ?2) AND | ||||
| 						messages.content ->> 'type' != 'vote' | ||||
| 								messages.content ->> 'type' != 'vote' AND | ||||
| 								(messages.content ->> 'channel' IS NULL OR ('#' || (messages.content ->> 'channel')) NOT IN (SELECT * FROM channels)) AND | ||||
| 								NOT EXISTS (SELECT * FROM messages_refs JOIN channels ON messages_refs.message = messages.id AND messages_refs.ref = channels.value) | ||||
| 					ORDER BY timestamp DESC LIMIT ?4 | ||||
| 				`, | ||||
| 				[JSON.stringify(this.following), start_time, end_time, k_max_results] | ||||
| 				[ | ||||
| 					JSON.stringify(this.following), | ||||
| 					start_time, | ||||
| 					end_time, | ||||
| 					k_max_results, | ||||
| 					JSON.stringify(Object.keys(this.channels_latest)), | ||||
| 				] | ||||
| 			); | ||||
| 			let t1 = new Date(); | ||||
| 			result = await this._fetch_related_messages(initial_messages); | ||||
| @@ -396,6 +406,9 @@ class TfTabNewsFeedElement extends LitElement { | ||||
| 			this._private_messages = | ||||
| 				JSON.stringify(this.private_messages) + | ||||
| 				JSON.stringify(this.grouped_private_messages); | ||||
| 			this._channels_latest = JSON.stringify( | ||||
| 				Object.keys(this.channels_latest ?? {}) | ||||
| 			); | ||||
| 			let now = new Date().valueOf(); | ||||
| 			let start_time = now - 24 * 60 * 60 * 1000; | ||||
| 			this.start_time = start_time; | ||||
| @@ -471,7 +484,9 @@ class TfTabNewsFeedElement extends LitElement { | ||||
| 			this._messages_following !== JSON.stringify(this.following) || | ||||
| 			this._private_messages !== | ||||
| 				JSON.stringify(this.private_messages) + | ||||
| 					JSON.stringify(this.grouped_private_messages) | ||||
| 					JSON.stringify(this.grouped_private_messages) || | ||||
| 			this._channels_latest !== | ||||
| 				JSON.stringify(Object.keys(this.channels_latest)) | ||||
| 		) { | ||||
| 			console.log( | ||||
| 				`loading messages for ${this.whoami} (following ${this.following.length})` | ||||
| @@ -523,6 +538,9 @@ class TfTabNewsFeedElement extends LitElement { | ||||
| 			`; | ||||
| 		} | ||||
| 		return cache(html` | ||||
| 			<style> | ||||
| 				${generate_theme()} | ||||
| 			</style> | ||||
| 			${this.unread_allowed() | ||||
| 				? html`<button | ||||
| 						class="w3-button w3-theme-d1" | ||||
|   | ||||
| @@ -7,7 +7,7 @@ import { | ||||
| 	until, | ||||
| } from './lit-all.min.js'; | ||||
| import * as tfrpc from '/static/tfrpc.js'; | ||||
| import {styles} from './tf-styles.js'; | ||||
| import {styles, generate_theme} from './tf-styles.js'; | ||||
|  | ||||
| class TfTabNewsElement extends LitElement { | ||||
| 	static get properties() { | ||||
| @@ -211,35 +211,6 @@ class TfTabNewsElement extends LitElement { | ||||
| 				> | ||||
| 					× | ||||
| 				</div> | ||||
| 				${this.is_administrator | ||||
| 					? html` | ||||
| 							<button | ||||
| 								class="w3-bar-item w3-button" | ||||
| 								@click=${() => | ||||
| 									this.dispatchEvent( | ||||
| 										new Event('refresh', {bubbles: true, composed: true}) | ||||
| 									)} | ||||
| 							> | ||||
| 								<span style="display: inline-block; width: 1.8em">↻</span> | ||||
| 								Sync now | ||||
| 							</button> | ||||
| 							<button | ||||
| 								class="w3-bar-item w3-button w3-ripple" | ||||
| 								@click=${() => | ||||
| 									this.dispatchEvent( | ||||
| 										new Event('toggle_stay_connected', { | ||||
| 											bubbles: true, | ||||
| 											composed: true, | ||||
| 										}) | ||||
| 									)} | ||||
| 							> | ||||
| 								<span style="display: inline-block; width: 1.8em" | ||||
| 									>${this.stay_connected ? '🔗' : '⛓️💥'}</span | ||||
| 								> | ||||
| 								${this.stay_connected ? 'Online mode' : 'Passive mode'} | ||||
| 							</button> | ||||
| 						` | ||||
| 					: undefined} | ||||
| 				${this.hash.startsWith('##') && | ||||
| 				this.channels.indexOf(this.hash.substring(2)) == -1 | ||||
| 					? html` | ||||
| @@ -334,6 +305,21 @@ class TfTabNewsElement extends LitElement { | ||||
| 							> | ||||
| 								↻ Sync now | ||||
| 							</button> | ||||
| 							<button | ||||
| 								class="w3-bar-item w3-button w3-ripple" | ||||
| 								@click=${() => | ||||
| 									this.dispatchEvent( | ||||
| 										new Event('toggle_stay_connected', { | ||||
| 											bubbles: true, | ||||
| 											composed: true, | ||||
| 										}) | ||||
| 									)} | ||||
| 							> | ||||
| 								<span style="display: inline-block; width: 1.8em" | ||||
| 									>${this.stay_connected ? '🔗' : '⛓️💥'}</span | ||||
| 								> | ||||
| 								${this.stay_connected ? 'Online mode' : 'Passive mode'} | ||||
| 							</button> | ||||
| 							<button | ||||
| 								class=${'w3-bar-item w3-button' + | ||||
| 								(this.peer_exchange !== false ? ' w3-hide' : '')} | ||||
| @@ -409,6 +395,9 @@ class TfTabNewsElement extends LitElement { | ||||
| 			</div>`; | ||||
| 		} | ||||
| 		return cache(html` | ||||
| 			<style> | ||||
| 				${generate_theme()} | ||||
| 			</style> | ||||
| 			${this.render_sidebar()} | ||||
| 			<div | ||||
| 				style="margin-left: 2in; padding: 0px; top: 0; height: 100vh; max-height: 100%; overflow: auto; contain: layout" | ||||
|   | ||||
| @@ -1,136 +0,0 @@ | ||||
| import {LitElement, html, unsafeHTML} from './lit-all.min.js'; | ||||
| import * as tfrpc from '/static/tfrpc.js'; | ||||
| import {styles} from './tf-styles.js'; | ||||
|  | ||||
| class TfTabQueryElement extends LitElement { | ||||
| 	static get properties() { | ||||
| 		return { | ||||
| 			whoami: {type: String}, | ||||
| 			users: {type: Object}, | ||||
| 			following: {type: Array}, | ||||
| 			query: {type: String}, | ||||
| 			expanded: {type: Object}, | ||||
| 			results: {type: Array}, | ||||
| 			error: {type: Object}, | ||||
| 			duration: {type: Number}, | ||||
| 		}; | ||||
| 	} | ||||
|  | ||||
| 	static styles = styles; | ||||
|  | ||||
| 	constructor() { | ||||
| 		super(); | ||||
| 		let self = this; | ||||
| 		this.whoami = null; | ||||
| 		this.users = {}; | ||||
| 		this.following = []; | ||||
| 		this.expanded = {}; | ||||
| 		this.duration = undefined; | ||||
| 	} | ||||
|  | ||||
| 	async search(query) { | ||||
| 		console.log('Searching...', this.whoami, query); | ||||
| 		this.results = []; | ||||
| 		this.error = undefined; | ||||
| 		this.duration = undefined; | ||||
| 		let search = this.renderRoot.getElementById('search'); | ||||
| 		if (search) { | ||||
| 			search.value = query; | ||||
| 			search.focus(); | ||||
| 		} | ||||
| 		await tfrpc.rpc.setHash('#sql=' + encodeURIComponent(query)); | ||||
| 		let start_time = new Date(); | ||||
| 		try { | ||||
| 			this.results = await tfrpc.rpc.query(query, []); | ||||
| 		} catch (error) { | ||||
| 			this.error = error; | ||||
| 		} | ||||
| 		let end_time = new Date(); | ||||
| 		this.duration = (end_time - start_time).valueOf(); | ||||
| 		console.log('Done.'); | ||||
| 		search = this.renderRoot.getElementById('search'); | ||||
| 		if (search) { | ||||
| 			search.value = query; | ||||
| 			search.focus(); | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	search_keydown(event) { | ||||
| 		if (event.keyCode == 13 && event.ctrlKey) { | ||||
| 			this.query = this.renderRoot.getElementById('search').value; | ||||
| 			event.preventDefault(); | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	on_expand(event) { | ||||
| 		if (event.detail.expanded) { | ||||
| 			let expand = {}; | ||||
| 			expand[event.detail.id] = true; | ||||
| 			this.expanded = Object.assign({}, this.expanded, expand); | ||||
| 		} else { | ||||
| 			delete this.expanded[event.detail.id]; | ||||
| 			this.expanded = Object.assign({}, this.expanded); | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	render_results() { | ||||
| 		if (!this.results?.length) { | ||||
| 			return html`<div>No results.</div>`; | ||||
| 		} else { | ||||
| 			let keys = Object.keys(this.results[0]).sort(); | ||||
| 			return html`<table style="width: 100%; max-width: 100%"> | ||||
| 				<tr> | ||||
| 					${keys.map((key) => html`<th>${key}</th>`)} | ||||
| 				</tr> | ||||
| 				${this.results.map( | ||||
| 					(row) => | ||||
| 						html`<tr> | ||||
| 							${keys.map((key) => html`<td>${row[key]}</td>`)} | ||||
| 						</tr>` | ||||
| 				)} | ||||
| 			</table>`; | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	render_error() { | ||||
| 		if (this.error) { | ||||
| 			return html`<h2 style="color: red">${this.error.message}</h2> | ||||
| 				<pre style="color: red">${this.error.stack}</pre>`; | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	render() { | ||||
| 		if (this.query !== this.last_query) { | ||||
| 			this.last_query = this.query; | ||||
| 			this.search(this.query); | ||||
| 		} | ||||
| 		let self = this; | ||||
| 		return html` | ||||
| 			<div style="display: flex; flex-direction: row; gap: 4px"> | ||||
| 				<textarea | ||||
| 					id="search" | ||||
| 					rows="8" | ||||
| 					class="w3-input w3-theme-d1" | ||||
| 					style="flex: 1; resize: vertical" | ||||
| 					@keydown=${this.search_keydown} | ||||
| 				> | ||||
| ${this.query}</textarea | ||||
| 				> | ||||
| 				<button | ||||
| 					class="w3-button w3-theme-d1" | ||||
| 					@click=${(event) => | ||||
| 						self.search(self.renderRoot.getElementById('search').value)} | ||||
| 				> | ||||
| 					Execute | ||||
| 				</button> | ||||
| 			</div> | ||||
| 			<div ?hidden=${this.duration === undefined}> | ||||
| 				Took ${this.duration / 1000.0} seconds. | ||||
| 			</div> | ||||
| 			<div ?hidden=${this.duration !== undefined}>Executing...</div> | ||||
| 			${this.render_error()} ${this.render_results()} | ||||
| 		`; | ||||
| 	} | ||||
| } | ||||
|  | ||||
| customElements.define('tf-tab-query', TfTabQueryElement); | ||||
| @@ -1,6 +1,6 @@ | ||||
| import {LitElement, html, unsafeHTML} from './lit-all.min.js'; | ||||
| import * as tfrpc from '/static/tfrpc.js'; | ||||
| import {styles} from './tf-styles.js'; | ||||
| import {styles, generate_theme} from './tf-styles.js'; | ||||
|  | ||||
| class TfTabSearchElement extends LitElement { | ||||
| 	static get properties() { | ||||
| @@ -11,6 +11,9 @@ class TfTabSearchElement extends LitElement { | ||||
| 			following: {type: Array}, | ||||
| 			query: {type: String}, | ||||
| 			expanded: {type: Object}, | ||||
| 			messages: {type: Array}, | ||||
| 			results: {type: Array}, | ||||
| 			error: {type: Object}, | ||||
| 		}; | ||||
| 	} | ||||
|  | ||||
| @@ -38,6 +41,21 @@ class TfTabSearchElement extends LitElement { | ||||
| 			search.select(); | ||||
| 		} | ||||
| 		await tfrpc.rpc.setHash('#q=' + encodeURIComponent(query)); | ||||
| 		this.error = undefined; | ||||
| 		this.results = []; | ||||
| 		this.messages = []; | ||||
| 		if (query.startsWith('sql:')) { | ||||
| 			this.messages = []; | ||||
| 			try { | ||||
| 				this.results = await tfrpc.rpc.query( | ||||
| 					query.substring('sql:'.length), | ||||
| 					[] | ||||
| 				); | ||||
| 			} catch (e) { | ||||
| 				this.results = []; | ||||
| 				this.error = e; | ||||
| 			} | ||||
| 		} else { | ||||
| 			let results = await tfrpc.rpc.query( | ||||
| 				` | ||||
| 					SELECT messages.id, messages.previous, messages.author, messages.sequence, messages.timestamp, messages.hash, json(messages.content) AS content, messages.signature | ||||
| @@ -55,7 +73,8 @@ class TfTabSearchElement extends LitElement { | ||||
| 				search.focus(); | ||||
| 				search.select(); | ||||
| 			} | ||||
| 		this.renderRoot.getElementById('news').messages = results; | ||||
| 			this.messages = results; | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	search_keydown(event) { | ||||
| @@ -87,6 +106,39 @@ class TfTabSearchElement extends LitElement { | ||||
| 		tfrpc.rpc.localStorageSet('drafts', JSON.stringify(this.drafts)); | ||||
| 	} | ||||
|  | ||||
| 	render_results() { | ||||
| 		if (this.error) { | ||||
| 			return html`<h2 style="color: red">${this.error.message}</h2> | ||||
| 				<pre style="color: red">${this.error.stack}</pre>`; | ||||
| 		} else if (this.messages?.length) { | ||||
| 			return html`<tf-news | ||||
| 				id="news" | ||||
| 				whoami=${this.whoami} | ||||
| 				.messages=${this.messages} | ||||
| 				.users=${this.users} | ||||
| 				.expanded=${this.expanded} | ||||
| 				.drafts=${this.drafts} | ||||
| 				@tf-expand=${this.on_expand} | ||||
| 				@tf-draft=${this.draft} | ||||
| 			></tf-news>`; | ||||
| 		} else if (this.results?.length) { | ||||
| 			let keys = Object.keys(this.results[0]).sort(); | ||||
| 			return html`<table style="width: 100%; max-width: 100%"> | ||||
| 				<tr> | ||||
| 					${keys.map((key) => html`<th>${key}</th>`)} | ||||
| 				</tr> | ||||
| 				${this.results.map( | ||||
| 					(row) => | ||||
| 						html`<tr> | ||||
| 							${keys.map((key) => html`<td>${row[key]}</td>`)} | ||||
| 						</tr>` | ||||
| 				)} | ||||
| 			</table>`; | ||||
| 		} else { | ||||
| 			return html`<div>No results.</div>`; | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	render() { | ||||
| 		if (this.query !== this.last_query) { | ||||
| 			this.last_query = this.query; | ||||
| @@ -94,11 +146,14 @@ class TfTabSearchElement extends LitElement { | ||||
| 		} | ||||
| 		let self = this; | ||||
| 		return html` | ||||
| 			<style>${generate_theme()}</style> | ||||
| 			<div class="w3-padding"> | ||||
| 				<div style="display: flex; flex-direction: row; gap: 4px"> | ||||
| 					<input type="text" class="w3-input w3-theme-d1" id="search" value=${this.query} style="flex: 1" @keydown=${this.search_keydown}></input> | ||||
| 					<button class="w3-button w3-theme-d1" @click=${(event) => self.search(self.renderRoot.getElementById('search').value)}>Search</button> | ||||
| 				</div> | ||||
| 			<tf-news id="news" whoami=${this.whoami} .messages=${this.messages} .users=${this.users} .expanded=${this.expanded} .drafts=${this.drafts} @tf-expand=${this.on_expand} @tf-draft=${this.draft}></tf-news> | ||||
| 				${this.render_results()} | ||||
| 			</div> | ||||
| 		`; | ||||
| 	} | ||||
| } | ||||
|   | ||||
| @@ -1,5 +1,5 @@ | ||||
| import {LitElement, html, unsafeHTML} from './lit-all.min.js'; | ||||
| import {styles} from './tf-styles.js'; | ||||
| import {styles, generate_theme} from './tf-styles.js'; | ||||
|  | ||||
| class TfTagElement extends LitElement { | ||||
| 	static get properties() { | ||||
| @@ -17,11 +17,15 @@ class TfTagElement extends LitElement { | ||||
|  | ||||
| 	render() { | ||||
| 		let number = this.count ? html` (${this.count})` : undefined; | ||||
| 		return html`<a | ||||
| 		return html` | ||||
| 			<style> | ||||
| 				${generate_theme()}</style | ||||
| 			><a | ||||
| 				href=${'#' + encodeURIComponent(this.tag)} | ||||
| 				class="w3-tag w3-theme-d1 w3-round-4 w3-button" | ||||
| 				>${this.tag}${number}</a | ||||
| 		> `; | ||||
| 			> | ||||
| 		`; | ||||
| 	} | ||||
| } | ||||
|  | ||||
|   | ||||
| @@ -1,6 +1,6 @@ | ||||
| import {LitElement, html} from './lit-all.min.js'; | ||||
| import * as tfrpc from '/static/tfrpc.js'; | ||||
| import {styles} from './tf-styles.js'; | ||||
| import {styles, generate_theme} from './tf-styles.js'; | ||||
|  | ||||
| class TfUserElement extends LitElement { | ||||
| 	static get properties() { | ||||
| @@ -58,7 +58,10 @@ class TfUserElement extends LitElement { | ||||
| 				/>`; | ||||
| 			} | ||||
| 		} | ||||
| 		return html` <div | ||||
| 		return html` <style> | ||||
| 				${generate_theme()} | ||||
| 			</style> | ||||
| 			<div | ||||
| 				style=${'display: inline-block; vertical-align: middle; text-wrap: nowrap; max-width: 100%; overflow: hidden; text-overflow: ellipsis' + | ||||
| 				(this.nolink ? '' : '; font-weight: bold')} | ||||
| 			> | ||||
|   | ||||
| @@ -1,5 +1,5 @@ | ||||
| { | ||||
| 	"type": "tildefriends-app", | ||||
| 	"emoji": "👋", | ||||
| 	"previous": "&ijyL/pyTwguBd9njagU7Vpc/1EyRermZuzrlq1mnzbY=.sha256" | ||||
| 	"previous": "&n1QkPkB5JoduFSx8UKOY3IlZqS2GwLiTUZv4ZrEOthQ=.sha256" | ||||
| } | ||||
|   | ||||
| @@ -340,10 +340,6 @@ | ||||
| 					<i class="fa fa-lock w3-text-purple w3-jumbo"></i> | ||||
| 					<p>libsodium</p> | ||||
| 				</a> | ||||
| 				<a href="https://github.com/openssl/openssl/releases" class="w3-col s3"> | ||||
| 					<i class="fa fa-shield-halved w3-text-green w3-jumbo"></i> | ||||
| 					<p>OpenSSL</p> | ||||
| 				</a> | ||||
| 				<a | ||||
| 					href="https://github.com/ianlancetaylor/libbacktrace" | ||||
| 					class="w3-col s3" | ||||
| @@ -351,13 +347,13 @@ | ||||
| 					<i class="fa fa-burst w3-text-pink w3-jumbo"></i> | ||||
| 					<p>libbacktrace</p> | ||||
| 				</a> | ||||
| 			</div> | ||||
|  | ||||
| 			<div class="w3-row" style="margin-top: 64px"> | ||||
| 				<a href="https://codemirror.net/docs/changelog/" class="w3-col s3"> | ||||
| 					<i class="fa fa-keyboard w3-text-indigo w3-jumbo"></i> | ||||
| 					<p>CodeMirror</p> | ||||
| 				</a> | ||||
| 			</div> | ||||
|  | ||||
| 			<div class="w3-row" style="margin-top: 64px"> | ||||
| 				<a href="https://github.com/jlfwong/speedscope/" class="w3-col s3"> | ||||
| 					<i class="fa fa-microscope w3-text-orange w3-jumbo"></i> | ||||
| 					<p>Speedscope</p> | ||||
| @@ -370,9 +366,6 @@ | ||||
| 					<i class="fa fa-book-atlas w3-text-purple w3-jumbo"></i> | ||||
| 					<p>c-ares</p> | ||||
| 				</a> | ||||
| 			</div> | ||||
|  | ||||
| 			<div class="w3-row" style="margin-top: 64px"> | ||||
| 				<a href="https://www.gnu.org/software/make/" class="w3-col s3"> | ||||
| 					<i class="fa fa-hammer w3-text-teal w3-jumbo"></i> | ||||
| 					<p>GNU Make</p> | ||||
|   | ||||
| @@ -149,7 +149,7 @@ exports.app_socket = async function socket(request, response) { | ||||
| 								parentApp: parentApp, | ||||
| 								id: blobId, | ||||
| 							}, | ||||
| 							await ssb.getIdentityInfo( | ||||
| 							await ssb_internal.getIdentityInfo( | ||||
| 								credentials?.session?.name, | ||||
| 								packageOwner, | ||||
| 								packageName | ||||
|   | ||||
							
								
								
									
										207
									
								
								core/client.js
									
									
									
									
									
								
							
							
						
						
									
										207
									
								
								core/client.js
									
									
									
									
									
								
							| @@ -9,18 +9,17 @@ | ||||
| import {LitElement, html, css, svg} from '/lit/lit-all.min.js'; | ||||
|  | ||||
| let cm6; | ||||
| let gSocket; | ||||
|  | ||||
| let gCurrentFile; | ||||
| let gFiles = {}; | ||||
| let gApp = {files: {}, emoji: '📦'}; | ||||
| let gEditor; | ||||
| let gOriginalInput; | ||||
| let gUnloading; | ||||
| let g_socket; | ||||
| let g_current_file; | ||||
| let g_files = {}; | ||||
| let g_app = {files: {}, emoji: '📦'}; | ||||
| let g_editor; | ||||
| let g_unloading; | ||||
|  | ||||
| let kErrorColor = '#dc322f'; | ||||
| let kDisconnectColor = '#f00'; | ||||
| let kStatusColor = '#fff'; | ||||
| let k_color_error = '#dc322f'; | ||||
| let k_color_disconnect = '#f00'; | ||||
| let k_color_status = '#fff'; | ||||
| /** \endcond */ | ||||
|  | ||||
| /** Functions that server-side app code can call through the app object. */ | ||||
| @@ -410,7 +409,7 @@ class TfNavigationElement extends LitElement { | ||||
| 							<link type="text/css" rel="stylesheet" href="/static/w3.css" /> | ||||
| 							<div | ||||
| 								class="w3-bar-item" | ||||
| 								style="color: ${this.status.color ?? kStatusColor}" | ||||
| 								style="color: ${this.status.color ?? k_color_status}" | ||||
| 							> | ||||
| 								${this.status.message} | ||||
| 							</div> | ||||
| @@ -429,7 +428,7 @@ class TfNavigationElement extends LitElement { | ||||
| 					<div class="w3-model w3-animate-top" style="position: absolute; left: 50%; transform: translate(-50%); z-index: 1"> | ||||
| 						<dijv class="w3-modal-content w3-card-4" style="display: block; padding: 1em"> | ||||
| 							<span id="close_error" @click=${self.clear_error} class="w3-button w3-display-topright">×</span> | ||||
| 							<div style="color: ${this.status.color ?? kErrorColor}"><b>ERROR:</b><p id="error" style="white-space: pre">${this.status.message}</p></div> | ||||
| 							<div style="color: ${this.status.color ?? k_color_error}"><b>ERROR:</b><p id="error" style="white-space: pre">${this.status.message}</p></div> | ||||
| 						</div> | ||||
| 					</div> | ||||
| 					` | ||||
| @@ -521,7 +520,7 @@ class TfFilesElement extends LitElement { | ||||
| 		for (let file of event.dataTransfer.files) { | ||||
| 			let buffer = await file.arrayBuffer(); | ||||
| 			let text = new TextDecoder('latin1').decode(buffer); | ||||
| 			gFiles[file.name] = { | ||||
| 			g_files[file.name] = { | ||||
| 				doc: cm6.EditorState.create({ | ||||
| 					doc: text, | ||||
| 					extensions: cm6.extensions, | ||||
| @@ -529,9 +528,9 @@ class TfFilesElement extends LitElement { | ||||
| 				buffer: buffer, | ||||
| 				isNew: true, | ||||
| 			}; | ||||
| 			gCurrentFile = file.name; | ||||
| 			g_current_file = file.name; | ||||
| 		} | ||||
| 		openFile(gCurrentFile); | ||||
| 		openFile(g_current_file); | ||||
| 		updateFiles(); | ||||
| 	} | ||||
|  | ||||
| @@ -895,11 +894,11 @@ async function edit() { | ||||
| 		: 'flex'; | ||||
|  | ||||
| 	try { | ||||
| 		if (!gEditor) { | ||||
| 		if (!g_editor) { | ||||
| 			cm6 = await import('/codemirror/cm6.js'); | ||||
| 			gEditor = cm6.TildeFriendsEditorView(document.getElementById('editor')); | ||||
| 			g_editor = cm6.TildeFriendsEditorView(document.getElementById('editor')); | ||||
| 		} | ||||
| 		gEditor.onDocChange = updateFiles; | ||||
| 		g_editor.onDocChange = updateFiles; | ||||
| 		await load(); | ||||
| 	} catch (error) { | ||||
| 		alert(`${error.message}\n\n${error.stack}`); | ||||
| @@ -932,13 +931,13 @@ function loadFile(name, id) { | ||||
| 			return response.text(); | ||||
| 		}) | ||||
| 		.then(function (text) { | ||||
| 			gFiles[name].doc = cm6.EditorState.create({ | ||||
| 			g_files[name].doc = cm6.EditorState.create({ | ||||
| 				doc: text, | ||||
| 				extensions: cm6.extensions, | ||||
| 			}); | ||||
| 			gFiles[name].original = gFiles[name].doc.doc.toString(); | ||||
| 			if (!Object.values(gFiles).some((x) => !x.doc)) { | ||||
| 				openFile(Object.keys(gFiles).sort()[0]); | ||||
| 			g_files[name].original = g_files[name].doc.doc.toString(); | ||||
| 			if (!Object.values(g_files).some((x) => !x.doc)) { | ||||
| 				openFile(Object.keys(g_files).sort()[0]); | ||||
| 			} | ||||
| 		}); | ||||
| } | ||||
| @@ -956,31 +955,31 @@ async function load(path) { | ||||
| 	} else if (response.status != 404) { | ||||
| 		throw new Error(response.status + ' ' + response.statusText); | ||||
| 	} | ||||
| 	gFiles = {}; | ||||
| 	g_files = {}; | ||||
| 	let isApp = false; | ||||
| 	let promises = []; | ||||
|  | ||||
| 	if (json && json['type'] == 'tildefriends-app') { | ||||
| 		isApp = true; | ||||
| 		Object.keys(json['files']).forEach(function (name) { | ||||
| 			gFiles[name] = {}; | ||||
| 			g_files[name] = {}; | ||||
| 			promises.push(loadFile(name, json['files'][name])); | ||||
| 		}); | ||||
| 		if (Object.keys(json['files']).length == 0) { | ||||
| 			document.getElementById('editPane').style.display = 'flex'; | ||||
| 		} | ||||
| 		gApp = json; | ||||
| 		gApp.emoji = gApp.emoji || '📦'; | ||||
| 		document.getElementById('icon').innerHTML = gApp.emoji; | ||||
| 		g_app = json; | ||||
| 		g_app.emoji = g_app.emoji || '📦'; | ||||
| 		document.getElementById('icon').innerHTML = g_app.emoji; | ||||
| 	} | ||||
| 	if (!isApp) { | ||||
| 		document.getElementById('editPane').style.display = 'flex'; | ||||
| 		let text = '// New script.\n'; | ||||
| 		gCurrentFile = 'app.js'; | ||||
| 		gFiles[gCurrentFile] = { | ||||
| 		g_current_file = 'app.js'; | ||||
| 		g_files[g_current_file] = { | ||||
| 			doc: cm6.EditorState.create({doc: text, extensions: cm6.extensions}), | ||||
| 		}; | ||||
| 		openFile(gCurrentFile); | ||||
| 		openFile(g_current_file); | ||||
| 	} | ||||
| 	return Promise.all(promises); | ||||
| } | ||||
| @@ -1001,13 +1000,14 @@ function closeEditor() { | ||||
|  */ | ||||
| function save(save_to) { | ||||
| 	document.getElementById('save').disabled = true; | ||||
| 	if (gCurrentFile) { | ||||
| 		gFiles[gCurrentFile].doc = gEditor.state; | ||||
| 	if (g_current_file) { | ||||
| 		g_files[g_current_file].doc = g_editor.state; | ||||
| 		if ( | ||||
| 			!gFiles[gCurrentFile].isNew && | ||||
| 			!gFiles[gCurrentFile].doc.doc.toString() == gFiles[gCurrentFile].original | ||||
| 			!g_files[g_current_file].isNew && | ||||
| 			!g_files[g_current_file].doc.doc.toString() == | ||||
| 				g_files[g_current_file].original | ||||
| 		) { | ||||
| 			delete gFiles[gCurrentFile].buffer; | ||||
| 			delete g_files[g_current_file].buffer; | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| @@ -1022,8 +1022,8 @@ function save(save_to) { | ||||
| 	} | ||||
|  | ||||
| 	let promises = []; | ||||
| 	for (let name of Object.keys(gFiles)) { | ||||
| 		let file = gFiles[name]; | ||||
| 	for (let name of Object.keys(g_files)) { | ||||
| 		let file = g_files[name]; | ||||
| 		if (!file.isNew && file.doc.doc.toString() == file.original) { | ||||
| 			continue; | ||||
| 		} | ||||
| @@ -1065,14 +1065,14 @@ function save(save_to) { | ||||
| 			let app = { | ||||
| 				type: 'tildefriends-app', | ||||
| 				files: Object.fromEntries( | ||||
| 					Object.keys(gFiles).map((x) => [x, gFiles[x].id || gApp.files[x]]) | ||||
| 					Object.keys(g_files).map((x) => [x, g_files[x].id || g_app.files[x]]) | ||||
| 				), | ||||
| 				emoji: gApp.emoji || '📦', | ||||
| 				emoji: g_app.emoji || '📦', | ||||
| 			}; | ||||
| 			Object.values(gFiles).forEach(function (file) { | ||||
| 			Object.values(g_files).forEach(function (file) { | ||||
| 				delete file.id; | ||||
| 			}); | ||||
| 			gApp = JSON.parse(JSON.stringify(app)); | ||||
| 			g_app = JSON.parse(JSON.stringify(app)); | ||||
|  | ||||
| 			return fetch(save_path + 'save', { | ||||
| 				method: 'POST', | ||||
| @@ -1087,7 +1087,7 @@ function save(save_to) { | ||||
|  | ||||
| 				if (save_path != window.location.pathname) { | ||||
| 					alert('Saved to ' + save_path + '.'); | ||||
| 				} else if (!gFiles['app.js']) { | ||||
| 				} else if (!g_files['app.js']) { | ||||
| 					window.location.reload(); | ||||
| 				} else { | ||||
| 					reconnect(save_path); | ||||
| @@ -1099,7 +1099,7 @@ function save(save_to) { | ||||
| 		}) | ||||
| 		.finally(function () { | ||||
| 			document.getElementById('save').disabled = false; | ||||
| 			Object.values(gFiles).forEach(function (file) { | ||||
| 			Object.values(g_files).forEach(function (file) { | ||||
| 				file.original = file.doc.doc.toString(); | ||||
| 			}); | ||||
| 			updateFiles(); | ||||
| @@ -1112,8 +1112,8 @@ function save(save_to) { | ||||
| function changeIcon() { | ||||
| 	let value = prompt('Enter a new app icon emoji:'); | ||||
| 	if (value !== undefined) { | ||||
| 		gApp.emoji = value || '📦'; | ||||
| 		document.getElementById('icon').innerHTML = gApp.emoji; | ||||
| 		g_app.emoji = value || '📦'; | ||||
| 		document.getElementById('icon').innerHTML = g_app.emoji; | ||||
| 	} | ||||
| } | ||||
|  | ||||
| @@ -1190,9 +1190,12 @@ function api_postMessage(message) { | ||||
| function api_error(error) { | ||||
| 	if (error) { | ||||
| 		if (typeof error == 'string') { | ||||
| 			setStatusMessage('⚠️ ' + error, kErrorColor); | ||||
| 			setStatusMessage('⚠️ ' + error, k_color_error); | ||||
| 		} else { | ||||
| 			setStatusMessage('⚠️ ' + error.message + '\n' + error.stack, kErrorColor); | ||||
| 			setStatusMessage( | ||||
| 				'⚠️ ' + error.message + '\n' + error.stack, | ||||
| 				k_color_error | ||||
| 			); | ||||
| 		} | ||||
| 	} | ||||
| 	console.log('error', error); | ||||
| @@ -1309,7 +1312,7 @@ function api_setHash(hash) { | ||||
|  */ | ||||
| function _receive_websocket_message(message) { | ||||
| 	if (message && message.action == 'session') { | ||||
| 		setStatusMessage('🟢 Executing...', kStatusColor); | ||||
| 		setStatusMessage('🟢 Executing...', k_color_status); | ||||
| 		let navigation = document.getElementsByTagName('tf-navigation')[0]; | ||||
| 		navigation.credentials = message.credentials; | ||||
| 		navigation.identities = message.identities; | ||||
| @@ -1409,7 +1412,7 @@ function setStatusMessage(message, color) { | ||||
| 	document.getElementsByTagName('tf-navigation')[0].status = { | ||||
| 		message: message, | ||||
| 		color: color, | ||||
| 		is_error: color == kErrorColor, | ||||
| 		is_error: color == k_color_error, | ||||
| 	}; | ||||
| } | ||||
|  | ||||
| @@ -1419,11 +1422,11 @@ function setStatusMessage(message, color) { | ||||
|  */ | ||||
| function send(value) { | ||||
| 	try { | ||||
| 		if (gSocket && gSocket.readyState == gSocket.OPEN) { | ||||
| 			gSocket.send(JSON.stringify(value)); | ||||
| 		if (g_socket && g_socket.readyState == g_socket.OPEN) { | ||||
| 			g_socket.send(JSON.stringify(value)); | ||||
| 		} | ||||
| 	} catch (error) { | ||||
| 		setStatusMessage('🤷 Send failed: ' + error.toString(), kErrorColor); | ||||
| 		setStatusMessage('🤷 Send failed: ' + error.toString(), k_color_error); | ||||
| 	} | ||||
| } | ||||
|  | ||||
| @@ -1438,7 +1441,7 @@ function hashChange() { | ||||
|  * Make sure the app is connected on window focus, and notify the app. | ||||
|  */ | ||||
| function focus() { | ||||
| 	if (gSocket && gSocket.readyState == gSocket.CLOSED) { | ||||
| 	if (g_socket && g_socket.readyState == g_socket.CLOSED) { | ||||
| 		connectSocket(); | ||||
| 	} else { | ||||
| 		send({event: 'focus'}); | ||||
| @@ -1449,10 +1452,8 @@ function focus() { | ||||
|  * Notify the app of lost focus. | ||||
|  */ | ||||
| function blur() { | ||||
| 	if (gSocket && gSocket.readyState == gSocket.OPEN) { | ||||
| 	send({event: 'blur'}); | ||||
| } | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * Handle a message. | ||||
| @@ -1508,8 +1509,8 @@ function message(event) { | ||||
|  * @param path The path to which the WebSocket should be connected. | ||||
|  */ | ||||
| function reconnect(path) { | ||||
| 	let oldSocket = gSocket; | ||||
| 	gSocket = null; | ||||
| 	let oldSocket = g_socket; | ||||
| 	g_socket = null; | ||||
| 	if (oldSocket) { | ||||
| 		oldSocket.onopen = null; | ||||
| 		oldSocket.onclose = null; | ||||
| @@ -1524,24 +1525,24 @@ function reconnect(path) { | ||||
|  * @param path The path to which to connect. | ||||
|  */ | ||||
| function connectSocket(path) { | ||||
| 	if (!gSocket || gSocket.readyState != gSocket.OPEN) { | ||||
| 		if (gSocket) { | ||||
| 			gSocket.onopen = null; | ||||
| 			gSocket.onclose = null; | ||||
| 			gSocket.onmessage = null; | ||||
| 			gSocket.close(); | ||||
| 	if (!g_socket || g_socket.readyState != g_socket.OPEN) { | ||||
| 		if (g_socket) { | ||||
| 			g_socket.onopen = null; | ||||
| 			g_socket.onclose = null; | ||||
| 			g_socket.onmessage = null; | ||||
| 			g_socket.close(); | ||||
| 		} | ||||
| 		setStatusMessage('⚪ Connecting...', kStatusColor); | ||||
| 		gSocket = new WebSocket( | ||||
| 		setStatusMessage('⚪ Connecting...', k_color_status); | ||||
| 		g_socket = new WebSocket( | ||||
| 			(window.location.protocol == 'https:' ? 'wss://' : 'ws://') + | ||||
| 				window.location.hostname + | ||||
| 				(window.location.port.length ? ':' + window.location.port : '') + | ||||
| 				'/app/socket' | ||||
| 		); | ||||
| 		gSocket.onopen = function () { | ||||
| 			setStatusMessage('🟡 Authenticating...', kStatusColor); | ||||
| 		g_socket.onopen = function () { | ||||
| 			setStatusMessage('🟡 Authenticating...', k_color_status); | ||||
| 			let connect_path = path ?? window.location.pathname; | ||||
| 			gSocket.send( | ||||
| 			g_socket.send( | ||||
| 				JSON.stringify({ | ||||
| 					action: 'hello', | ||||
| 					path: connect_path, | ||||
| @@ -1553,12 +1554,12 @@ function connectSocket(path) { | ||||
| 				}) | ||||
| 			); | ||||
| 		}; | ||||
| 		gSocket.onmessage = function (event) { | ||||
| 		g_socket.onmessage = function (event) { | ||||
| 			_receive_websocket_message(JSON.parse(event.data)); | ||||
| 		}; | ||||
| 		gSocket.onclose = function (event) { | ||||
| 			if (gUnloading) { | ||||
| 				setStatusMessage('⚪ Closing...', kStatusColor); | ||||
| 		g_socket.onclose = function (event) { | ||||
| 			if (g_unloading) { | ||||
| 				setStatusMessage('⚪ Closing...', k_color_status); | ||||
| 			} else { | ||||
| 				const k_codes = { | ||||
| 					1000: 'Normal closure', | ||||
| @@ -1579,7 +1580,7 @@ function connectSocket(path) { | ||||
| 				}; | ||||
| 				setStatusMessage( | ||||
| 					'🔴 Closed: ' + (k_codes[event.code] || event.code), | ||||
| 					kDisconnectColor | ||||
| 					k_color_disconnect | ||||
| 				); | ||||
| 			} | ||||
| 		}; | ||||
| @@ -1592,24 +1593,24 @@ function connectSocket(path) { | ||||
|  */ | ||||
| function openFile(name) { | ||||
| 	let newDoc = | ||||
| 		name && gFiles[name] | ||||
| 			? gFiles[name].doc | ||||
| 		name && g_files[name] | ||||
| 			? g_files[name].doc | ||||
| 			: cm6.EditorState.create({doc: '', extensions: cm6.extensions}); | ||||
| 	let oldDoc = gEditor.state; | ||||
| 	gEditor.setState(newDoc); | ||||
| 	let oldDoc = g_editor.state; | ||||
| 	g_editor.setState(newDoc); | ||||
|  | ||||
| 	if (gFiles[gCurrentFile]) { | ||||
| 		gFiles[gCurrentFile].doc = oldDoc; | ||||
| 	if (g_files[g_current_file]) { | ||||
| 		g_files[g_current_file].doc = oldDoc; | ||||
| 		if ( | ||||
| 			!gFiles[gCurrentFile].isNew && | ||||
| 			gFiles[gCurrentFile].doc.doc.toString() == oldDoc.doc.toString() | ||||
| 			!g_files[g_current_file].isNew && | ||||
| 			g_files[g_current_file].doc.doc.toString() == oldDoc.doc.toString() | ||||
| 		) { | ||||
| 			delete gFiles[gCurrentFile].buffer; | ||||
| 			delete g_files[g_current_file].buffer; | ||||
| 		} | ||||
| 	} | ||||
| 	gCurrentFile = name; | ||||
| 	g_current_file = name; | ||||
| 	updateFiles(); | ||||
| 	gEditor.focus(); | ||||
| 	g_editor.focus(); | ||||
| } | ||||
|  | ||||
| /** | ||||
| @@ -1619,19 +1620,19 @@ function updateFiles() { | ||||
| 	let files = document.getElementsByTagName('tf-files-pane')[0]; | ||||
| 	if (files) { | ||||
| 		files.files = Object.fromEntries( | ||||
| 			Object.keys(gFiles).map((file) => [ | ||||
| 			Object.keys(g_files).map((file) => [ | ||||
| 				file, | ||||
| 				{ | ||||
| 					clean: | ||||
| 						(file == gCurrentFile | ||||
| 							? gEditor.state.doc.toString() | ||||
| 							: gFiles[file].doc.doc.toString()) == gFiles[file].original, | ||||
| 						(file == g_current_file | ||||
| 							? g_editor.state.doc.toString() | ||||
| 							: g_files[file].doc.doc.toString()) == g_files[file].original, | ||||
| 				}, | ||||
| 			]) | ||||
| 		); | ||||
| 		files.current = gCurrentFile; | ||||
| 		files.current = g_current_file; | ||||
| 	} | ||||
| 	gEditor.focus(); | ||||
| 	g_editor.focus(); | ||||
| } | ||||
|  | ||||
| /** | ||||
| @@ -1639,7 +1640,7 @@ function updateFiles() { | ||||
|  * @param name The file's name. | ||||
|  */ | ||||
| function makeNewFile(name) { | ||||
| 	gFiles[name] = { | ||||
| 	g_files[name] = { | ||||
| 		doc: cm6.EditorState.create({extensions: cm6.extensions}), | ||||
| 	}; | ||||
| 	openFile(name); | ||||
| @@ -1650,7 +1651,7 @@ function makeNewFile(name) { | ||||
|  */ | ||||
| function newFile() { | ||||
| 	let name = prompt('Name of new file:', 'file.js'); | ||||
| 	if (name && !gFiles[name]) { | ||||
| 	if (name && !g_files[name]) { | ||||
| 		makeNewFile(name); | ||||
| 	} | ||||
| } | ||||
| @@ -1659,9 +1660,9 @@ function newFile() { | ||||
|  * Prompt to remove a file. | ||||
|  */ | ||||
| function removeFile() { | ||||
| 	if (confirm('Remove ' + gCurrentFile + '?')) { | ||||
| 		delete gFiles[gCurrentFile]; | ||||
| 		openFile(Object.keys(gFiles)[0]); | ||||
| 	if (confirm('Remove ' + g_current_file + '?')) { | ||||
| 		delete g_files[g_current_file]; | ||||
| 		openFile(Object.keys(g_files)[0]); | ||||
| 	} | ||||
| } | ||||
|  | ||||
| @@ -1677,13 +1678,13 @@ async function appExport() { | ||||
| 		`${name}.json`, | ||||
| 		JSON.stringify({ | ||||
| 			type: 'tildefriends-app', | ||||
| 			emoji: gApp.emoji || '📦', | ||||
| 			emoji: g_app.emoji || '📦', | ||||
| 		}) | ||||
| 	); | ||||
| 	for (let file of Object.keys(gFiles)) { | ||||
| 	for (let file of Object.keys(g_files)) { | ||||
| 		zip.file( | ||||
| 			`${name}/${file}`, | ||||
| 			gFiles[file].buffer ?? gFiles[file].doc.doc.toString() | ||||
| 			g_files[file].buffer ?? g_files[file].doc.doc.toString() | ||||
| 		); | ||||
| 	} | ||||
| 	let content = await zip.generateAsync({ | ||||
| @@ -1802,9 +1803,9 @@ async function sourcePretty() { | ||||
| 	let babel = (await import('/prettier/babel.mjs')).default; | ||||
| 	let estree = (await import('/prettier/estree.mjs')).default; | ||||
| 	let prettier_html = (await import('/prettier/html.mjs')).default; | ||||
| 	let source = gEditor.state.doc.toString(); | ||||
| 	let source = g_editor.state.doc.toString(); | ||||
| 	let formatted = await prettier.format(source, { | ||||
| 		parser: gCurrentFile?.toLowerCase()?.endsWith('.html') ? 'html' : 'babel', | ||||
| 		parser: g_current_file?.toLowerCase()?.endsWith('.html') ? 'html' : 'babel', | ||||
| 		plugins: [babel, estree, prettier_html], | ||||
| 		trailingComma: 'es5', | ||||
| 		useTabs: true, | ||||
| @@ -1813,10 +1814,10 @@ async function sourcePretty() { | ||||
| 		bracketSpacing: false, | ||||
| 	}); | ||||
| 	if (source !== formatted) { | ||||
| 		gEditor.dispatch({ | ||||
| 		g_editor.dispatch({ | ||||
| 			changes: { | ||||
| 				from: 0, | ||||
| 				to: gEditor.state.doc.length, | ||||
| 				to: g_editor.state.doc.length, | ||||
| 				insert: formatted, | ||||
| 			}, | ||||
| 		}); | ||||
| @@ -1861,7 +1862,7 @@ window.addEventListener('load', function () { | ||||
| 	window.addEventListener('message', message, false); | ||||
| 	window.addEventListener('online', connectSocket); | ||||
| 	window.addEventListener('beforeunload', function () { | ||||
| 		gUnloading = true; | ||||
| 		g_unloading = true; | ||||
| 	}); | ||||
| 	document.getElementById('name').value = window.location.pathname; | ||||
| 	document | ||||
|   | ||||
							
								
								
									
										89
									
								
								core/core.js
									
									
									
									
									
								
							
							
						
						
									
										89
									
								
								core/core.js
									
									
									
									
									
								
							| @@ -7,7 +7,6 @@ | ||||
|  | ||||
| /** \cond */ | ||||
| import * as app from './app.js'; | ||||
| import * as http from './http.js'; | ||||
|  | ||||
| export {invoke, getProcessBlob}; | ||||
| /** \endcond */ | ||||
| @@ -179,6 +178,7 @@ async function getProcessBlob(blobId, key, options) { | ||||
| 			process.task = new Task(); | ||||
| 			process.packageOwner = options.packageOwner; | ||||
| 			process.packageName = options.packageName; | ||||
| 			process.url = options?.url; | ||||
| 			process.eventHandlers = {}; | ||||
| 			if (!options?.script || options?.script === 'app.js') { | ||||
| 				process.app = new app.App(); | ||||
| @@ -192,7 +192,6 @@ async function getProcessBlob(blobId, key, options) { | ||||
| 			}); | ||||
| 			gProcesses[key] = process; | ||||
| 			process.task.onExit = function (exitCode, terminationSignal) { | ||||
| 				broadcastEvent('onSessionEnd', [getUser(process, process)]); | ||||
| 				process.task = null; | ||||
| 				delete gProcesses[key]; | ||||
| 			}; | ||||
| @@ -200,29 +199,6 @@ async function getProcessBlob(blobId, key, options) { | ||||
| 				core: { | ||||
| 					broadcast: broadcast.bind(process), | ||||
| 					user: getUser(process, process), | ||||
| 					users: async function () { | ||||
| 						try { | ||||
| 							return JSON.parse(await new Database('auth').get('users')); | ||||
| 						} catch { | ||||
| 							return []; | ||||
| 						} | ||||
| 					}, | ||||
| 					permissionsGranted: async function () { | ||||
| 						let user = process?.credentials?.session?.name; | ||||
| 						let settings = await loadSettings(); | ||||
| 						if ( | ||||
| 							user && | ||||
| 							options?.packageOwner && | ||||
| 							options?.packageName && | ||||
| 							settings.userPermissions && | ||||
| 							settings.userPermissions[user] && | ||||
| 							settings.userPermissions[user][options.packageOwner] | ||||
| 						) { | ||||
| 							return settings.userPermissions[user][options.packageOwner][ | ||||
| 								options.packageName | ||||
| 							]; | ||||
| 						} | ||||
| 					}, | ||||
| 					allPermissionsGranted: async function () { | ||||
| 						let user = process?.credentials?.session?.name; | ||||
| 						let settings = await loadSettings(); | ||||
| @@ -236,11 +212,6 @@ async function getProcessBlob(blobId, key, options) { | ||||
| 							return settings.userPermissions[user]; | ||||
| 						} | ||||
| 					}, | ||||
| 					permissionsForUser: async function (user) { | ||||
| 						let settings = await loadSettings(); | ||||
| 						return settings?.permissions?.[user] ?? []; | ||||
| 					}, | ||||
| 					getSockets: getSockets, | ||||
| 					permissionTest: async function (permission) { | ||||
| 						let user = process?.credentials?.session?.name; | ||||
| 						let settings = await loadSettings(); | ||||
| @@ -301,11 +272,6 @@ async function getProcessBlob(blobId, key, options) { | ||||
| 							throw Error(`Permission denied: ${permission}.`); | ||||
| 						} | ||||
| 					}, | ||||
| 					app: { | ||||
| 						owner: options?.packageOwner, | ||||
| 						name: options?.packageName, | ||||
| 					}, | ||||
| 					url: options?.url, | ||||
| 				}, | ||||
| 			}; | ||||
| 			process.sendIdentities = async function () { | ||||
| @@ -314,7 +280,7 @@ async function getProcessBlob(blobId, key, options) { | ||||
| 						{ | ||||
| 							action: 'identities', | ||||
| 						}, | ||||
| 						await ssb.getIdentityInfo( | ||||
| 						await ssb_internal.getIdentityInfo( | ||||
| 							process?.credentials?.session?.name, | ||||
| 							options?.packageOwner, | ||||
| 							options?.packageName | ||||
| @@ -357,7 +323,7 @@ async function getProcessBlob(blobId, key, options) { | ||||
| 						options.packageName, | ||||
| 						'setActiveIdentity', | ||||
| 						[ | ||||
| 							await ssb.getActiveIdentity( | ||||
| 							await imports.ssb.getActiveIdentity( | ||||
| 								process.credentials?.session?.name, | ||||
| 								options.packageOwner, | ||||
| 								options.packageName | ||||
| @@ -370,19 +336,6 @@ async function getProcessBlob(blobId, key, options) { | ||||
| 				} | ||||
| 			}; | ||||
| 			if (process.credentials?.permissions?.administration) { | ||||
| 				imports.core.globalSettingsDescriptions = async function () { | ||||
| 					let settings = Object.assign({}, defaultGlobalSettings()); | ||||
| 					for (let [key, value] of Object.entries(await loadSettings())) { | ||||
| 						if (settings[key]) { | ||||
| 							settings[key].value = value; | ||||
| 						} | ||||
| 					} | ||||
| 					return settings; | ||||
| 				}; | ||||
| 				imports.core.globalSettingsGet = async function (key) { | ||||
| 					let settings = await loadSettings(); | ||||
| 					return settings?.[key]; | ||||
| 				}; | ||||
| 				imports.core.globalSettingsSet = async function (key, value) { | ||||
| 					await imports.core.permissionTest('set_global_setting'); | ||||
| 					print('Setting', key, value); | ||||
| @@ -464,26 +417,6 @@ async function getProcessBlob(blobId, key, options) { | ||||
| 				} | ||||
| 			}; | ||||
| 			imports.ssb.setActiveIdentity = (id) => process.setActiveIdentity(id); | ||||
| 			imports.ssb.getActiveIdentity = () => | ||||
| 				ssb.getActiveIdentity( | ||||
| 					process.credentials?.session?.name, | ||||
| 					options.packageOwner, | ||||
| 					options.packageName | ||||
| 				); | ||||
| 			imports.ssb.getOwnerIdentities = function () { | ||||
| 				if (options.packageOwner) { | ||||
| 					return ssb.getIdentities(options.packageOwner); | ||||
| 				} | ||||
| 			}; | ||||
| 			imports.ssb.getIdentities = function () { | ||||
| 				if ( | ||||
| 					process.credentials && | ||||
| 					process.credentials.session && | ||||
| 					process.credentials.session.name | ||||
| 				) { | ||||
| 					return ssb.getIdentities(process.credentials.session.name); | ||||
| 				} | ||||
| 			}; | ||||
| 			imports.ssb.getPrivateKey = function (id) { | ||||
| 				if ( | ||||
| 					process.credentials && | ||||
| @@ -553,13 +486,6 @@ async function getProcessBlob(blobId, key, options) { | ||||
| 					); | ||||
| 				} | ||||
| 			}; | ||||
| 			imports.ssb.addEventListener = undefined; | ||||
| 			imports.ssb.removeEventListener = undefined; | ||||
| 			imports.ssb.getIdentityInfo = undefined; | ||||
| 			imports.fetch = async function (url, options) { | ||||
| 				let settings = await loadSettings(); | ||||
| 				return http.fetch(url, options, settings?.fetch_hosts); | ||||
| 			}; | ||||
|  | ||||
| 			if ( | ||||
| 				process.credentials && | ||||
| @@ -663,7 +589,6 @@ async function getProcessBlob(blobId, key, options) { | ||||
| 			} catch (e) { | ||||
| 				printError(e); | ||||
| 			} | ||||
| 			broadcastEvent('onSessionBegin', [getUser(process, process)]); | ||||
| 			if (process.app) { | ||||
| 				process.app.send({action: 'ready', version: version()}); | ||||
| 				await process.sendPermissions(); | ||||
| @@ -689,19 +614,19 @@ async function getProcessBlob(blobId, key, options) { | ||||
| /** | ||||
|  * SSB message added callback. | ||||
|  */ | ||||
| ssb.addEventListener('message', function () { | ||||
| ssb_internal.addEventListener('message', function () { | ||||
| 	broadcastEvent('onMessage', [...arguments]); | ||||
| }); | ||||
|  | ||||
| ssb.addEventListener('blob', function () { | ||||
| ssb_internal.addEventListener('blob', function () { | ||||
| 	broadcastEvent('onBlob', [...arguments]); | ||||
| }); | ||||
|  | ||||
| ssb.addEventListener('broadcasts', function () { | ||||
| ssb_internal.addEventListener('broadcasts', function () { | ||||
| 	broadcastEvent('onBroadcastsChanged', []); | ||||
| }); | ||||
|  | ||||
| ssb.addEventListener('connections', function () { | ||||
| ssb_internal.addEventListener('connections', function () { | ||||
| 	broadcastEvent('onConnectionsChanged', []); | ||||
| }); | ||||
|  | ||||
|   | ||||
							
								
								
									
										121
									
								
								core/http.js
									
									
									
									
									
								
							
							
						
						
									
										121
									
								
								core/http.js
									
									
									
									
									
								
							| @@ -1,121 +0,0 @@ | ||||
| /** | ||||
|  * \file | ||||
|  * \defgroup tfhttp Tilde Friends HTTP Client JS | ||||
|  * Tilde Friends server-side HTTP client. | ||||
|  * @{ | ||||
|  */ | ||||
|  | ||||
| /** | ||||
|  * Parse a URL into protocol, host, path, and port parts. | ||||
|  * @param url | ||||
|  * @return An object of the URL parts. | ||||
|  */ | ||||
| function parseUrl(url) { | ||||
| 	// XXX: Hack. | ||||
| 	let match = url.match(new RegExp('(\\w+)://([^/:]+)(?::(\\d+))?(.*)')); | ||||
| 	return { | ||||
| 		protocol: match[1], | ||||
| 		host: match[2], | ||||
| 		path: match[4], | ||||
| 		port: match[3] ? parseInt(match[3]) : match[1] == 'http' ? 80 : 443, | ||||
| 	}; | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * Parse an HTTP response into headers and body content. | ||||
|  * @param data The response data, headers and body included. | ||||
|  * @return headers and body data. | ||||
|  */ | ||||
| function parseResponse(data) { | ||||
| 	let firstLine; | ||||
| 	let headers = {}; | ||||
| 	while (true) { | ||||
| 		let endLine = data.indexOf('\r\n'); | ||||
| 		let line = data.substring(0, endLine); | ||||
| 		data = data.substring(endLine + 2); | ||||
| 		if (!line.length) { | ||||
| 			break; | ||||
| 		} else if (!firstLine) { | ||||
| 			firstLine = line; | ||||
| 		} else { | ||||
| 			let colon = line.indexOf(':'); | ||||
| 			headers[line.substring(colon)] = line.substring(colon + 1); | ||||
| 		} | ||||
| 	} | ||||
| 	return {headers: headers, body: data}; | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * Make an HTTP request. | ||||
|  * @param url The URL. | ||||
|  * @param options Request options. | ||||
|  * @param allowed_hosts List of allowed hosts. | ||||
|  * @return A promise resolved with the response headers and body. | ||||
|  */ | ||||
| export function fetch(url, options, allowed_hosts) { | ||||
| 	let parsed = parseUrl(url); | ||||
| 	return new Promise(function (resolve, reject) { | ||||
| 		if ((allowed_hosts ?? []).indexOf(parsed.host) == -1) { | ||||
| 			throw new Error(`fetch() request to host ${parsed.host} is not allowed.`); | ||||
| 		} | ||||
| 		let socket = new Socket(); | ||||
| 		let buffer = new Uint8Array(0); | ||||
|  | ||||
| 		return socket | ||||
| 			.connect(parsed.host, parsed.port) | ||||
| 			.then(function () { | ||||
| 				socket.read(function (data) { | ||||
| 					if (data && data.length) { | ||||
| 						let newBuffer = new Uint8Array(buffer.length + data.length); | ||||
| 						newBuffer.set(buffer, 0); | ||||
| 						newBuffer.set(data, buffer.length); | ||||
| 						buffer = newBuffer; | ||||
| 					} else { | ||||
| 						let result = parseHttpResponse(buffer); | ||||
| 						if (!result) { | ||||
| 							reject(new Exception('Parse failed.')); | ||||
| 						} | ||||
| 						if (typeof result == 'number') { | ||||
| 							if (result == -2) { | ||||
| 								reject('Incomplete request.'); | ||||
| 							} else { | ||||
| 								reject('Bad request.'); | ||||
| 							} | ||||
| 						} else if (typeof result == 'object') { | ||||
| 							resolve({ | ||||
| 								body: buffer.slice(result.bytes_parsed), | ||||
| 								status: result.status, | ||||
| 								message: result.message, | ||||
| 								headers: result.headers, | ||||
| 							}); | ||||
| 						} else { | ||||
| 							reject(new Exception('Unexpected parse result.')); | ||||
| 						} | ||||
| 						resolve(parseResponse(utf8Decode(buffer))); | ||||
| 					} | ||||
| 				}); | ||||
|  | ||||
| 				if (parsed.port == 443) { | ||||
| 					return socket.startTls(); | ||||
| 				} | ||||
| 			}) | ||||
| 			.then(function () { | ||||
| 				let body = | ||||
| 					typeof options?.body == 'string' | ||||
| 						? utf8Encode(options.body) | ||||
| 						: options.body || new Uint8Array(0); | ||||
| 				let headers = utf8Encode( | ||||
| 					`${options?.method ?? 'GET'} ${parsed.path} HTTP/1.0\r\nHost: ${parsed.host}\r\nConnection: close\r\nContent-Length: ${body.length}\r\n\r\n` | ||||
| 				); | ||||
| 				let fullRequest = new Uint8Array(headers.length + body.length); | ||||
| 				fullRequest.set(headers, 0); | ||||
| 				fullRequest.set(body, headers.length); | ||||
| 				socket.write(fullRequest); | ||||
| 			}) | ||||
| 			.catch(function (error) { | ||||
| 				reject(error); | ||||
| 			}); | ||||
| 	}); | ||||
| } | ||||
|  | ||||
| /** @} */ | ||||
							
								
								
									
										2
									
								
								deps/codemirror/cm6.js
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								deps/codemirror/cm6.js
									
									
									
									
										vendored
									
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
							
								
								
									
										248
									
								
								deps/codemirror_src/package-lock.json
									
									
									
										generated
									
									
										vendored
									
									
								
							
							
						
						
									
										248
									
								
								deps/codemirror_src/package-lock.json
									
									
									
										generated
									
									
										vendored
									
									
								
							| @@ -19,9 +19,9 @@ | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/@codemirror/autocomplete": { | ||||
|       "version": "6.18.7", | ||||
|       "resolved": "https://registry.npmjs.org/@codemirror/autocomplete/-/autocomplete-6.18.7.tgz", | ||||
|       "integrity": "sha512-8EzdeIoWPJDsMBwz3zdzwXnUpCzMiCyz5/A3FIPpriaclFCGDkAzK13sMcnsu5rowqiyeQN2Vs2TsOcoDPZirQ==", | ||||
|       "version": "6.19.1", | ||||
|       "resolved": "https://registry.npmjs.org/@codemirror/autocomplete/-/autocomplete-6.19.1.tgz", | ||||
|       "integrity": "sha512-q6NenYkEy2fn9+JyjIxMWcNjzTL/IhwqfzOut1/G3PrIFkrbl4AL7Wkse5tLrQUUyqGoAKU5+Pi5jnnXxH5HGw==", | ||||
|       "license": "MIT", | ||||
|       "dependencies": { | ||||
|         "@codemirror/language": "^6.0.0", | ||||
| @@ -31,9 +31,9 @@ | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/@codemirror/commands": { | ||||
|       "version": "6.8.1", | ||||
|       "resolved": "https://registry.npmjs.org/@codemirror/commands/-/commands-6.8.1.tgz", | ||||
|       "integrity": "sha512-KlGVYufHMQzxbdQONiLyGQDUW0itrLZwq3CcY7xpv9ZLRHqzkBSoteocBHtMCoY7/Ci4xhzSrToIeLg7FxHuaw==", | ||||
|       "version": "6.10.0", | ||||
|       "resolved": "https://registry.npmjs.org/@codemirror/commands/-/commands-6.10.0.tgz", | ||||
|       "integrity": "sha512-2xUIc5mHXQzT16JnyOFkh8PvfeXuIut3pslWGfsGOhxP/lpgRm9HOl/mpzLErgt5mXDovqA0d11P21gofRLb9w==", | ||||
|       "license": "MIT", | ||||
|       "dependencies": { | ||||
|         "@codemirror/language": "^6.0.0", | ||||
| @@ -56,9 +56,9 @@ | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/@codemirror/lang-html": { | ||||
|       "version": "6.4.10", | ||||
|       "resolved": "https://registry.npmjs.org/@codemirror/lang-html/-/lang-html-6.4.10.tgz", | ||||
|       "integrity": "sha512-h/SceTVsN5r+WE+TVP2g3KDvNoSzbSrtZXCKo4vkKdbfT5t4otuVgngGdFukOO/rwRD2++pCxoh6xD4TEVMkQA==", | ||||
|       "version": "6.4.11", | ||||
|       "resolved": "https://registry.npmjs.org/@codemirror/lang-html/-/lang-html-6.4.11.tgz", | ||||
|       "integrity": "sha512-9NsXp7Nwp891pQchI7gPdTwBuSuT3K65NGTHWHNJ55HjYcHLllr0rbIZNdOzas9ztc1EUVBlHou85FFZS4BNnw==", | ||||
|       "license": "MIT", | ||||
|       "dependencies": { | ||||
|         "@codemirror/autocomplete": "^6.0.0", | ||||
| @@ -69,7 +69,7 @@ | ||||
|         "@codemirror/view": "^6.17.0", | ||||
|         "@lezer/common": "^1.0.0", | ||||
|         "@lezer/css": "^1.1.0", | ||||
|         "@lezer/html": "^1.3.0" | ||||
|         "@lezer/html": "^1.3.12" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/@codemirror/lang-javascript": { | ||||
| @@ -112,9 +112,9 @@ | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/@codemirror/lint": { | ||||
|       "version": "6.8.5", | ||||
|       "resolved": "https://registry.npmjs.org/@codemirror/lint/-/lint-6.8.5.tgz", | ||||
|       "integrity": "sha512-s3n3KisH7dx3vsoeGMxsbRAgKe4O1vbrnKBClm99PU0fWxmxsx5rR2PfqQgIt+2MMJBHbiJ5rfIdLYfB9NNvsA==", | ||||
|       "version": "6.9.1", | ||||
|       "resolved": "https://registry.npmjs.org/@codemirror/lint/-/lint-6.9.1.tgz", | ||||
|       "integrity": "sha512-te7To1EQHePBQQzasDKWmK2xKINIXpk+xAiSYr9ZN+VB4KaT+/Hi2PEkeErTk5BV3PTz1TLyQL4MtJfPkKZ9sw==", | ||||
|       "license": "MIT", | ||||
|       "dependencies": { | ||||
|         "@codemirror/state": "^6.0.0", | ||||
| @@ -155,9 +155,9 @@ | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/@codemirror/view": { | ||||
|       "version": "6.38.3", | ||||
|       "resolved": "https://registry.npmjs.org/@codemirror/view/-/view-6.38.3.tgz", | ||||
|       "integrity": "sha512-x2t87+oqwB1mduiQZ6huIghjMt4uZKFEdj66IcXw7+a5iBEvv9lh7EWDRHI7crnD4BMGpnyq/RzmCGbiEZLcvQ==", | ||||
|       "version": "6.38.6", | ||||
|       "resolved": "https://registry.npmjs.org/@codemirror/view/-/view-6.38.6.tgz", | ||||
|       "integrity": "sha512-qiS0z1bKs5WOvHIAC0Cybmv4AJSkAXgX5aD6Mqd2epSLlVJsQl8NG23jCVouIgkh4All/mrbdsf2UOLFnJw0tw==", | ||||
|       "license": "MIT", | ||||
|       "dependencies": { | ||||
|         "@codemirror/state": "^6.5.0", | ||||
| @@ -217,9 +217,9 @@ | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/@lezer/common": { | ||||
|       "version": "1.2.3", | ||||
|       "resolved": "https://registry.npmjs.org/@lezer/common/-/common-1.2.3.tgz", | ||||
|       "integrity": "sha512-w7ojc8ejBqr2REPsWxJjrMFsA/ysDCFICn8zEOR9mrqzOu2amhITYuLD8ag6XZf0CFXDrhKqw7+tW8cX66NaDA==", | ||||
|       "version": "1.3.0", | ||||
|       "resolved": "https://registry.npmjs.org/@lezer/common/-/common-1.3.0.tgz", | ||||
|       "integrity": "sha512-L9X8uHCYU310o99L3/MpJKYxPzXPOS7S0NmBaM7UO/x2Kb2WbmMLSkfvdr1KxRIFYOpbY0Jhn7CfLSUDzL8arQ==", | ||||
|       "license": "MIT" | ||||
|     }, | ||||
|     "node_modules/@lezer/css": { | ||||
| @@ -234,18 +234,18 @@ | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/@lezer/highlight": { | ||||
|       "version": "1.2.1", | ||||
|       "resolved": "https://registry.npmjs.org/@lezer/highlight/-/highlight-1.2.1.tgz", | ||||
|       "integrity": "sha512-Z5duk4RN/3zuVO7Jq0pGLJ3qynpxUVsh7IbUbGj88+uV2ApSAn6kWg2au3iJb+0Zi7kKtqffIESgNcRXWZWmSA==", | ||||
|       "version": "1.2.2", | ||||
|       "resolved": "https://registry.npmjs.org/@lezer/highlight/-/highlight-1.2.2.tgz", | ||||
|       "integrity": "sha512-z8TQwaBXXQIvG6i2g3e9cgMwUUXu9Ib7jo2qRRggdhwKpM56Dw3PM3wmexn+EGaaOZ7az0K7sjc3/gcGW7sz7A==", | ||||
|       "license": "MIT", | ||||
|       "dependencies": { | ||||
|         "@lezer/common": "^1.0.0" | ||||
|         "@lezer/common": "^1.3.0" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/@lezer/html": { | ||||
|       "version": "1.3.10", | ||||
|       "resolved": "https://registry.npmjs.org/@lezer/html/-/html-1.3.10.tgz", | ||||
|       "integrity": "sha512-dqpT8nISx/p9Do3AchvYGV3qYc4/rKr3IBZxlHmpIKam56P47RSHkSF5f13Vu9hebS1jM0HmtJIwLbWz1VIY6w==", | ||||
|       "version": "1.3.12", | ||||
|       "resolved": "https://registry.npmjs.org/@lezer/html/-/html-1.3.12.tgz", | ||||
|       "integrity": "sha512-RJ7eRWdaJe3bsiiLLHjCFT1JMk8m1YP9kaUbvu2rMLEoOnke9mcTVDyfOslsln0LtujdWespjJ39w6zo+RsQYw==", | ||||
|       "license": "MIT", | ||||
|       "dependencies": { | ||||
|         "@lezer/common": "^1.2.0", | ||||
| @@ -360,9 +360,9 @@ | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/@rollup/rollup-android-arm-eabi": { | ||||
|       "version": "4.52.2", | ||||
|       "resolved": "https://registry.npmjs.org/@rollup/rollup-android-arm-eabi/-/rollup-android-arm-eabi-4.52.2.tgz", | ||||
|       "integrity": "sha512-o3pcKzJgSGt4d74lSZ+OCnHwkKBeAbFDmbEm5gg70eA8VkyCuC/zV9TwBnmw6VjDlRdF4Pshfb+WE9E6XY1PoQ==", | ||||
|       "version": "4.52.5", | ||||
|       "resolved": "https://registry.npmjs.org/@rollup/rollup-android-arm-eabi/-/rollup-android-arm-eabi-4.52.5.tgz", | ||||
|       "integrity": "sha512-8c1vW4ocv3UOMp9K+gToY5zL2XiiVw3k7f1ksf4yO1FlDFQ1C2u72iACFnSOceJFsWskc2WZNqeRhFRPzv+wtQ==", | ||||
|       "cpu": [ | ||||
|         "arm" | ||||
|       ], | ||||
| @@ -373,9 +373,9 @@ | ||||
|       ] | ||||
|     }, | ||||
|     "node_modules/@rollup/rollup-android-arm64": { | ||||
|       "version": "4.52.2", | ||||
|       "resolved": "https://registry.npmjs.org/@rollup/rollup-android-arm64/-/rollup-android-arm64-4.52.2.tgz", | ||||
|       "integrity": "sha512-cqFSWO5tX2vhC9hJTK8WAiPIm4Q8q/cU8j2HQA0L3E1uXvBYbOZMhE2oFL8n2pKB5sOCHY6bBuHaRwG7TkfJyw==", | ||||
|       "version": "4.52.5", | ||||
|       "resolved": "https://registry.npmjs.org/@rollup/rollup-android-arm64/-/rollup-android-arm64-4.52.5.tgz", | ||||
|       "integrity": "sha512-mQGfsIEFcu21mvqkEKKu2dYmtuSZOBMmAl5CFlPGLY94Vlcm+zWApK7F/eocsNzp8tKmbeBP8yXyAbx0XHsFNA==", | ||||
|       "cpu": [ | ||||
|         "arm64" | ||||
|       ], | ||||
| @@ -386,9 +386,9 @@ | ||||
|       ] | ||||
|     }, | ||||
|     "node_modules/@rollup/rollup-darwin-arm64": { | ||||
|       "version": "4.52.2", | ||||
|       "resolved": "https://registry.npmjs.org/@rollup/rollup-darwin-arm64/-/rollup-darwin-arm64-4.52.2.tgz", | ||||
|       "integrity": "sha512-vngduywkkv8Fkh3wIZf5nFPXzWsNsVu1kvtLETWxTFf/5opZmflgVSeLgdHR56RQh71xhPhWoOkEBvbehwTlVA==", | ||||
|       "version": "4.52.5", | ||||
|       "resolved": "https://registry.npmjs.org/@rollup/rollup-darwin-arm64/-/rollup-darwin-arm64-4.52.5.tgz", | ||||
|       "integrity": "sha512-takF3CR71mCAGA+v794QUZ0b6ZSrgJkArC+gUiG6LB6TQty9T0Mqh3m2ImRBOxS2IeYBo4lKWIieSvnEk2OQWA==", | ||||
|       "cpu": [ | ||||
|         "arm64" | ||||
|       ], | ||||
| @@ -399,9 +399,9 @@ | ||||
|       ] | ||||
|     }, | ||||
|     "node_modules/@rollup/rollup-darwin-x64": { | ||||
|       "version": "4.52.2", | ||||
|       "resolved": "https://registry.npmjs.org/@rollup/rollup-darwin-x64/-/rollup-darwin-x64-4.52.2.tgz", | ||||
|       "integrity": "sha512-h11KikYrUCYTrDj6h939hhMNlqU2fo/X4NB0OZcys3fya49o1hmFaczAiJWVAFgrM1NCP6RrO7lQKeVYSKBPSQ==", | ||||
|       "version": "4.52.5", | ||||
|       "resolved": "https://registry.npmjs.org/@rollup/rollup-darwin-x64/-/rollup-darwin-x64-4.52.5.tgz", | ||||
|       "integrity": "sha512-W901Pla8Ya95WpxDn//VF9K9u2JbocwV/v75TE0YIHNTbhqUTv9w4VuQ9MaWlNOkkEfFwkdNhXgcLqPSmHy0fA==", | ||||
|       "cpu": [ | ||||
|         "x64" | ||||
|       ], | ||||
| @@ -412,9 +412,9 @@ | ||||
|       ] | ||||
|     }, | ||||
|     "node_modules/@rollup/rollup-freebsd-arm64": { | ||||
|       "version": "4.52.2", | ||||
|       "resolved": "https://registry.npmjs.org/@rollup/rollup-freebsd-arm64/-/rollup-freebsd-arm64-4.52.2.tgz", | ||||
|       "integrity": "sha512-/eg4CI61ZUkLXxMHyVlmlGrSQZ34xqWlZNW43IAU4RmdzWEx0mQJ2mN/Cx4IHLVZFL6UBGAh+/GXhgvGb+nVxw==", | ||||
|       "version": "4.52.5", | ||||
|       "resolved": "https://registry.npmjs.org/@rollup/rollup-freebsd-arm64/-/rollup-freebsd-arm64-4.52.5.tgz", | ||||
|       "integrity": "sha512-QofO7i7JycsYOWxe0GFqhLmF6l1TqBswJMvICnRUjqCx8b47MTo46W8AoeQwiokAx3zVryVnxtBMcGcnX12LvA==", | ||||
|       "cpu": [ | ||||
|         "arm64" | ||||
|       ], | ||||
| @@ -425,9 +425,9 @@ | ||||
|       ] | ||||
|     }, | ||||
|     "node_modules/@rollup/rollup-freebsd-x64": { | ||||
|       "version": "4.52.2", | ||||
|       "resolved": "https://registry.npmjs.org/@rollup/rollup-freebsd-x64/-/rollup-freebsd-x64-4.52.2.tgz", | ||||
|       "integrity": "sha512-QOWgFH5X9+p+S1NAfOqc0z8qEpJIoUHf7OWjNUGOeW18Mx22lAUOiA9b6r2/vpzLdfxi/f+VWsYjUOMCcYh0Ng==", | ||||
|       "version": "4.52.5", | ||||
|       "resolved": "https://registry.npmjs.org/@rollup/rollup-freebsd-x64/-/rollup-freebsd-x64-4.52.5.tgz", | ||||
|       "integrity": "sha512-jr21b/99ew8ujZubPo9skbrItHEIE50WdV86cdSoRkKtmWa+DDr6fu2c/xyRT0F/WazZpam6kk7IHBerSL7LDQ==", | ||||
|       "cpu": [ | ||||
|         "x64" | ||||
|       ], | ||||
| @@ -438,9 +438,9 @@ | ||||
|       ] | ||||
|     }, | ||||
|     "node_modules/@rollup/rollup-linux-arm-gnueabihf": { | ||||
|       "version": "4.52.2", | ||||
|       "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm-gnueabihf/-/rollup-linux-arm-gnueabihf-4.52.2.tgz", | ||||
|       "integrity": "sha512-kDWSPafToDd8LcBYd1t5jw7bD5Ojcu12S3uT372e5HKPzQt532vW+rGFFOaiR0opxePyUkHrwz8iWYEyH1IIQA==", | ||||
|       "version": "4.52.5", | ||||
|       "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm-gnueabihf/-/rollup-linux-arm-gnueabihf-4.52.5.tgz", | ||||
|       "integrity": "sha512-PsNAbcyv9CcecAUagQefwX8fQn9LQ4nZkpDboBOttmyffnInRy8R8dSg6hxxl2Re5QhHBf6FYIDhIj5v982ATQ==", | ||||
|       "cpu": [ | ||||
|         "arm" | ||||
|       ], | ||||
| @@ -451,9 +451,9 @@ | ||||
|       ] | ||||
|     }, | ||||
|     "node_modules/@rollup/rollup-linux-arm-musleabihf": { | ||||
|       "version": "4.52.2", | ||||
|       "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm-musleabihf/-/rollup-linux-arm-musleabihf-4.52.2.tgz", | ||||
|       "integrity": "sha512-gKm7Mk9wCv6/rkzwCiUC4KnevYhlf8ztBrDRT9g/u//1fZLapSRc+eDZj2Eu2wpJ+0RzUKgtNijnVIB4ZxyL+w==", | ||||
|       "version": "4.52.5", | ||||
|       "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm-musleabihf/-/rollup-linux-arm-musleabihf-4.52.5.tgz", | ||||
|       "integrity": "sha512-Fw4tysRutyQc/wwkmcyoqFtJhh0u31K+Q6jYjeicsGJJ7bbEq8LwPWV/w0cnzOqR2m694/Af6hpFayLJZkG2VQ==", | ||||
|       "cpu": [ | ||||
|         "arm" | ||||
|       ], | ||||
| @@ -464,9 +464,9 @@ | ||||
|       ] | ||||
|     }, | ||||
|     "node_modules/@rollup/rollup-linux-arm64-gnu": { | ||||
|       "version": "4.52.2", | ||||
|       "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm64-gnu/-/rollup-linux-arm64-gnu-4.52.2.tgz", | ||||
|       "integrity": "sha512-66lA8vnj5mB/rtDNwPgrrKUOtCLVQypkyDa2gMfOefXK6rcZAxKLO9Fy3GkW8VkPnENv9hBkNOFfGLf6rNKGUg==", | ||||
|       "version": "4.52.5", | ||||
|       "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm64-gnu/-/rollup-linux-arm64-gnu-4.52.5.tgz", | ||||
|       "integrity": "sha512-a+3wVnAYdQClOTlyapKmyI6BLPAFYs0JM8HRpgYZQO02rMR09ZcV9LbQB+NL6sljzG38869YqThrRnfPMCDtZg==", | ||||
|       "cpu": [ | ||||
|         "arm64" | ||||
|       ], | ||||
| @@ -477,9 +477,9 @@ | ||||
|       ] | ||||
|     }, | ||||
|     "node_modules/@rollup/rollup-linux-arm64-musl": { | ||||
|       "version": "4.52.2", | ||||
|       "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm64-musl/-/rollup-linux-arm64-musl-4.52.2.tgz", | ||||
|       "integrity": "sha512-s+OPucLNdJHvuZHuIz2WwncJ+SfWHFEmlC5nKMUgAelUeBUnlB4wt7rXWiyG4Zn07uY2Dd+SGyVa9oyLkVGOjA==", | ||||
|       "version": "4.52.5", | ||||
|       "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm64-musl/-/rollup-linux-arm64-musl-4.52.5.tgz", | ||||
|       "integrity": "sha512-AvttBOMwO9Pcuuf7m9PkC1PUIKsfaAJ4AYhy944qeTJgQOqJYJ9oVl2nYgY7Rk0mkbsuOpCAYSs6wLYB2Xiw0Q==", | ||||
|       "cpu": [ | ||||
|         "arm64" | ||||
|       ], | ||||
| @@ -490,9 +490,9 @@ | ||||
|       ] | ||||
|     }, | ||||
|     "node_modules/@rollup/rollup-linux-loong64-gnu": { | ||||
|       "version": "4.52.2", | ||||
|       "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-loong64-gnu/-/rollup-linux-loong64-gnu-4.52.2.tgz", | ||||
|       "integrity": "sha512-8wTRM3+gVMDLLDdaT6tKmOE3lJyRy9NpJUS/ZRWmLCmOPIJhVyXwjBo+XbrrwtV33Em1/eCTd5TuGJm4+DmYjw==", | ||||
|       "version": "4.52.5", | ||||
|       "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-loong64-gnu/-/rollup-linux-loong64-gnu-4.52.5.tgz", | ||||
|       "integrity": "sha512-DkDk8pmXQV2wVrF6oq5tONK6UHLz/XcEVow4JTTerdeV1uqPeHxwcg7aFsfnSm9L+OO8WJsWotKM2JJPMWrQtA==", | ||||
|       "cpu": [ | ||||
|         "loong64" | ||||
|       ], | ||||
| @@ -503,9 +503,9 @@ | ||||
|       ] | ||||
|     }, | ||||
|     "node_modules/@rollup/rollup-linux-ppc64-gnu": { | ||||
|       "version": "4.52.2", | ||||
|       "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-ppc64-gnu/-/rollup-linux-ppc64-gnu-4.52.2.tgz", | ||||
|       "integrity": "sha512-6yqEfgJ1anIeuP2P/zhtfBlDpXUb80t8DpbYwXQ3bQd95JMvUaqiX+fKqYqUwZXqdJDd8xdilNtsHM2N0cFm6A==", | ||||
|       "version": "4.52.5", | ||||
|       "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-ppc64-gnu/-/rollup-linux-ppc64-gnu-4.52.5.tgz", | ||||
|       "integrity": "sha512-W/b9ZN/U9+hPQVvlGwjzi+Wy4xdoH2I8EjaCkMvzpI7wJUs8sWJ03Rq96jRnHkSrcHTpQe8h5Tg3ZzUPGauvAw==", | ||||
|       "cpu": [ | ||||
|         "ppc64" | ||||
|       ], | ||||
| @@ -516,9 +516,9 @@ | ||||
|       ] | ||||
|     }, | ||||
|     "node_modules/@rollup/rollup-linux-riscv64-gnu": { | ||||
|       "version": "4.52.2", | ||||
|       "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-riscv64-gnu/-/rollup-linux-riscv64-gnu-4.52.2.tgz", | ||||
|       "integrity": "sha512-sshYUiYVSEI2B6dp4jMncwxbrUqRdNApF2c3bhtLAU0qA8Lrri0p0NauOsTWh3yCCCDyBOjESHMExonp7Nzc0w==", | ||||
|       "version": "4.52.5", | ||||
|       "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-riscv64-gnu/-/rollup-linux-riscv64-gnu-4.52.5.tgz", | ||||
|       "integrity": "sha512-sjQLr9BW7R/ZiXnQiWPkErNfLMkkWIoCz7YMn27HldKsADEKa5WYdobaa1hmN6slu9oWQbB6/jFpJ+P2IkVrmw==", | ||||
|       "cpu": [ | ||||
|         "riscv64" | ||||
|       ], | ||||
| @@ -529,9 +529,9 @@ | ||||
|       ] | ||||
|     }, | ||||
|     "node_modules/@rollup/rollup-linux-riscv64-musl": { | ||||
|       "version": "4.52.2", | ||||
|       "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-riscv64-musl/-/rollup-linux-riscv64-musl-4.52.2.tgz", | ||||
|       "integrity": "sha512-duBLgd+3pqC4MMwBrKkFxaZerUxZcYApQVC5SdbF5/e/589GwVvlRUnyqMFbM8iUSb1BaoX/3fRL7hB9m2Pj8Q==", | ||||
|       "version": "4.52.5", | ||||
|       "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-riscv64-musl/-/rollup-linux-riscv64-musl-4.52.5.tgz", | ||||
|       "integrity": "sha512-hq3jU/kGyjXWTvAh2awn8oHroCbrPm8JqM7RUpKjalIRWWXE01CQOf/tUNWNHjmbMHg/hmNCwc/Pz3k1T/j/Lg==", | ||||
|       "cpu": [ | ||||
|         "riscv64" | ||||
|       ], | ||||
| @@ -542,9 +542,9 @@ | ||||
|       ] | ||||
|     }, | ||||
|     "node_modules/@rollup/rollup-linux-s390x-gnu": { | ||||
|       "version": "4.52.2", | ||||
|       "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-s390x-gnu/-/rollup-linux-s390x-gnu-4.52.2.tgz", | ||||
|       "integrity": "sha512-tzhYJJidDUVGMgVyE+PmxENPHlvvqm1KILjjZhB8/xHYqAGeizh3GBGf9u6WdJpZrz1aCpIIHG0LgJgH9rVjHQ==", | ||||
|       "version": "4.52.5", | ||||
|       "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-s390x-gnu/-/rollup-linux-s390x-gnu-4.52.5.tgz", | ||||
|       "integrity": "sha512-gn8kHOrku8D4NGHMK1Y7NA7INQTRdVOntt1OCYypZPRt6skGbddska44K8iocdpxHTMMNui5oH4elPH4QOLrFQ==", | ||||
|       "cpu": [ | ||||
|         "s390x" | ||||
|       ], | ||||
| @@ -555,9 +555,9 @@ | ||||
|       ] | ||||
|     }, | ||||
|     "node_modules/@rollup/rollup-linux-x64-gnu": { | ||||
|       "version": "4.52.2", | ||||
|       "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-x64-gnu/-/rollup-linux-x64-gnu-4.52.2.tgz", | ||||
|       "integrity": "sha512-opH8GSUuVcCSSyHHcl5hELrmnk4waZoVpgn/4FDao9iyE4WpQhyWJ5ryl5M3ocp4qkRuHfyXnGqg8M9oKCEKRA==", | ||||
|       "version": "4.52.5", | ||||
|       "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-x64-gnu/-/rollup-linux-x64-gnu-4.52.5.tgz", | ||||
|       "integrity": "sha512-hXGLYpdhiNElzN770+H2nlx+jRog8TyynpTVzdlc6bndktjKWyZyiCsuDAlpd+j+W+WNqfcyAWz9HxxIGfZm1Q==", | ||||
|       "cpu": [ | ||||
|         "x64" | ||||
|       ], | ||||
| @@ -568,9 +568,9 @@ | ||||
|       ] | ||||
|     }, | ||||
|     "node_modules/@rollup/rollup-linux-x64-musl": { | ||||
|       "version": "4.52.2", | ||||
|       "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-x64-musl/-/rollup-linux-x64-musl-4.52.2.tgz", | ||||
|       "integrity": "sha512-LSeBHnGli1pPKVJ79ZVJgeZWWZXkEe/5o8kcn23M8eMKCUANejchJbF/JqzM4RRjOJfNRhKJk8FuqL1GKjF5oQ==", | ||||
|       "version": "4.52.5", | ||||
|       "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-x64-musl/-/rollup-linux-x64-musl-4.52.5.tgz", | ||||
|       "integrity": "sha512-arCGIcuNKjBoKAXD+y7XomR9gY6Mw7HnFBv5Rw7wQRvwYLR7gBAgV7Mb2QTyjXfTveBNFAtPt46/36vV9STLNg==", | ||||
|       "cpu": [ | ||||
|         "x64" | ||||
|       ], | ||||
| @@ -581,9 +581,9 @@ | ||||
|       ] | ||||
|     }, | ||||
|     "node_modules/@rollup/rollup-openharmony-arm64": { | ||||
|       "version": "4.52.2", | ||||
|       "resolved": "https://registry.npmjs.org/@rollup/rollup-openharmony-arm64/-/rollup-openharmony-arm64-4.52.2.tgz", | ||||
|       "integrity": "sha512-uPj7MQ6/s+/GOpolavm6BPo+6CbhbKYyZHUDvZ/SmJM7pfDBgdGisFX3bY/CBDMg2ZO4utfhlApkSfZ92yXw7Q==", | ||||
|       "version": "4.52.5", | ||||
|       "resolved": "https://registry.npmjs.org/@rollup/rollup-openharmony-arm64/-/rollup-openharmony-arm64-4.52.5.tgz", | ||||
|       "integrity": "sha512-QoFqB6+/9Rly/RiPjaomPLmR/13cgkIGfA40LHly9zcH1S0bN2HVFYk3a1eAyHQyjs3ZJYlXvIGtcCs5tko9Cw==", | ||||
|       "cpu": [ | ||||
|         "arm64" | ||||
|       ], | ||||
| @@ -594,9 +594,9 @@ | ||||
|       ] | ||||
|     }, | ||||
|     "node_modules/@rollup/rollup-win32-arm64-msvc": { | ||||
|       "version": "4.52.2", | ||||
|       "resolved": "https://registry.npmjs.org/@rollup/rollup-win32-arm64-msvc/-/rollup-win32-arm64-msvc-4.52.2.tgz", | ||||
|       "integrity": "sha512-Z9MUCrSgIaUeeHAiNkm3cQyst2UhzjPraR3gYYfOjAuZI7tcFRTOD+4cHLPoS/3qinchth+V56vtqz1Tv+6KPA==", | ||||
|       "version": "4.52.5", | ||||
|       "resolved": "https://registry.npmjs.org/@rollup/rollup-win32-arm64-msvc/-/rollup-win32-arm64-msvc-4.52.5.tgz", | ||||
|       "integrity": "sha512-w0cDWVR6MlTstla1cIfOGyl8+qb93FlAVutcor14Gf5Md5ap5ySfQ7R9S/NjNaMLSFdUnKGEasmVnu3lCMqB7w==", | ||||
|       "cpu": [ | ||||
|         "arm64" | ||||
|       ], | ||||
| @@ -607,9 +607,9 @@ | ||||
|       ] | ||||
|     }, | ||||
|     "node_modules/@rollup/rollup-win32-ia32-msvc": { | ||||
|       "version": "4.52.2", | ||||
|       "resolved": "https://registry.npmjs.org/@rollup/rollup-win32-ia32-msvc/-/rollup-win32-ia32-msvc-4.52.2.tgz", | ||||
|       "integrity": "sha512-+GnYBmpjldD3XQd+HMejo+0gJGwYIOfFeoBQv32xF/RUIvccUz20/V6Otdv+57NE70D5pa8W/jVGDoGq0oON4A==", | ||||
|       "version": "4.52.5", | ||||
|       "resolved": "https://registry.npmjs.org/@rollup/rollup-win32-ia32-msvc/-/rollup-win32-ia32-msvc-4.52.5.tgz", | ||||
|       "integrity": "sha512-Aufdpzp7DpOTULJCuvzqcItSGDH73pF3ko/f+ckJhxQyHtp67rHw3HMNxoIdDMUITJESNE6a8uh4Lo4SLouOUg==", | ||||
|       "cpu": [ | ||||
|         "ia32" | ||||
|       ], | ||||
| @@ -620,9 +620,9 @@ | ||||
|       ] | ||||
|     }, | ||||
|     "node_modules/@rollup/rollup-win32-x64-gnu": { | ||||
|       "version": "4.52.2", | ||||
|       "resolved": "https://registry.npmjs.org/@rollup/rollup-win32-x64-gnu/-/rollup-win32-x64-gnu-4.52.2.tgz", | ||||
|       "integrity": "sha512-ApXFKluSB6kDQkAqZOKXBjiaqdF1BlKi+/eqnYe9Ee7U2K3pUDKsIyr8EYm/QDHTJIM+4X+lI0gJc3TTRhd+dA==", | ||||
|       "version": "4.52.5", | ||||
|       "resolved": "https://registry.npmjs.org/@rollup/rollup-win32-x64-gnu/-/rollup-win32-x64-gnu-4.52.5.tgz", | ||||
|       "integrity": "sha512-UGBUGPFp1vkj6p8wCRraqNhqwX/4kNQPS57BCFc8wYh0g94iVIW33wJtQAx3G7vrjjNtRaxiMUylM0ktp/TRSQ==", | ||||
|       "cpu": [ | ||||
|         "x64" | ||||
|       ], | ||||
| @@ -633,9 +633,9 @@ | ||||
|       ] | ||||
|     }, | ||||
|     "node_modules/@rollup/rollup-win32-x64-msvc": { | ||||
|       "version": "4.52.2", | ||||
|       "resolved": "https://registry.npmjs.org/@rollup/rollup-win32-x64-msvc/-/rollup-win32-x64-msvc-4.52.2.tgz", | ||||
|       "integrity": "sha512-ARz+Bs8kY6FtitYM96PqPEVvPXqEZmPZsSkXvyX19YzDqkCaIlhCieLLMI5hxO9SRZ2XtCtm8wxhy0iJ2jxNfw==", | ||||
|       "version": "4.52.5", | ||||
|       "resolved": "https://registry.npmjs.org/@rollup/rollup-win32-x64-msvc/-/rollup-win32-x64-msvc-4.52.5.tgz", | ||||
|       "integrity": "sha512-TAcgQh2sSkykPRWLrdyy2AiceMckNf5loITqXxFI5VuQjS5tSuw3WlwdN8qv8vzjLAUTvYaH/mVjSFpbkFbpTg==", | ||||
|       "cpu": [ | ||||
|         "x64" | ||||
|       ], | ||||
| @@ -805,12 +805,12 @@ | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/resolve": { | ||||
|       "version": "1.22.10", | ||||
|       "resolved": "https://registry.npmjs.org/resolve/-/resolve-1.22.10.tgz", | ||||
|       "integrity": "sha512-NPRy+/ncIMeDlTAsuqwKIiferiawhefFJtkNSW0qZJEqMEb+qBt/77B/jGeeek+F0uOeN05CDa6HXbbIgtVX4w==", | ||||
|       "version": "1.22.11", | ||||
|       "resolved": "https://registry.npmjs.org/resolve/-/resolve-1.22.11.tgz", | ||||
|       "integrity": "sha512-RfqAvLnMl313r7c9oclB1HhUEAezcpLjz95wFH4LVuhk9JF/r22qmVP9AMmOU4vMX7Q8pN8jwNg/CSpdFnMjTQ==", | ||||
|       "license": "MIT", | ||||
|       "dependencies": { | ||||
|         "is-core-module": "^2.16.0", | ||||
|         "is-core-module": "^2.16.1", | ||||
|         "path-parse": "^1.0.7", | ||||
|         "supports-preserve-symlinks-flag": "^1.0.0" | ||||
|       }, | ||||
| @@ -825,9 +825,9 @@ | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/rollup": { | ||||
|       "version": "4.52.2", | ||||
|       "resolved": "https://registry.npmjs.org/rollup/-/rollup-4.52.2.tgz", | ||||
|       "integrity": "sha512-I25/2QgoROE1vYV+NQ1En9T9UFB9Cmfm2CJ83zZOlaDpvz29wGQSZXWKw7MiNXau7wYgB/T9fVIdIuEQ+KbiiA==", | ||||
|       "version": "4.52.5", | ||||
|       "resolved": "https://registry.npmjs.org/rollup/-/rollup-4.52.5.tgz", | ||||
|       "integrity": "sha512-3GuObel8h7Kqdjt0gxkEzaifHTqLVW56Y/bjN7PSQtkKr0w3V/QYSdt6QWYtd7A1xUtYQigtdUfgj1RvWVtorw==", | ||||
|       "license": "MIT", | ||||
|       "dependencies": { | ||||
|         "@types/estree": "1.0.8" | ||||
| @@ -840,28 +840,28 @@ | ||||
|         "npm": ">=8.0.0" | ||||
|       }, | ||||
|       "optionalDependencies": { | ||||
|         "@rollup/rollup-android-arm-eabi": "4.52.2", | ||||
|         "@rollup/rollup-android-arm64": "4.52.2", | ||||
|         "@rollup/rollup-darwin-arm64": "4.52.2", | ||||
|         "@rollup/rollup-darwin-x64": "4.52.2", | ||||
|         "@rollup/rollup-freebsd-arm64": "4.52.2", | ||||
|         "@rollup/rollup-freebsd-x64": "4.52.2", | ||||
|         "@rollup/rollup-linux-arm-gnueabihf": "4.52.2", | ||||
|         "@rollup/rollup-linux-arm-musleabihf": "4.52.2", | ||||
|         "@rollup/rollup-linux-arm64-gnu": "4.52.2", | ||||
|         "@rollup/rollup-linux-arm64-musl": "4.52.2", | ||||
|         "@rollup/rollup-linux-loong64-gnu": "4.52.2", | ||||
|         "@rollup/rollup-linux-ppc64-gnu": "4.52.2", | ||||
|         "@rollup/rollup-linux-riscv64-gnu": "4.52.2", | ||||
|         "@rollup/rollup-linux-riscv64-musl": "4.52.2", | ||||
|         "@rollup/rollup-linux-s390x-gnu": "4.52.2", | ||||
|         "@rollup/rollup-linux-x64-gnu": "4.52.2", | ||||
|         "@rollup/rollup-linux-x64-musl": "4.52.2", | ||||
|         "@rollup/rollup-openharmony-arm64": "4.52.2", | ||||
|         "@rollup/rollup-win32-arm64-msvc": "4.52.2", | ||||
|         "@rollup/rollup-win32-ia32-msvc": "4.52.2", | ||||
|         "@rollup/rollup-win32-x64-gnu": "4.52.2", | ||||
|         "@rollup/rollup-win32-x64-msvc": "4.52.2", | ||||
|         "@rollup/rollup-android-arm-eabi": "4.52.5", | ||||
|         "@rollup/rollup-android-arm64": "4.52.5", | ||||
|         "@rollup/rollup-darwin-arm64": "4.52.5", | ||||
|         "@rollup/rollup-darwin-x64": "4.52.5", | ||||
|         "@rollup/rollup-freebsd-arm64": "4.52.5", | ||||
|         "@rollup/rollup-freebsd-x64": "4.52.5", | ||||
|         "@rollup/rollup-linux-arm-gnueabihf": "4.52.5", | ||||
|         "@rollup/rollup-linux-arm-musleabihf": "4.52.5", | ||||
|         "@rollup/rollup-linux-arm64-gnu": "4.52.5", | ||||
|         "@rollup/rollup-linux-arm64-musl": "4.52.5", | ||||
|         "@rollup/rollup-linux-loong64-gnu": "4.52.5", | ||||
|         "@rollup/rollup-linux-ppc64-gnu": "4.52.5", | ||||
|         "@rollup/rollup-linux-riscv64-gnu": "4.52.5", | ||||
|         "@rollup/rollup-linux-riscv64-musl": "4.52.5", | ||||
|         "@rollup/rollup-linux-s390x-gnu": "4.52.5", | ||||
|         "@rollup/rollup-linux-x64-gnu": "4.52.5", | ||||
|         "@rollup/rollup-linux-x64-musl": "4.52.5", | ||||
|         "@rollup/rollup-openharmony-arm64": "4.52.5", | ||||
|         "@rollup/rollup-win32-arm64-msvc": "4.52.5", | ||||
|         "@rollup/rollup-win32-ia32-msvc": "4.52.5", | ||||
|         "@rollup/rollup-win32-x64-gnu": "4.52.5", | ||||
|         "@rollup/rollup-win32-x64-msvc": "4.52.5", | ||||
|         "fsevents": "~2.3.2" | ||||
|       } | ||||
|     }, | ||||
| @@ -925,9 +925,9 @@ | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/style-mod": { | ||||
|       "version": "4.1.2", | ||||
|       "resolved": "https://registry.npmjs.org/style-mod/-/style-mod-4.1.2.tgz", | ||||
|       "integrity": "sha512-wnD1HyVqpJUI2+eKZ+eo1UwghftP6yuFheBqqe+bWCotBjC2K1YnteJILRMs3SM4V/0dLEW1SC27MWP5y+mwmw==", | ||||
|       "version": "4.1.3", | ||||
|       "resolved": "https://registry.npmjs.org/style-mod/-/style-mod-4.1.3.tgz", | ||||
|       "integrity": "sha512-i/n8VsZydrugj3Iuzll8+x/00GH2vnYsk1eomD8QiRrSAeW6ItbCQDtfXCeJHd0iwiNagqjQkvpvREEPtW3IoQ==", | ||||
|       "license": "MIT" | ||||
|     }, | ||||
|     "node_modules/supports-preserve-symlinks-flag": { | ||||
|   | ||||
							
								
								
									
										2
									
								
								deps/libbacktrace
									
									
									
									
										vendored
									
									
								
							
							
								
								
								
								
								
							
						
						
									
										2
									
								
								deps/libbacktrace
									
									
									
									
										vendored
									
									
								
							 Submodule deps/libbacktrace updated: f1104f3270...2f67a3abfd
									
								
							
							
								
								
									
										1
									
								
								deps/openssl_src
									
									
									
									
										vendored
									
									
								
							
							
								
								
								
								
								
							
						
						
									
										1
									
								
								deps/openssl_src
									
									
									
									
										vendored
									
									
								
							 Submodule deps/openssl_src deleted from c4da9ac23d
									
								
							
							
								
								
									
										2
									
								
								deps/speedscope/index.html
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								deps/speedscope/index.html
									
									
									
									
										vendored
									
									
								
							| @@ -11,7 +11,7 @@ | ||||
|     <link rel="icon" type="image/x-icon" href="favicon-FOKUP5Y5.ico"> | ||||
|   </head> | ||||
|   <body> | ||||
|     <script src="speedscope-HCR63FMT.js"></script> | ||||
|     <script src="speedscope-432XE7GS.js"></script> | ||||
|      | ||||
|      | ||||
|      | ||||
|   | ||||
							
								
								
									
										6
									
								
								deps/speedscope/release.txt
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										6
									
								
								deps/speedscope/release.txt
									
									
									
									
										vendored
									
									
								
							| @@ -1,3 +1,3 @@ | ||||
| speedscope@1.23.1 | ||||
| Mon Aug 11 11:43:09 PDT 2025 | ||||
| 0cec0f82c334aed6cf19d43cabeadcda0d95e0fc | ||||
| speedscope@1.24.0 | ||||
| Mon Oct 20 18:11:29 PDT 2025 | ||||
| fc76932551754a442cd5c4f0afdba28032d14d8a | ||||
|   | ||||
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							| @@ -41,11 +41,9 @@ options: | ||||
|                                  ssb_port (default: 8008): Port on which to listen for SSB secure handshake connections. | ||||
|                                  http_local_only (default: false): Whether to bind http(s) to the loopback address.  Otherwise any. | ||||
|                                  http_port (default: 12345): Port on which to listen for HTTP connections. | ||||
|                                  https_port (default: 0): Port on which to listen for secure HTTP connections. | ||||
|                                  out_http_port_file (default: ""): File to which to write bound HTTP port. | ||||
|                                  blob_fetch_age_seconds (default: -1): Only blobs mentioned more recently than this age will be automatically fetched. | ||||
|                                  blob_expire_age_seconds (default: -1): Blobs older than this will be automatically deleted. | ||||
|                                  fetch_hosts (default: ""): Comma-separated list of host names to which HTTP fetch requests are allowed.  None if empty. | ||||
|                                  http_redirect (default: ""): If connecting by HTTP and HTTPS is configured, Location header prefix (ie, "http://example.com") | ||||
|                                  index (default: "/~core/intro/"): Default path. | ||||
|                                  index_map (default: ""): Mappings from hostname to redirect path, one per line, as in: "www.tildefriends.net=/~core/index/" | ||||
|   | ||||
							
								
								
									
										8
									
								
								metadata/en-US/changelogs/44.txt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										8
									
								
								metadata/en-US/changelogs/44.txt
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,8 @@ | ||||
| * Remove OpenSSL. | ||||
| * Exclude messages for subscribed channels from the general feed. | ||||
| * Remove the query tab in favor of searching for "sql:SELECT ...". | ||||
| * Faster load times. | ||||
| * Fix contact groups expanding/collapsing. | ||||
| * Give channel subscribe/unsubscribe similar grouping treatment to follows/blocks. | ||||
| * Slightly improved following/blocking message display. | ||||
| * Updates: CodeMirror, libbacktrace, and speedscope 1.24.0. | ||||
| @@ -292,7 +292,7 @@ public class TildeFriendsActivity extends Activity { | ||||
| 		StrictMode.setThreadPolicy( | ||||
| 			new StrictMode.ThreadPolicy.Builder() | ||||
| 				.detectAll() | ||||
| 				.penaltyDialog() | ||||
| 				//.penaltyDialog() | ||||
| 				.penaltyLog() | ||||
| 				.build()); | ||||
| 		StrictMode.setVmPolicy( | ||||
|   | ||||
							
								
								
									
										623
									
								
								src/api.js.c
									
									
									
									
									
								
							
							
						
						
									
										623
									
								
								src/api.js.c
									
									
									
									
									
								
							| @@ -74,7 +74,7 @@ static void _tf_api_core_apps_after_work(tf_ssb_t* ssb, int status, void* user_d | ||||
| 	JSValue result = JS_NewObject(context); | ||||
| 	for (int i = 0; i < work->count; i++) | ||||
| 	{ | ||||
| 		JS_SetPropertyStr(context, result, work->apps[i].app, JS_NewString(context, work->apps[i].path)); | ||||
| 		JS_SetPropertyStr(context, result, work->apps[i].app, JS_NewString(context, work->apps[i].path ? work->apps[i].path : "")); | ||||
| 	} | ||||
| 	JSValue error = JS_Call(context, work->promise[0], JS_UNDEFINED, 1, &result); | ||||
| 	tf_util_report_error(context, error); | ||||
| @@ -147,6 +147,55 @@ static JSValue _tf_api_core_apps(JSContext* context, JSValueConst this_val, int | ||||
| 	return result; | ||||
| } | ||||
|  | ||||
| typedef struct _users_t | ||||
| { | ||||
| 	const char* users; | ||||
| 	JSContext* context; | ||||
| 	JSValue promise[2]; | ||||
| } users_t; | ||||
|  | ||||
| static void _tf_api_core_users_work(tf_ssb_t* ssb, void* user_data) | ||||
| { | ||||
| 	users_t* work = user_data; | ||||
| 	work->users = tf_ssb_db_get_property(ssb, "auth", "users"); | ||||
| } | ||||
|  | ||||
| static void _tf_api_core_users_after_work(tf_ssb_t* ssb, int status, void* user_data) | ||||
| { | ||||
| 	users_t* work = user_data; | ||||
| 	JSContext* context = work->context; | ||||
| 	JSValue result = JS_UNDEFINED; | ||||
| 	if (work->users) | ||||
| 	{ | ||||
| 		result = JS_ParseJSON(context, work->users, strlen(work->users), NULL); | ||||
| 		tf_free((void*)work->users); | ||||
| 	} | ||||
| 	if (JS_IsUndefined(result)) | ||||
| 	{ | ||||
| 		result = JS_NewArray(context); | ||||
| 	} | ||||
| 	JSValue error = JS_Call(context, JS_IsArray(context, result) ? work->promise[0] : work->promise[1], JS_UNDEFINED, 1, &result); | ||||
| 	tf_util_report_error(context, error); | ||||
| 	JS_FreeValue(context, error); | ||||
| 	JS_FreeValue(context, result); | ||||
| 	JS_FreeValue(context, work->promise[0]); | ||||
| 	JS_FreeValue(context, work->promise[1]); | ||||
| 	tf_free(work); | ||||
| } | ||||
|  | ||||
| static JSValue _tf_api_core_users(JSContext* context, JSValueConst this_val, int argc, JSValueConst* argv, int magic, JSValue* data) | ||||
| { | ||||
| 	tf_task_t* task = tf_task_get(context); | ||||
| 	tf_ssb_t* ssb = tf_task_get_ssb(task); | ||||
| 	users_t* work = tf_malloc(sizeof(users_t)); | ||||
| 	*work = (users_t) { | ||||
| 		.context = context, | ||||
| 	}; | ||||
| 	JSValue result = JS_NewPromiseCapability(context, work->promise); | ||||
| 	tf_ssb_run_work(ssb, _tf_api_core_users_work, _tf_api_core_users_after_work, work); | ||||
| 	return result; | ||||
| } | ||||
|  | ||||
| static JSValue _tf_api_core_register(JSContext* context, JSValueConst this_val, int argc, JSValueConst* argv, int magic, JSValue* data) | ||||
| { | ||||
| 	JSValue event_name = argv[0]; | ||||
| @@ -210,6 +259,546 @@ static JSValue _tf_api_core_unregister(JSContext* context, JSValueConst this_val | ||||
| 	return JS_UNDEFINED; | ||||
| } | ||||
|  | ||||
| typedef struct _permissions_for_user_t | ||||
| { | ||||
| 	const char* user; | ||||
| 	const char* settings; | ||||
| 	JSContext* context; | ||||
| 	JSValue promise[2]; | ||||
| } permissions_for_user_t; | ||||
|  | ||||
| static void _tf_api_core_permissions_for_user_work(tf_ssb_t* ssb, void* user_data) | ||||
| { | ||||
| 	permissions_for_user_t* work = user_data; | ||||
| 	work->settings = tf_ssb_db_get_property(ssb, "core", "settings"); | ||||
| } | ||||
|  | ||||
| static void _tf_api_core_permissions_for_user_after_work(tf_ssb_t* ssb, int status, void* user_data) | ||||
| { | ||||
| 	permissions_for_user_t* work = user_data; | ||||
| 	JSContext* context = work->context; | ||||
| 	JSValue result = JS_UNDEFINED; | ||||
| 	if (work->settings) | ||||
| 	{ | ||||
| 		JSValue json = JS_ParseJSON(context, work->settings, strlen(work->settings), NULL); | ||||
| 		if (JS_IsObject(json)) | ||||
| 		{ | ||||
| 			JSValue permissions = JS_GetPropertyStr(context, json, "permissions"); | ||||
| 			if (JS_IsObject(permissions)) | ||||
| 			{ | ||||
| 				result = JS_GetPropertyStr(context, permissions, work->user); | ||||
| 			} | ||||
| 			JS_FreeValue(context, permissions); | ||||
| 		} | ||||
| 		JS_FreeValue(context, json); | ||||
| 		tf_free((void*)work->settings); | ||||
| 	} | ||||
| 	if (JS_IsUndefined(result)) | ||||
| 	{ | ||||
| 		result = JS_NewArray(context); | ||||
| 	} | ||||
| 	JSValue error = JS_Call(context, JS_IsArray(context, result) ? work->promise[0] : work->promise[1], JS_UNDEFINED, 1, &result); | ||||
| 	tf_util_report_error(context, error); | ||||
| 	JS_FreeValue(context, error); | ||||
| 	JS_FreeValue(context, result); | ||||
| 	JS_FreeValue(context, work->promise[0]); | ||||
| 	JS_FreeValue(context, work->promise[1]); | ||||
| 	JS_FreeCString(context, work->user); | ||||
| 	tf_free(work); | ||||
| } | ||||
|  | ||||
| static JSValue _tf_api_core_permissionsForUser(JSContext* context, JSValueConst this_val, int argc, JSValueConst* argv, int magic, JSValue* data) | ||||
| { | ||||
| 	tf_task_t* task = tf_task_get(context); | ||||
| 	tf_ssb_t* ssb = tf_task_get_ssb(task); | ||||
| 	permissions_for_user_t* work = tf_malloc(sizeof(permissions_for_user_t)); | ||||
| 	*work = (permissions_for_user_t) { | ||||
| 		.context = context, | ||||
| 		.user = JS_ToCString(context, argv[0]), | ||||
| 	}; | ||||
| 	JSValue result = JS_NewPromiseCapability(context, work->promise); | ||||
| 	tf_ssb_run_work(ssb, _tf_api_core_permissions_for_user_work, _tf_api_core_permissions_for_user_after_work, work); | ||||
| 	return result; | ||||
| } | ||||
|  | ||||
| typedef struct _permissions_granted_t | ||||
| { | ||||
| 	JSContext* context; | ||||
| 	const char* user; | ||||
| 	const char* package_owner; | ||||
| 	const char* package_name; | ||||
| 	const char* settings; | ||||
| 	JSValue promise[2]; | ||||
| } permissions_granted_t; | ||||
|  | ||||
| static void _tf_api_core_permissions_granted_work(tf_ssb_t* ssb, void* user_data) | ||||
| { | ||||
| 	permissions_granted_t* work = user_data; | ||||
| 	work->settings = tf_ssb_db_get_property(ssb, "core", "settings"); | ||||
| } | ||||
|  | ||||
| static void _tf_api_core_permissions_granted_after_work(tf_ssb_t* ssb, int status, void* user_data) | ||||
| { | ||||
| 	permissions_granted_t* work = user_data; | ||||
| 	JSContext* context = work->context; | ||||
| 	JSValue result = JS_UNDEFINED; | ||||
| 	if (work->settings) | ||||
| 	{ | ||||
| 		JSValue json = JS_ParseJSON(context, work->settings, strlen(work->settings), NULL); | ||||
| 		if (JS_IsObject(json) && | ||||
| 			work->user && | ||||
| 			work->package_owner && | ||||
| 			work->package_name) | ||||
| 		{ | ||||
| 			JSValue user_permissions = JS_GetPropertyStr(context, json, "userPermissions"); | ||||
| 			if (JS_IsObject(user_permissions)) | ||||
| 			{ | ||||
| 				JSValue user = JS_GetPropertyStr(context, user_permissions, work->user); | ||||
| 				if (JS_IsObject(user)) | ||||
| 				{ | ||||
| 					JSValue package_owner = JS_GetPropertyStr(context, user, work->package_owner); | ||||
| 					if (JS_IsObject(package_owner)) | ||||
| 					{ | ||||
| 						result = JS_GetPropertyStr(context, package_owner, work->package_name); | ||||
| 					} | ||||
| 					JS_FreeValue(context, package_owner); | ||||
| 				} | ||||
| 				JS_FreeValue(context, user); | ||||
| 			} | ||||
| 			JS_FreeValue(context, user_permissions); | ||||
| 		} | ||||
| 		JS_FreeValue(context, json); | ||||
| 		tf_free((void*)work->settings); | ||||
| 	} | ||||
|  | ||||
| 	JSValue error = JS_Call(context, work->promise[0], JS_UNDEFINED, 1, &result); | ||||
| 	tf_util_report_error(context, error); | ||||
| 	JS_FreeValue(context, error); | ||||
| 	JS_FreeValue(context, result); | ||||
| 	JS_FreeValue(context, work->promise[0]); | ||||
| 	JS_FreeValue(context, work->promise[1]); | ||||
| 	tf_free((void*)work->user); | ||||
| 	tf_free((void*)work->package_owner); | ||||
| 	tf_free((void*)work->package_name); | ||||
| 	tf_free(work); | ||||
| } | ||||
|  | ||||
| static const char* _tf_ssb_get_process_credentials_session_name(JSContext* context, JSValue process) | ||||
| { | ||||
| 	JSValue credentials = JS_IsObject(process) ? JS_GetPropertyStr(context, process, "credentials") : JS_UNDEFINED; | ||||
| 	JSValue session = JS_IsObject(credentials) ? JS_GetPropertyStr(context, credentials, "session") : JS_UNDEFINED; | ||||
| 	JSValue name_value = JS_IsObject(session) ? JS_GetPropertyStr(context, session, "name") : JS_UNDEFINED; | ||||
| 	const char* name = JS_IsString(name_value) ? JS_ToCString(context, name_value) : NULL; | ||||
| 	const char* result = tf_strdup(name); | ||||
| 	JS_FreeCString(context, name); | ||||
| 	JS_FreeValue(context, name_value); | ||||
| 	JS_FreeValue(context, session); | ||||
| 	JS_FreeValue(context, credentials); | ||||
| 	return result; | ||||
| } | ||||
|  | ||||
| static JSValue _tf_api_core_permissionsGranted(JSContext* context, JSValueConst this_val, int argc, JSValueConst* argv, int magic, JSValue* data) | ||||
| { | ||||
| 	tf_task_t* task = tf_task_get(context); | ||||
| 	tf_ssb_t* ssb = tf_task_get_ssb(task); | ||||
| 	JSValue process = data[0]; | ||||
| 	JSValue package_owner_value = JS_GetPropertyStr(context, process, "packageOwner"); | ||||
| 	JSValue package_name_value = JS_GetPropertyStr(context, process, "packageName"); | ||||
| 	const char* package_owner = JS_ToCString(context, package_owner_value); | ||||
| 	const char* package_name = JS_ToCString(context, package_name_value); | ||||
| 	permissions_granted_t* work = tf_malloc(sizeof(permissions_granted_t)); | ||||
| 	*work = (permissions_granted_t) { | ||||
| 		.context = context, | ||||
| 		.user = _tf_ssb_get_process_credentials_session_name(context, process), | ||||
| 		.package_owner = tf_strdup(package_owner), | ||||
| 		.package_name = tf_strdup(package_name), | ||||
| 	}; | ||||
| 	JS_FreeCString(context, package_owner); | ||||
| 	JS_FreeCString(context, package_name); | ||||
| 	JS_FreeValue(context, package_owner_value); | ||||
| 	JS_FreeValue(context, package_name_value); | ||||
| 	JSValue result = JS_NewPromiseCapability(context, work->promise); | ||||
| 	tf_ssb_run_work(ssb, _tf_api_core_permissions_granted_work, _tf_api_core_permissions_granted_after_work, work); | ||||
| 	return result; | ||||
| } | ||||
|  | ||||
| typedef struct _active_identity_work_t | ||||
| { | ||||
| 	JSContext* context; | ||||
| 	const char* name; | ||||
| 	const char* package_owner; | ||||
| 	const char* package_name; | ||||
| 	char identity[k_id_base64_len]; | ||||
| 	int result; | ||||
| 	JSValue promise[2]; | ||||
| } active_identity_work_t; | ||||
|  | ||||
| static void _tf_ssb_getActiveIdentity_visit(const char* identity, void* user_data) | ||||
| { | ||||
| 	active_identity_work_t* request = user_data; | ||||
| 	if (!*request->identity) | ||||
| 	{ | ||||
| 		snprintf(request->identity, sizeof(request->identity), "@%s", identity); | ||||
| 	} | ||||
| } | ||||
|  | ||||
| static void _tf_ssb_getActiveIdentity_work(tf_ssb_t* ssb, void* user_data) | ||||
| { | ||||
| 	active_identity_work_t* request = user_data; | ||||
| 	sqlite3* db = tf_ssb_acquire_db_reader(ssb); | ||||
| 	tf_ssb_db_identity_get_active(db, request->name, request->package_owner, request->package_name, request->identity, sizeof(request->identity)); | ||||
| 	tf_ssb_release_db_reader(ssb, db); | ||||
|  | ||||
| 	if (!*request->identity) | ||||
| 	{ | ||||
| 		tf_ssb_db_identity_visit(ssb, request->name, _tf_ssb_getActiveIdentity_visit, request); | ||||
| 	} | ||||
|  | ||||
| 	if (!*request->identity && tf_ssb_db_user_has_permission(ssb, NULL, request->name, "administration")) | ||||
| 	{ | ||||
| 		tf_ssb_whoami(ssb, request->identity, sizeof(request->identity)); | ||||
| 	} | ||||
| } | ||||
|  | ||||
| static void _tf_ssb_getActiveIdentity_after_work(tf_ssb_t* ssb, int status, void* user_data) | ||||
| { | ||||
| 	active_identity_work_t* request = user_data; | ||||
| 	JSContext* context = request->context; | ||||
| 	if (request->result == 0) | ||||
| 	{ | ||||
| 		JSValue identity = JS_NewString(context, request->identity); | ||||
| 		JSValue error = JS_Call(context, request->promise[0], JS_UNDEFINED, 1, &identity); | ||||
| 		JS_FreeValue(context, identity); | ||||
| 		tf_util_report_error(context, error); | ||||
| 		JS_FreeValue(context, error); | ||||
| 	} | ||||
| 	else | ||||
| 	{ | ||||
| 		JSValue error = JS_Call(context, request->promise[1], JS_UNDEFINED, 0, NULL); | ||||
| 		tf_util_report_error(context, error); | ||||
| 		JS_FreeValue(context, error); | ||||
| 	} | ||||
| 	JS_FreeValue(context, request->promise[0]); | ||||
| 	JS_FreeValue(context, request->promise[1]); | ||||
| 	tf_free((void*)request->name); | ||||
| 	tf_free((void*)request->package_owner); | ||||
| 	tf_free((void*)request->package_name); | ||||
| 	tf_free(request); | ||||
| } | ||||
|  | ||||
| static JSValue _tf_ssb_getActiveIdentity(JSContext* context, JSValueConst this_val, int argc, JSValueConst* argv, int magic, JSValue* data) | ||||
| { | ||||
| 	tf_task_t* task = tf_task_get(context); | ||||
| 	tf_ssb_t* ssb = tf_task_get_ssb(task); | ||||
| 	JSValue process = data[0]; | ||||
| 	JSValue package_owner_value = JS_GetPropertyStr(context, process, "packageOwner"); | ||||
| 	JSValue package_name_value = JS_GetPropertyStr(context, process, "packageName"); | ||||
|  | ||||
| 	const char* name = _tf_ssb_get_process_credentials_session_name(context, process); | ||||
| 	const char* package_owner = JS_ToCString(context, package_owner_value); | ||||
| 	const char* package_name = JS_ToCString(context, package_name_value); | ||||
| 	active_identity_work_t* work = tf_malloc(sizeof(active_identity_work_t)); | ||||
| 	*work = (active_identity_work_t) { | ||||
| 		.context = context, | ||||
| 		.name = tf_strdup(name), | ||||
| 		.package_owner = tf_strdup(package_owner), | ||||
| 		.package_name = tf_strdup(package_name), | ||||
| 	}; | ||||
| 	JSValue result = JS_NewPromiseCapability(context, work->promise); | ||||
| 	tf_free((void*)name); | ||||
| 	JS_FreeCString(context, package_owner); | ||||
| 	JS_FreeCString(context, package_name); | ||||
|  | ||||
| 	JS_FreeValue(context, package_owner_value); | ||||
| 	JS_FreeValue(context, package_name_value); | ||||
| 	tf_ssb_run_work(ssb, _tf_ssb_getActiveIdentity_work, _tf_ssb_getActiveIdentity_after_work, work); | ||||
| 	return result; | ||||
| } | ||||
|  | ||||
| typedef struct _identities_visit_t | ||||
| { | ||||
| 	JSContext* context; | ||||
| 	JSValue promise[2]; | ||||
| 	const char** identities; | ||||
| 	int count; | ||||
| 	char user[]; | ||||
| } identities_visit_t; | ||||
|  | ||||
| static void _tf_ssb_getIdentities_visit(const char* identity, void* user_data) | ||||
| { | ||||
| 	identities_visit_t* work = user_data; | ||||
| 	work->identities = tf_resize_vec(work->identities, (work->count + 1) * sizeof(const char*)); | ||||
| 	char id[k_id_base64_len]; | ||||
| 	snprintf(id, sizeof(id), "@%s", identity); | ||||
| 	work->identities[work->count++] = tf_strdup(id); | ||||
| } | ||||
|  | ||||
| static void _tf_ssb_get_all_identities_work(tf_ssb_t* ssb, void* user_data) | ||||
| { | ||||
| 	tf_ssb_db_identity_visit_all(ssb, _tf_ssb_getIdentities_visit, user_data); | ||||
| } | ||||
|  | ||||
| static void _tf_ssb_get_identities_after_work(tf_ssb_t* ssb, int status, void* user_data) | ||||
| { | ||||
| 	identities_visit_t* work = user_data; | ||||
| 	JSContext* context = tf_ssb_get_context(ssb); | ||||
| 	JSValue result = JS_NewArray(context); | ||||
| 	for (int i = 0; i < work->count; i++) | ||||
| 	{ | ||||
| 		JS_SetPropertyUint32(context, result, i, JS_NewString(context, work->identities[i])); | ||||
| 		tf_free((void*)work->identities[i]); | ||||
| 	} | ||||
| 	tf_free(work->identities); | ||||
|  | ||||
| 	JSValue error = JS_Call(context, work->promise[0], JS_UNDEFINED, 1, &result); | ||||
| 	JS_FreeValue(context, result); | ||||
| 	tf_util_report_error(context, error); | ||||
| 	JS_FreeValue(context, error); | ||||
| 	JS_FreeValue(context, work->promise[0]); | ||||
| 	JS_FreeValue(context, work->promise[1]); | ||||
| 	tf_free(work); | ||||
| } | ||||
|  | ||||
| static JSValue _tf_ssb_getAllIdentities(JSContext* context, JSValueConst this_val, int argc, JSValueConst* argv) | ||||
| { | ||||
| 	JSValue result = JS_UNDEFINED; | ||||
| 	tf_task_t* task = tf_task_get(context); | ||||
| 	tf_ssb_t* ssb = tf_task_get_ssb(task); | ||||
| 	if (ssb) | ||||
| 	{ | ||||
| 		identities_visit_t* work = tf_malloc(sizeof(identities_visit_t)); | ||||
| 		*work = (identities_visit_t) { | ||||
| 			.context = context, | ||||
| 		}; | ||||
|  | ||||
| 		result = JS_NewPromiseCapability(context, work->promise); | ||||
| 		tf_ssb_run_work(ssb, _tf_ssb_get_all_identities_work, _tf_ssb_get_identities_after_work, work); | ||||
| 	} | ||||
| 	return result; | ||||
| } | ||||
|  | ||||
| static void _tf_ssb_get_identities_work(tf_ssb_t* ssb, void* user_data) | ||||
| { | ||||
| 	identities_visit_t* work = user_data; | ||||
| 	if (tf_ssb_db_user_has_permission(ssb, NULL, work->user, "administration")) | ||||
| 	{ | ||||
| 		char id[k_id_base64_len] = ""; | ||||
| 		if (tf_ssb_whoami(ssb, id, sizeof(id))) | ||||
| 		{ | ||||
| 			_tf_ssb_getIdentities_visit(*id == '@' ? id + 1 : id, work); | ||||
| 		} | ||||
| 	} | ||||
| 	tf_ssb_db_identity_visit(ssb, work->user, _tf_ssb_getIdentities_visit, user_data); | ||||
| } | ||||
|  | ||||
| static JSValue _tf_ssb_getIdentities(JSContext* context, JSValueConst this_val, int argc, JSValueConst* argv, int magic, JSValue* data) | ||||
| { | ||||
| 	JSValue result = JS_UNDEFINED; | ||||
| 	tf_task_t* task = tf_task_get(context); | ||||
| 	tf_ssb_t* ssb = tf_task_get_ssb(task); | ||||
| 	JSValue process = data[0]; | ||||
| 	if (ssb) | ||||
| 	{ | ||||
| 		const char* user = _tf_ssb_get_process_credentials_session_name(context, process); | ||||
| 		if (user) | ||||
| 		{ | ||||
| 			size_t user_length = user ? strlen(user) : 0; | ||||
| 			identities_visit_t* work = tf_malloc(sizeof(identities_visit_t) + user_length + 1); | ||||
| 			*work = (identities_visit_t) { | ||||
| 				.context = context, | ||||
| 			}; | ||||
| 			memcpy(work->user, user, user_length + 1); | ||||
| 			tf_free((void*)user); | ||||
|  | ||||
| 			result = JS_NewPromiseCapability(context, work->promise); | ||||
| 			tf_ssb_run_work(ssb, _tf_ssb_get_identities_work, _tf_ssb_get_identities_after_work, work); | ||||
| 		} | ||||
| 	} | ||||
| 	return result; | ||||
| } | ||||
|  | ||||
| static JSValue _tf_ssb_getOwnerIdentities(JSContext* context, JSValueConst this_val, int argc, JSValueConst* argv, int magic, JSValue* data) | ||||
| { | ||||
| 	JSValue result = JS_UNDEFINED; | ||||
| 	tf_task_t* task = tf_task_get(context); | ||||
| 	tf_ssb_t* ssb = tf_task_get_ssb(task); | ||||
| 	JSValue process = data[0]; | ||||
| 	if (ssb) | ||||
| 	{ | ||||
| 		JSValue value = JS_GetPropertyStr(context, process, "packageOwner"); | ||||
| 		const char* user = JS_ToCString(context, value); | ||||
| 		size_t user_length = user ? strlen(user) : 0; | ||||
| 		identities_visit_t* work = tf_malloc(sizeof(identities_visit_t) + user_length + 1); | ||||
| 		*work = (identities_visit_t) { | ||||
| 			.context = context, | ||||
| 		}; | ||||
| 		memcpy(work->user, user, user_length + 1); | ||||
| 		JS_FreeCString(context, user); | ||||
| 		JS_FreeValue(context, value); | ||||
|  | ||||
| 		result = JS_NewPromiseCapability(context, work->promise); | ||||
| 		tf_ssb_run_work(ssb, _tf_ssb_get_identities_work, _tf_ssb_get_identities_after_work, work); | ||||
| 	} | ||||
| 	return result; | ||||
| } | ||||
|  | ||||
| typedef struct _settings_descriptions_get_t | ||||
| { | ||||
| 	const char* settings; | ||||
| 	JSContext* context; | ||||
| 	JSValue promise[2]; | ||||
| } settings_descriptions_get_t; | ||||
|  | ||||
| static void _tf_ssb_get_settings_descriptions_work(tf_ssb_t* ssb, void* user_data) | ||||
| { | ||||
| 	settings_descriptions_get_t* work = user_data; | ||||
| 	work->settings = tf_ssb_db_get_property(ssb, "core", "settings"); | ||||
| } | ||||
|  | ||||
| static void _tf_ssb_get_settings_descriptions_after_work(tf_ssb_t* ssb, int status, void* user_data) | ||||
| { | ||||
| 	settings_descriptions_get_t* work = user_data; | ||||
| 	JSContext* context = work->context; | ||||
| 	JSValue result = JS_NewObject(context); | ||||
| 	JSValue settings = JS_ParseJSON(context, work->settings ? work->settings : "", work->settings ? strlen(work->settings) : 0, NULL); | ||||
| 	const char* name; | ||||
| 	const char* type; | ||||
| 	tf_setting_kind_t kind; | ||||
| 	const char* description; | ||||
| 	for (int i = 0; tf_util_get_global_setting_by_index(i, &name, &type, &kind, &description); i++) | ||||
| 	{ | ||||
| 		JSValue entry = JS_NewObject(context); | ||||
| 		JS_SetPropertyStr(context, entry, "type", JS_NewString(context, type)); | ||||
| 		JS_SetPropertyStr(context, entry, "description", JS_NewString(context, description)); | ||||
| 		switch (kind) | ||||
| 		{ | ||||
| 		case k_kind_unknown: | ||||
| 			break; | ||||
| 		case k_kind_bool: | ||||
| 			JS_SetPropertyStr(context, entry, "default_value", JS_NewBool(context, tf_util_get_default_global_setting_bool(name))); | ||||
| 			break; | ||||
| 		case k_kind_int: | ||||
| 			JS_SetPropertyStr(context, entry, "default_value", JS_NewInt32(context, tf_util_get_default_global_setting_int(name))); | ||||
| 			break; | ||||
| 		case k_kind_string: | ||||
| 			JS_SetPropertyStr(context, entry, "default_value", JS_NewString(context, tf_util_get_default_global_setting_string(name))); | ||||
| 			break; | ||||
| 		} | ||||
| 		if (JS_IsObject(settings)) | ||||
| 		{ | ||||
| 			JS_SetPropertyStr(context, entry, "value", JS_GetPropertyStr(context, settings, name)); | ||||
| 		} | ||||
| 		JS_SetPropertyStr(context, result, name, entry); | ||||
| 	} | ||||
| 	JS_FreeValue(context, settings); | ||||
| 	JSValue error = JS_Call(context, work->promise[0], JS_UNDEFINED, 1, &result); | ||||
| 	tf_util_report_error(context, error); | ||||
| 	JS_FreeValue(context, error); | ||||
| 	JS_FreeValue(context, result); | ||||
| 	JS_FreeValue(context, work->promise[0]); | ||||
| 	JS_FreeValue(context, work->promise[1]); | ||||
| 	tf_free((void*)work->settings); | ||||
| 	tf_free(work); | ||||
| } | ||||
|  | ||||
| static JSValue _tf_ssb_globalSettingsDescriptions(JSContext* context, JSValueConst this_val, int argc, JSValueConst* argv) | ||||
| { | ||||
| 	tf_task_t* task = tf_task_get(context); | ||||
| 	tf_ssb_t* ssb = tf_task_get_ssb(task); | ||||
| 	settings_descriptions_get_t* work = tf_malloc(sizeof(settings_descriptions_get_t)); | ||||
| 	*work = (settings_descriptions_get_t) { .context = context }; | ||||
| 	JSValue result = JS_NewPromiseCapability(context, work->promise); | ||||
| 	tf_ssb_run_work(ssb, _tf_ssb_get_settings_descriptions_work, _tf_ssb_get_settings_descriptions_after_work, work); | ||||
| 	return result; | ||||
| } | ||||
|  | ||||
| typedef struct _settings_get_t | ||||
| { | ||||
| 	const char* key; | ||||
| 	tf_setting_kind_t kind; | ||||
| 	void* value; | ||||
| 	JSContext* context; | ||||
| 	JSValue promise[2]; | ||||
| } settings_get_t; | ||||
|  | ||||
| static void _tf_ssb_settings_get_work(tf_ssb_t* ssb, void* user_data) | ||||
| { | ||||
| 	settings_get_t* work = user_data; | ||||
| 	work->kind = tf_util_get_global_setting_kind(work->key); | ||||
| 	switch (work->kind) | ||||
| 	{ | ||||
| 	case k_kind_unknown: | ||||
| 		break; | ||||
| 	case k_kind_bool: | ||||
| 		{ | ||||
| 			sqlite3* db = tf_ssb_acquire_db_reader(ssb); | ||||
| 			bool value = false; | ||||
| 			tf_ssb_db_get_global_setting_bool(db, work->key, &value); | ||||
| 			work->value = (void*)(intptr_t)value; | ||||
| 			tf_ssb_release_db_reader(ssb, db); | ||||
| 		} | ||||
| 		break; | ||||
| 	case k_kind_int: | ||||
| 		{ | ||||
| 			sqlite3* db = tf_ssb_acquire_db_reader(ssb); | ||||
| 			int64_t value = 0; | ||||
| 			tf_ssb_db_get_global_setting_int64(db, work->key, &value); | ||||
| 			work->value = (void*)(intptr_t)value; | ||||
| 			tf_ssb_release_db_reader(ssb, db); | ||||
| 		} | ||||
| 		break; | ||||
| 	case k_kind_string: | ||||
| 		{ | ||||
| 			sqlite3* db = tf_ssb_acquire_db_reader(ssb); | ||||
| 			work->value = (void*)tf_ssb_db_get_global_setting_string_alloc(db, work->key); | ||||
| 			tf_ssb_release_db_reader(ssb, db); | ||||
| 		} | ||||
| 		break; | ||||
| 	} | ||||
| } | ||||
|  | ||||
| static void _tf_ssb_settings_get_after_work(tf_ssb_t* ssb, int status, void* user_data) | ||||
| { | ||||
| 	settings_get_t* work = user_data; | ||||
| 	JSContext* context = tf_ssb_get_context(ssb); | ||||
| 	JSValue result = JS_UNDEFINED; | ||||
| 	switch (work->kind) | ||||
| 	{ | ||||
| 	case k_kind_unknown: | ||||
| 		break; | ||||
| 	case k_kind_bool: | ||||
| 		result = work->value ? JS_TRUE : JS_FALSE; | ||||
| 		break; | ||||
| 	case k_kind_int: | ||||
| 		result = JS_NewInt64(context, (int64_t)(intptr_t)work->value); | ||||
| 		break; | ||||
| 	case k_kind_string: | ||||
| 		result = JS_NewString(context, (const char*)work->value); | ||||
| 		tf_free(work->value); | ||||
| 		break; | ||||
| 	} | ||||
|  | ||||
| 	JSValue error = JS_Call(context, work->promise[0], JS_UNDEFINED, 1, &result); | ||||
| 	tf_util_report_error(context, error); | ||||
| 	JS_FreeValue(context, error); | ||||
| 	JS_FreeValue(context, result); | ||||
| 	JS_FreeValue(context, work->promise[0]); | ||||
| 	JS_FreeValue(context, work->promise[1]); | ||||
| 	JS_FreeCString(context, work->key); | ||||
| 	tf_free(work); | ||||
| } | ||||
|  | ||||
| static JSValue _tf_ssb_globalSettingsGet(JSContext* context, JSValueConst this_val, int argc, JSValueConst* argv) | ||||
| { | ||||
| 	tf_task_t* task = tf_task_get(context); | ||||
| 	tf_ssb_t* ssb = tf_task_get_ssb(task); | ||||
| 	settings_get_t* work = tf_malloc(sizeof(settings_get_t)); | ||||
| 	*work = (settings_get_t) { .context = context, .key = JS_ToCString(context, argv[0]) }; | ||||
| 	JSValue result = JS_NewPromiseCapability(context, work->promise); | ||||
| 	tf_ssb_run_work(ssb, _tf_ssb_settings_get_work, _tf_ssb_settings_get_after_work, work); | ||||
| 	return result; | ||||
| } | ||||
|  | ||||
| static JSValue _tf_api_register_imports(JSContext* context, JSValueConst this_val, int argc, JSValueConst* argv) | ||||
| { | ||||
| 	JSValue imports = argv[0]; | ||||
| @@ -218,6 +807,38 @@ static JSValue _tf_api_register_imports(JSContext* context, JSValueConst this_va | ||||
| 	JS_SetPropertyStr(context, core, "apps", JS_NewCFunctionData(context, _tf_api_core_apps, 1, 0, 1, &process)); | ||||
| 	JS_SetPropertyStr(context, core, "register", JS_NewCFunctionData(context, _tf_api_core_register, 2, 0, 1, &process)); | ||||
| 	JS_SetPropertyStr(context, core, "unregister", JS_NewCFunctionData(context, _tf_api_core_unregister, 2, 0, 1, &process)); | ||||
|  | ||||
| 	JS_SetPropertyStr(context, core, "users", JS_NewCFunctionData(context, _tf_api_core_users, 0, 0, 1, &process)); | ||||
|  | ||||
| 	JS_SetPropertyStr(context, core, "permissionsForUser", JS_NewCFunctionData(context, _tf_api_core_permissionsForUser, 1, 0, 1, &process)); | ||||
| 	JS_SetPropertyStr(context, core, "permissionsGranted", JS_NewCFunctionData(context, _tf_api_core_permissionsGranted, 0, 0, 1, &process)); | ||||
|  | ||||
| 	JSValue app = JS_NewObject(context); | ||||
| 	JS_SetPropertyStr(context, app, "owner", JS_GetPropertyStr(context, process, "packageOwner")); | ||||
| 	JS_SetPropertyStr(context, app, "name", JS_GetPropertyStr(context, process, "packageName")); | ||||
| 	JS_SetPropertyStr(context, core, "app", app); | ||||
|  | ||||
| 	JS_SetPropertyStr(context, core, "url", JS_GetPropertyStr(context, process, "url")); | ||||
|  | ||||
| 	JSValue ssb = JS_GetPropertyStr(context, imports, "ssb"); | ||||
| 	JS_SetPropertyStr(context, ssb, "getAllIdentities", JS_NewCFunction(context, _tf_ssb_getAllIdentities, "getAllIdentities", 0)); | ||||
| 	JS_SetPropertyStr(context, ssb, "getActiveIdentity", JS_NewCFunctionData(context, _tf_ssb_getActiveIdentity, 0, 0, 1, &process)); | ||||
| 	JS_SetPropertyStr(context, ssb, "getIdentities", JS_NewCFunctionData(context, _tf_ssb_getIdentities, 0, 0, 1, &process)); | ||||
| 	JS_SetPropertyStr(context, ssb, "getOwnerIdentities", JS_NewCFunctionData(context, _tf_ssb_getOwnerIdentities, 0, 0, 1, &process)); | ||||
| 	JS_FreeValue(context, ssb); | ||||
|  | ||||
| 	JSValue credentials = JS_GetPropertyStr(context, process, "credentials"); | ||||
| 	JSValue permissions = JS_IsObject(credentials) ? JS_GetPropertyStr(context, credentials, "permissions") : JS_UNDEFINED; | ||||
| 	JSValue administration = JS_IsObject(permissions) ? JS_GetPropertyStr(context, permissions, "administration") : JS_UNDEFINED; | ||||
| 	if (JS_ToBool(context, administration) > 0) | ||||
| 	{ | ||||
| 		JS_SetPropertyStr(context, core, "globalSettingsDescriptions", JS_NewCFunction(context, _tf_ssb_globalSettingsDescriptions, "globalSettingsDescriptions", 0)); | ||||
| 		JS_SetPropertyStr(context, core, "globalSettingsGet", JS_NewCFunction(context, _tf_ssb_globalSettingsGet, "globalSettingsGet", 1)); | ||||
| 	} | ||||
| 	JS_FreeValue(context, administration); | ||||
| 	JS_FreeValue(context, permissions); | ||||
| 	JS_FreeValue(context, credentials); | ||||
|  | ||||
| 	JS_FreeValue(context, core); | ||||
| 	return JS_UNDEFINED; | ||||
| } | ||||
|   | ||||
| @@ -1,42 +0,0 @@ | ||||
| #include "bcrypt.js.h" | ||||
|  | ||||
| #include "task.h" | ||||
|  | ||||
| #include "ow-crypt.h" | ||||
| #include "quickjs.h" | ||||
| #include "uv.h" | ||||
|  | ||||
| static JSValue _crypt_hashpw(JSContext* context, JSValueConst this_val, int argc, JSValueConst* argv) | ||||
| { | ||||
| 	const char* key = JS_ToCString(context, argv[0]); | ||||
| 	const char* salt = JS_ToCString(context, argv[1]); | ||||
| 	char output[7 + 22 + 31 + 1]; | ||||
| 	char* hash = crypt_rn(key, salt, output, sizeof(output)); | ||||
| 	JSValue result = JS_NewString(context, hash); | ||||
| 	JS_FreeCString(context, key); | ||||
| 	JS_FreeCString(context, salt); | ||||
| 	return result; | ||||
| } | ||||
|  | ||||
| static JSValue _crypt_gensalt(JSContext* context, JSValueConst this_val, int argc, JSValueConst* argv) | ||||
| { | ||||
| 	int length = 0; | ||||
| 	JS_ToInt32(context, &length, argv[0]); | ||||
| 	char buffer[16]; | ||||
| 	tf_task_t* task = tf_task_get(context); | ||||
| 	size_t bytes = uv_random(tf_task_get_loop(task), &(uv_random_t) { 0 }, buffer, sizeof(buffer), 0, NULL) == 0 ? sizeof(buffer) : 0; | ||||
| 	char output[7 + 22 + 1]; | ||||
| 	char* salt = crypt_gensalt_rn("$2b$", length, buffer, bytes, output, sizeof(output)); | ||||
| 	JSValue result = JS_NewString(context, salt); | ||||
| 	return result; | ||||
| } | ||||
|  | ||||
| void tf_bcrypt_register(JSContext* context) | ||||
| { | ||||
| 	JSValue global = JS_GetGlobalObject(context); | ||||
| 	JSValue bcrypt = JS_NewObject(context); | ||||
| 	JS_SetPropertyStr(context, global, "bCrypt", bcrypt); | ||||
| 	JS_SetPropertyStr(context, bcrypt, "hashpw", JS_NewCFunction(context, _crypt_hashpw, "hashpw", 2)); | ||||
| 	JS_SetPropertyStr(context, bcrypt, "gensalt", JS_NewCFunction(context, _crypt_gensalt, "gensalt", 1)); | ||||
| 	JS_FreeValue(context, global); | ||||
| } | ||||
| @@ -1,19 +0,0 @@ | ||||
| #pragma once | ||||
|  | ||||
| /** | ||||
| ** \defgroup bcrypt_js bCrypt | ||||
| ** Exposes bcrypt to script, where it is used for hashing and verifying | ||||
| ** passwords. | ||||
| ** @{ | ||||
| */ | ||||
|  | ||||
| /** A JS context. */ | ||||
| typedef struct JSContext JSContext; | ||||
|  | ||||
| /** | ||||
| ** Register the bcrypt script interface. | ||||
| ** @param context The JS context. | ||||
| */ | ||||
| void tf_bcrypt_register(JSContext* context); | ||||
|  | ||||
| /** @} */ | ||||
							
								
								
									
										140
									
								
								src/http.c
									
									
									
									
									
								
							
							
						
						
									
										140
									
								
								src/http.c
									
									
									
									
									
								
							| @@ -2,7 +2,6 @@ | ||||
|  | ||||
| #include "log.h" | ||||
| #include "mem.h" | ||||
| #include "tls.h" | ||||
| #include "trace.h" | ||||
| #include "util.js.h" | ||||
|  | ||||
| @@ -20,10 +19,12 @@ | ||||
|  | ||||
| static const int k_timeout_ms = 60000; | ||||
|  | ||||
| static tf_http_t** s_http_instances; | ||||
| int s_http_instance_count; | ||||
|  | ||||
| typedef struct _tf_http_connection_t | ||||
| { | ||||
| 	tf_http_t* http; | ||||
| 	tf_tls_session_t* tls; | ||||
| 	uv_tcp_t tcp; | ||||
| 	uv_shutdown_t shutdown; | ||||
| 	uv_timer_t timeout; | ||||
| @@ -75,7 +76,6 @@ typedef struct _tf_http_handler_t | ||||
| typedef struct _tf_http_listener_t | ||||
| { | ||||
| 	tf_http_t* http; | ||||
| 	tf_tls_context_t* tls; | ||||
| 	uv_tcp_t tcp; | ||||
| 	tf_http_cleanup_t* cleanup; | ||||
| 	void* user_data; | ||||
| @@ -106,7 +106,6 @@ typedef struct _tf_http_t | ||||
| static const char* _http_connection_get_header(const tf_http_connection_t* connection, const char* name); | ||||
| static void _http_connection_destroy(tf_http_connection_t* connection, const char* reason); | ||||
| static void _http_timer_reset(tf_http_connection_t* connection); | ||||
| static void _http_tls_update(tf_http_connection_t* connection); | ||||
| static void _http_builtin_404_handler(tf_http_request_t* request); | ||||
|  | ||||
| tf_http_t* tf_http_create(uv_loop_t* loop) | ||||
| @@ -115,6 +114,8 @@ tf_http_t* tf_http_create(uv_loop_t* loop) | ||||
| 	*http = (tf_http_t) { | ||||
| 		.loop = loop, | ||||
| 	}; | ||||
| 	s_http_instances = tf_resize_vec(s_http_instances, sizeof(tf_http_t*) * (s_http_instance_count + 1)); | ||||
| 	s_http_instances[s_http_instance_count++] = http; | ||||
| 	return http; | ||||
| } | ||||
|  | ||||
| @@ -255,11 +256,6 @@ static void _http_connection_destroy(tf_http_connection_t* connection, const cha | ||||
| 	{ | ||||
| 		uv_close((uv_handle_t*)&connection->timeout, _http_connection_on_close); | ||||
| 	} | ||||
| 	if (connection->tls) | ||||
| 	{ | ||||
| 		tf_tls_session_destroy(connection->tls); | ||||
| 		connection->tls = NULL; | ||||
| 	} | ||||
|  | ||||
| 	if (connection->ref_count == 0 && !connection->tcp.data && !connection->shutdown.data && !connection->timeout.data) | ||||
| 	{ | ||||
| @@ -441,7 +437,6 @@ static void _http_add_body_bytes(tf_http_connection_t* connection, const void* d | ||||
| 			*request = (tf_http_request_t) { | ||||
| 				.http = connection->http, | ||||
| 				.connection = connection, | ||||
| 				.is_tls = connection->tls != NULL, | ||||
| 				.method = connection->method, | ||||
| 				.path = connection->path, | ||||
| 				.query = connection->query, | ||||
| @@ -581,23 +576,9 @@ static void _http_on_read(uv_stream_t* stream, ssize_t read_size, const uv_buf_t | ||||
| 	tf_http_connection_t* connection = stream->data; | ||||
| 	_http_timer_reset(connection); | ||||
| 	if (read_size > 0) | ||||
| 	{ | ||||
| 		if (connection->tls) | ||||
| 		{ | ||||
| 			if (tf_tls_session_write_encrypted(connection->tls, buffer->base, read_size) < 0) | ||||
| 			{ | ||||
| 				_http_connection_destroy(connection, "tf_tls_session_write_encrypted"); | ||||
| 			} | ||||
| 			else | ||||
| 			{ | ||||
| 				_http_tls_update(connection); | ||||
| 			} | ||||
| 		} | ||||
| 		else | ||||
| 	{ | ||||
| 		_http_on_read_plain(connection, buffer->base, read_size); | ||||
| 	} | ||||
| 	} | ||||
| 	else if (read_size < 0) | ||||
| 	{ | ||||
| 		_http_connection_destroy(connection, uv_strerror(read_size)); | ||||
| @@ -638,17 +619,6 @@ static void _http_on_connection(uv_stream_t* stream, int status) | ||||
|  | ||||
| 	tf_http_connection_t* connection = tf_malloc(sizeof(tf_http_connection_t)); | ||||
| 	*connection = (tf_http_connection_t) { .http = http, .tcp = { .data = connection }, .is_receiving_headers = true }; | ||||
| 	if (listener->tls) | ||||
| 	{ | ||||
| 		connection->tls = tf_tls_context_create_session(listener->tls); | ||||
| 		if (!connection->tls) | ||||
| 		{ | ||||
| 			_http_connection_destroy(connection, "tf_tls_context_create_session"); | ||||
| 			return; | ||||
| 		} | ||||
| 		tf_tls_session_start_accept(connection->tls); | ||||
| 		connection->is_handshaking = true; | ||||
| 	} | ||||
| 	int r = uv_tcp_init(connection->http->loop, &connection->tcp); | ||||
| 	if (r) | ||||
| 	{ | ||||
| @@ -689,21 +659,15 @@ static void _http_on_connection(uv_stream_t* stream, int status) | ||||
| 		return; | ||||
| 	} | ||||
|  | ||||
| 	if (connection->tls) | ||||
| 	{ | ||||
| 		_http_tls_update(connection); | ||||
| 	} | ||||
|  | ||||
| 	http->connections = tf_resize_vec(http->connections, sizeof(tf_http_connection_t*) * (http->connections_count + 1)); | ||||
| 	http->connections[http->connections_count++] = connection; | ||||
| } | ||||
|  | ||||
| int tf_http_listen(tf_http_t* http, int port, bool local_only, tf_tls_context_t* tls, tf_http_cleanup_t* cleanup, void* user_data) | ||||
| int tf_http_listen(tf_http_t* http, int port, bool local_only, tf_http_cleanup_t* cleanup, void* user_data) | ||||
| { | ||||
| 	tf_http_listener_t* listener = tf_malloc(sizeof(tf_http_listener_t)); | ||||
| 	*listener = (tf_http_listener_t) { | ||||
| 		.http = http, | ||||
| 		.tls = tls, | ||||
| 		.tcp = { .data = listener }, | ||||
| 		.cleanup = cleanup, | ||||
| 		.user_data = user_data, | ||||
| @@ -872,6 +836,21 @@ void tf_http_destroy(tf_http_t* http) | ||||
| 		http->handlers_count = 0; | ||||
|  | ||||
| 		tf_free(http); | ||||
|  | ||||
| 		for (int i = 0; i < s_http_instance_count; i++) | ||||
| 		{ | ||||
| 			if (s_http_instances[i] == http) | ||||
| 			{ | ||||
| 				s_http_instances[i] = s_http_instances[--s_http_instance_count]; | ||||
| 				break; | ||||
| 			} | ||||
| 		} | ||||
| 		if (s_http_instance_count == 0) | ||||
| 		{ | ||||
| 			tf_free(s_http_instances); | ||||
| 			s_http_instances = NULL; | ||||
| 		} | ||||
| 		tf_printf("http %p destroyed\n", http); | ||||
| 	} | ||||
| 	else | ||||
| 	{ | ||||
| @@ -928,72 +907,11 @@ static void _http_write_internal(tf_http_connection_t* connection, const void* d | ||||
| 	} | ||||
| } | ||||
|  | ||||
| static void _http_tls_update(tf_http_connection_t* connection) | ||||
| { | ||||
| 	bool again = true; | ||||
| 	while (again) | ||||
| 	{ | ||||
| 		again = false; | ||||
|  | ||||
| 		if (connection->is_handshaking && connection->tls) | ||||
| 		{ | ||||
| 			switch (tf_tls_session_handshake(connection->tls)) | ||||
| 			{ | ||||
| 			case k_tls_handshake_done: | ||||
| 				connection->is_handshaking = false; | ||||
| 				break; | ||||
| 			case k_tls_handshake_more: | ||||
| 				break; | ||||
| 			case k_tls_handshake_failed: | ||||
| 				_http_connection_destroy(connection, "tf_tls_session_handshake"); | ||||
| 				return; | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		/* Maybe we became disconnected and cleaned up our TLS session. */ | ||||
| 		if (connection->tls) | ||||
| 		{ | ||||
| 			char buffer[8192]; | ||||
| 			int r = tf_tls_session_read_encrypted(connection->tls, buffer, sizeof(buffer)); | ||||
| 			if (r > 0) | ||||
| 			{ | ||||
| 				_http_write_internal(connection, buffer, r); | ||||
| 				again = true; | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		if (connection->tls) | ||||
| 		{ | ||||
| 			char buffer[8192]; | ||||
| 			int r = tf_tls_session_read_plain(connection->tls, buffer, sizeof(buffer)); | ||||
| 			if (r > 0) | ||||
| 			{ | ||||
| 				_http_on_read_plain(connection, buffer, r); | ||||
| 				again = true; | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
|  | ||||
| static void _http_write(tf_http_connection_t* connection, const void* data, size_t size) | ||||
| { | ||||
| 	_http_timer_reset(connection); | ||||
| 	if (connection->tls) | ||||
| 	{ | ||||
| 		int r = tf_tls_session_write_plain(connection->tls, data, size); | ||||
| 		if (r < (ssize_t)size) | ||||
| 		{ | ||||
| 			char buffer[8192]; | ||||
| 			tf_tls_session_get_error(connection->tls, buffer, sizeof(buffer)); | ||||
| 			tf_printf("tf_tls_session_write_plain: %s\n", buffer); | ||||
| 		} | ||||
| 		_http_tls_update(connection); | ||||
| 	} | ||||
| 	else | ||||
| 	{ | ||||
| 	_http_write_internal(connection, data, size); | ||||
| } | ||||
| } | ||||
|  | ||||
| void tf_http_request_websocket_send(tf_http_request_t* request, int op_code, const void* data, size_t size) | ||||
| { | ||||
| @@ -1215,3 +1133,19 @@ const char* tf_http_get_cookie(const char* cookie_header, const char* name) | ||||
| 	} | ||||
| 	return NULL; | ||||
| } | ||||
|  | ||||
| void tf_http_debug_destroy() | ||||
| { | ||||
| 	for (int i = 0; i < s_http_instance_count; i++) | ||||
| 	{ | ||||
| 		tf_http_t* http = s_http_instances[i]; | ||||
| 		tf_printf("http %p[%d]\n", http, i); | ||||
| 		tf_printf("  connections = %d\n", http->connections_count); | ||||
| 		for (int j = 0; j < http->connections_count; j++) | ||||
| 		{ | ||||
| 			tf_http_connection_t* connection = http->connections[j]; | ||||
| 			tf_printf("    connection %p[%d] %s tcp=%p timeout=%p shutdown=%p rc=%d\n", connection, j, connection->trace_name, connection->tcp.data, connection->timeout.data, | ||||
| 				connection->shutdown.data, connection->ref_count); | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
|   | ||||
							
								
								
									
										13
									
								
								src/http.h
									
									
									
									
									
								
							
							
						
						
									
										13
									
								
								src/http.h
									
									
									
									
									
								
							| @@ -23,9 +23,6 @@ typedef struct _tf_http_request_t tf_http_request_t; | ||||
| /** An HTTP instance. */ | ||||
| typedef struct _tf_http_t tf_http_t; | ||||
|  | ||||
| /** A TLS context. */ | ||||
| typedef struct _tf_tls_context_t tf_tls_context_t; | ||||
|  | ||||
| /** A trace instance. */ | ||||
| typedef struct _tf_trace_t tf_trace_t; | ||||
|  | ||||
| @@ -68,8 +65,6 @@ typedef struct _tf_http_request_t | ||||
| 	tf_http_t* http; | ||||
| 	/** The HTTP connection associated with this request. */ | ||||
| 	tf_http_connection_t* connection; | ||||
| 	/** True if this is an HTTPS session. */ | ||||
| 	bool is_tls; | ||||
| 	/** The HTTP method of the request (GET/POST/...). */ | ||||
| 	const char* method; | ||||
| 	/** The HTTP request path. */ | ||||
| @@ -117,12 +112,11 @@ void tf_http_set_trace(tf_http_t* http, tf_trace_t* trace); | ||||
| ** @param http The HTTP instance. | ||||
| ** @param port The port on which to listen, or 0 to assign a free port. | ||||
| ** @param local_only Only access connections on localhost, otherwise any address. | ||||
| ** @param tls An optional TLS context to use for HTTPS requests. | ||||
| ** @param cleanup A function called when the HTTP instance is being cleaned up. | ||||
| ** @param user_data User data passed to the cleanup callback. | ||||
| ** @return The port number on which the HTTP instance is now listening. | ||||
| */ | ||||
| int tf_http_listen(tf_http_t* http, int port, bool local_only, tf_tls_context_t* tls, tf_http_cleanup_t* cleanup, void* user_data); | ||||
| int tf_http_listen(tf_http_t* http, int port, bool local_only, tf_http_cleanup_t* cleanup, void* user_data); | ||||
|  | ||||
| /** | ||||
| ** Add an HTTP request handler. | ||||
| @@ -238,4 +232,9 @@ const char* tf_http_status_text(int status); | ||||
| */ | ||||
| bool tf_http_pattern_matches(const char* pattern, const char* path); | ||||
|  | ||||
| /** | ||||
| ** Log debug information to diagnose shutdown problems. | ||||
| */ | ||||
| void tf_http_debug_destroy(); | ||||
|  | ||||
| /** @} */ | ||||
|   | ||||
| @@ -4,9 +4,9 @@ | ||||
| #include "http.h" | ||||
| #include "log.h" | ||||
| #include "mem.h" | ||||
| #include "sha1.h" | ||||
| #include "ssb.db.h" | ||||
| #include "task.h" | ||||
| #include "tls.h" | ||||
| #include "trace.h" | ||||
| #include "util.js.h" | ||||
| #include "version.h" | ||||
| @@ -14,8 +14,6 @@ | ||||
| #include "sodium/crypto_sign.h" | ||||
| #include "sodium/utils.h" | ||||
|  | ||||
| #include <openssl/sha.h> | ||||
|  | ||||
| #define CYAN "\e[1;36m" | ||||
| #define MAGENTA "\e[1;35m" | ||||
| #define YELLOW "\e[1;33m" | ||||
| @@ -169,8 +167,13 @@ static JSValue _httpd_websocket_upgrade(JSContext* context, JSValueConst this_va | ||||
| 		uint8_t* key_magic = alloca(size); | ||||
| 		memcpy(key_magic, header_sec_websocket_key, key_length); | ||||
| 		memcpy(key_magic + key_length, k_magic, 36); | ||||
|  | ||||
| 		uint8_t digest[20]; | ||||
| 		SHA1(key_magic, size, digest); | ||||
| 		SHA1_CTX sha1 = { 0 }; | ||||
| 		SHA1Init(&sha1); | ||||
| 		SHA1Update(&sha1, key_magic, size); | ||||
| 		SHA1Final(digest, &sha1); | ||||
|  | ||||
| 		char key[41] = { 0 }; | ||||
| 		tf_base64_encode(digest, sizeof(digest), key, sizeof(key)); | ||||
|  | ||||
| @@ -253,11 +256,6 @@ JSValue tf_httpd_make_response_object(JSContext* context, tf_http_request_t* req | ||||
|  | ||||
| bool tf_httpd_redirect(tf_http_request_t* request) | ||||
| { | ||||
| 	if (request->is_tls) | ||||
| 	{ | ||||
| 		return false; | ||||
| 	} | ||||
|  | ||||
| 	http_user_data_t* user_data = tf_http_get_user_data(request->http); | ||||
| 	if (!user_data || !*user_data->redirect) | ||||
| 	{ | ||||
| @@ -272,16 +270,12 @@ bool tf_httpd_redirect(tf_http_request_t* request) | ||||
|  | ||||
| typedef struct _httpd_listener_t | ||||
| { | ||||
| 	tf_tls_context_t* tls; | ||||
| 	int padding; | ||||
| } httpd_listener_t; | ||||
|  | ||||
| static void _httpd_listener_cleanup(void* user_data) | ||||
| { | ||||
| 	httpd_listener_t* listener = user_data; | ||||
| 	if (listener->tls) | ||||
| 	{ | ||||
| 		tf_tls_context_destroy(listener->tls); | ||||
| 	} | ||||
| 	tf_free(listener); | ||||
| } | ||||
|  | ||||
| @@ -570,7 +564,7 @@ static void _httpd_endpoint_add_slash(tf_http_request_t* request) | ||||
| 		host = tf_http_request_get_header(request, "host"); | ||||
| 	} | ||||
| 	char url[1024]; | ||||
| 	snprintf(url, sizeof(url), "%s%s%s/", request->is_tls ? "https://" : "http://", host, request->path); | ||||
| 	snprintf(url, sizeof(url), "%s%s%s/", "http://", host, request->path); | ||||
| 	const char* headers[] = { | ||||
| 		"Location", | ||||
| 		url, | ||||
| @@ -856,31 +850,6 @@ bool tf_httpd_is_name_valid(const char* name) | ||||
| 	return true; | ||||
| } | ||||
|  | ||||
| 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) | ||||
| { | ||||
| 	JS_NewClassID(&_httpd_request_class_id); | ||||
| @@ -909,36 +878,14 @@ tf_http_t* tf_httpd_create(JSContext* context) | ||||
| 	tf_http_set_trace(http, tf_task_get_trace(task)); | ||||
|  | ||||
| 	int64_t http_port = 0; | ||||
| 	int64_t https_port = 0; | ||||
| 	char out_http_port_file[512] = ""; | ||||
| 	bool local_only = false; | ||||
| 	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_db_get_global_setting_bool(db, "http_local_only", &local_only); | ||||
| 	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, "/codemirror/*", tf_httpd_endpoint_static, NULL, task); | ||||
| 	tf_http_add_handler(http, "/lit/*", tf_httpd_endpoint_static, NULL, task); | ||||
| @@ -973,7 +920,7 @@ tf_http_t* tf_httpd_create(JSContext* context) | ||||
| 	{ | ||||
| 		httpd_listener_t* listener = tf_malloc(sizeof(httpd_listener_t)); | ||||
| 		*listener = (httpd_listener_t) { 0 }; | ||||
| 		int assigned_port = tf_http_listen(http, http_port, local_only, NULL, _httpd_listener_cleanup, listener); | ||||
| 		int assigned_port = tf_http_listen(http, http_port, local_only, _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) | ||||
| @@ -992,26 +939,6 @@ tf_http_t* tf_httpd_create(JSContext* context) | ||||
| 			} | ||||
| 			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, local_only, 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); | ||||
| 		} | ||||
| 	} | ||||
| 	return http; | ||||
| } | ||||
|   | ||||
| @@ -37,11 +37,11 @@ typedef struct _login_request_t | ||||
| const char* tf_httpd_make_set_session_cookie_header(tf_http_request_t* request, const char* session_cookie) | ||||
| { | ||||
| 	const char* k_pattern = "session=%s; path=/; Max-Age=%" PRId64 "; %sSameSite=Strict; HttpOnly"; | ||||
| 	int length = session_cookie ? snprintf(NULL, 0, k_pattern, session_cookie, k_httpd_auth_refresh_interval, request->is_tls ? "Secure; " : "") : 0; | ||||
| 	int length = session_cookie ? snprintf(NULL, 0, k_pattern, session_cookie, k_httpd_auth_refresh_interval, "") : 0; | ||||
| 	char* cookie = length ? tf_malloc(length + 1) : NULL; | ||||
| 	if (cookie) | ||||
| 	{ | ||||
| 		snprintf(cookie, length + 1, k_pattern, session_cookie, k_httpd_auth_refresh_interval, request->is_tls ? "Secure; " : ""); | ||||
| 		snprintf(cookie, length + 1, k_pattern, session_cookie, k_httpd_auth_refresh_interval, ""); | ||||
| 	} | ||||
| 	return cookie; | ||||
| } | ||||
| @@ -226,7 +226,7 @@ static void _httpd_endpoint_login_work(tf_ssb_t* ssb, void* user_data) | ||||
| 		} | ||||
| 		else | ||||
| 		{ | ||||
| 			snprintf(login->location_header, sizeof(login->location_header), "%s%s/", request->is_tls ? "https://" : "http://", tf_http_request_get_header(request, "host")); | ||||
| 			snprintf(login->location_header, sizeof(login->location_header), "%s%s/", "http://", tf_http_request_get_header(request, "host")); | ||||
| 		} | ||||
| 		goto done; | ||||
| 	} | ||||
| @@ -332,7 +332,7 @@ static void _httpd_endpoint_login_work(tf_ssb_t* ssb, void* user_data) | ||||
| 		} | ||||
| 		else | ||||
| 		{ | ||||
| 			snprintf(login->location_header, sizeof(login->location_header), "%s%s/", request->is_tls ? "https://" : "http://", tf_http_request_get_header(request, "host")); | ||||
| 			snprintf(login->location_header, sizeof(login->location_header), "%s%s/", "http://", tf_http_request_get_header(request, "host")); | ||||
| 		} | ||||
| 		login->set_cookie_header = tf_httpd_make_set_session_cookie_header(request, send_session); | ||||
| 		tf_free((void*)send_session); | ||||
| @@ -416,8 +416,7 @@ void tf_httpd_endpoint_login(tf_http_request_t* request) | ||||
|  | ||||
| void tf_httpd_endpoint_logout(tf_http_request_t* request) | ||||
| { | ||||
| 	const char* k_set_cookie = request->is_tls ? "session=; path=/; Secure; SameSite=Strict; expires=Thu, 01 Jan 1970 00:00:00 GMT; HttpOnly" | ||||
| 											   : "session=; path=/; SameSite=Strict; expires=Thu, 01 Jan 1970 00:00:00 GMT; HttpOnly"; | ||||
| 	const char* k_set_cookie = "session=; path=/; SameSite=Strict; expires=Thu, 01 Jan 1970 00:00:00 GMT; HttpOnly"; | ||||
| 	const char* k_location_format = "/login%s%s"; | ||||
| 	int length = snprintf(NULL, 0, k_location_format, request->query ? "?" : "", request->query); | ||||
| 	char* location = alloca(length + 1); | ||||
|   | ||||
							
								
								
									
										
											BIN
										
									
								
								src/ios/tildefriends512.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								src/ios/tildefriends512.png
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| After Width: | Height: | Size: 44 KiB | 
| @@ -1502,13 +1502,11 @@ static int _tf_run_task(const tf_run_args_t* args, int index) | ||||
| 	tf_ssb_t* ssb = tf_task_get_ssb(task); | ||||
| 	sqlite3* db = tf_ssb_acquire_db_reader(ssb); | ||||
| 	int64_t http_port = 0; | ||||
| 	int64_t https_port = 0; | ||||
| 	char out_http_port_file[512] = ""; | ||||
| 	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 (http_port || https_port || *out_http_port_file) | ||||
| 	if (http_port || *out_http_port_file) | ||||
| 	{ | ||||
| 		if (args->zip) | ||||
| 		{ | ||||
| @@ -1869,7 +1867,6 @@ static void _startup(int argc, char* argv[]) | ||||
| 	prctl(PR_SET_PDEATHSIG, SIGKILL); | ||||
| #endif | ||||
| 	tf_mem_replace_uv_allocator(); | ||||
| 	tf_mem_replace_tls_allocator(); | ||||
| 	tf_mem_replace_sqlite_allocator(); | ||||
| 	uv_setup_args(argc, argv); | ||||
| 	tf_taskstub_startup(); | ||||
|   | ||||
							
								
								
									
										28
									
								
								src/mem.c
									
									
									
									
									
								
							
							
						
						
									
										28
									
								
								src/mem.c
									
									
									
									
									
								
							| @@ -7,8 +7,6 @@ | ||||
| #include "sqlite3.h" | ||||
| #include "uv.h" | ||||
|  | ||||
| #include <openssl/crypto.h> | ||||
|  | ||||
| #include <stdbool.h> | ||||
| #include <string.h> | ||||
|  | ||||
| @@ -19,7 +17,6 @@ static bool s_mem_tracking; | ||||
| static tf_mem_node_t* s_mem_tracked; | ||||
| static int64_t s_tf_malloc_size; | ||||
| static int64_t s_uv_malloc_size; | ||||
| static int64_t s_tls_malloc_size; | ||||
| static int64_t s_js_malloc_size; | ||||
| static int64_t s_sqlite_malloc_size; | ||||
|  | ||||
| @@ -387,31 +384,6 @@ size_t tf_mem_get_uv_malloc_size() | ||||
| 	return s_uv_malloc_size; | ||||
| } | ||||
|  | ||||
| static void* _tf_tls_alloc(size_t size, const char* file, int line) | ||||
| { | ||||
| 	return _tf_alloc(&s_tls_malloc_size, size); | ||||
| } | ||||
|  | ||||
| static void* _tf_tls_realloc(void* ptr, size_t size, const char* file, int line) | ||||
| { | ||||
| 	return _tf_realloc(&s_tls_malloc_size, ptr, size); | ||||
| } | ||||
|  | ||||
| static void _tf_tls_free(void* ptr, const char* file, int line) | ||||
| { | ||||
| 	_tf_free(&s_tls_malloc_size, ptr); | ||||
| } | ||||
|  | ||||
| void tf_mem_replace_tls_allocator() | ||||
| { | ||||
| 	CRYPTO_set_mem_functions(_tf_tls_alloc, _tf_tls_realloc, _tf_tls_free); | ||||
| } | ||||
|  | ||||
| size_t tf_mem_get_tls_malloc_size() | ||||
| { | ||||
| 	return s_tls_malloc_size; | ||||
| } | ||||
|  | ||||
| void* tf_malloc(size_t size) | ||||
| { | ||||
| 	return _tf_alloc(&s_tf_malloc_size, size); | ||||
|   | ||||
							
								
								
									
										13
									
								
								src/mem.h
									
									
									
									
									
								
							
							
						
						
									
										13
									
								
								src/mem.h
									
									
									
									
									
								
							| @@ -3,7 +3,7 @@ | ||||
| /** | ||||
| ** \defgroup mem Memory management | ||||
| ** tf_malloc() and friends use malloc() behind the scenes but optionally | ||||
| ** track memory per system (OpenSSL, sqlite, libuv, ...) and store callstacks | ||||
| ** track memory per system (sqlite, libuv, ...) and store callstacks | ||||
| ** to help debug leaks. | ||||
| ** @{ | ||||
| */ | ||||
| @@ -38,17 +38,6 @@ void tf_mem_replace_uv_allocator(); | ||||
| */ | ||||
| size_t tf_mem_get_uv_malloc_size(); | ||||
|  | ||||
| /** | ||||
| ** Register a custom allocator with OpenSSL. | ||||
| */ | ||||
| void tf_mem_replace_tls_allocator(); | ||||
|  | ||||
| /** | ||||
| ** Get the number of bytes currently allocated by OpenSSL. | ||||
| ** @return The allocated size in bytes. | ||||
| */ | ||||
| size_t tf_mem_get_tls_malloc_size(); | ||||
|  | ||||
| /** | ||||
| ** Register a custom allocator with SQLite. | ||||
| */ | ||||
|   | ||||
							
								
								
									
										329
									
								
								src/sha1.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										329
									
								
								src/sha1.c
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,329 @@ | ||||
| /* | ||||
|  * SHA1 hash implementation and interface functions | ||||
|  * Copyright (c) 2003-2005, Jouni Malinen <j@w1.fi> | ||||
|  * | ||||
|  * This software may be distributed under the terms of the BSD license. | ||||
|  * See README for more details. | ||||
|  */ | ||||
|  | ||||
| #include "sha1.h" | ||||
|  | ||||
| #include <stddef.h> | ||||
| #include <string.h> | ||||
|  | ||||
| /* ===== start - public domain SHA1 implementation ===== */ | ||||
|  | ||||
| /* | ||||
| SHA-1 in C | ||||
| By Steve Reid <sreid@sea-to-sky.net> | ||||
| 100% Public Domain | ||||
|  | ||||
| ----------------- | ||||
| Modified 7/98 | ||||
| By James H. Brown <jbrown@burgoyne.com> | ||||
| Still 100% Public Domain | ||||
|  | ||||
| Corrected a problem which generated improper hash values on 16 bit machines | ||||
| Routine SHA1Update changed from | ||||
| 	void SHA1Update(SHA1_CTX* context, unsigned char* data, unsigned int | ||||
| len) | ||||
| to | ||||
| 	void SHA1Update(SHA1_CTX* context, unsigned char* data, unsigned | ||||
| long len) | ||||
|  | ||||
| The 'len' parameter was declared an int which works fine on 32 bit machines. | ||||
| However, on 16 bit machines an int is too small for the shifts being done | ||||
| against it.  This caused the hash function to generate incorrect values if len | ||||
| was greater than 8191 (8K - 1) due to the 'len << 3' on line 3 of SHA1Update(). | ||||
|  | ||||
| Since the file IO in main() reads 16K at a time, any file 8K or larger would be | ||||
| guaranteed to generate the wrong hash (e.g. Test Vector #3, a million "a"s). | ||||
|  | ||||
| I also changed the declaration of variables i & j in SHA1Update to unsigned | ||||
| long from unsigned int for the same reason. | ||||
|  | ||||
| These changes should make no difference to any 32 bit implementations since an | ||||
| int and a long are the same size in those environments. | ||||
|  | ||||
| -- | ||||
| I also corrected a few compiler warnings generated by Borland C. | ||||
| 1. Added #include <process.h> for exit() prototype | ||||
| 2. Removed unused variable 'j' in SHA1Final | ||||
| 3. Changed exit(0) to return(0) at end of main. | ||||
|  | ||||
| ALL changes I made can be located by searching for comments containing 'JHB' | ||||
| ----------------- | ||||
| Modified 8/98 | ||||
| By Steve Reid <sreid@sea-to-sky.net> | ||||
| Still 100% public domain | ||||
|  | ||||
| 1- Removed #include <process.h> and used return() instead of exit() | ||||
| 2- Fixed overwriting of finalcount in SHA1Final() (discovered by Chris Hall) | ||||
| 3- Changed email address from steve@edmweb.com to sreid@sea-to-sky.net | ||||
|  | ||||
| ----------------- | ||||
| Modified 4/01 | ||||
| By Saul Kravitz <Saul.Kravitz@celera.com> | ||||
| Still 100% PD | ||||
| Modified to run on Compaq Alpha hardware. | ||||
|  | ||||
| ----------------- | ||||
| Modified 4/01 | ||||
| By Jouni Malinen <j@w1.fi> | ||||
| Minor changes to match the coding style used in Dynamics. | ||||
|  | ||||
| Modified September 24, 2004 | ||||
| By Jouni Malinen <j@w1.fi> | ||||
| Fixed alignment issue in SHA1Transform when SHA1HANDSOFF is defined. | ||||
|  | ||||
| ----------------- | ||||
| Modified September 29, 2025 | ||||
| By Cory McWilliams <cory@tildefriends.net> | ||||
| Adapted from | ||||
| https://web.mit.edu/freebsd/head/contrib/wpa/src/crypto/sha1-internal.c. | ||||
| Modified to build outside of FreeBSD.  Updated with clang-format. | ||||
|  | ||||
| */ | ||||
|  | ||||
| /* | ||||
| Test Vectors (from FIPS PUB 180-1) | ||||
| "abc" | ||||
|   A9993E36 4706816A BA3E2571 7850C26C 9CD0D89D | ||||
| "abcdbcdecdefdefgefghfghighijhijkijkljklmklmnlmnomnopnopq" | ||||
|   84983E44 1C3BD26E BAAE4AA1 F95129E5 E54670F1 | ||||
| A million repetitions of "a" | ||||
|   34AA973C D4C4DAA4 F61EEB2B DBAD2731 6534016F | ||||
| */ | ||||
|  | ||||
| #define SHA1HANDSOFF | ||||
|  | ||||
| #define rol(value, bits) (((value) << (bits)) | ((value) >> (32 - (bits)))) | ||||
|  | ||||
| /* blk0() and blk() perform the initial expand. */ | ||||
| /* I got the idea of expanding during the round function from SSLeay */ | ||||
| #ifndef WORDS_BIGENDIAN | ||||
| #define blk0(i) (block->l[i] = (rol(block->l[i], 24) & 0xFF00FF00) | (rol(block->l[i], 8) & 0x00FF00FF)) | ||||
| #else | ||||
| #define blk0(i) block->l[i] | ||||
| #endif | ||||
| #define blk(i) (block->l[i & 15] = rol(block->l[(i + 13) & 15] ^ block->l[(i + 8) & 15] ^ block->l[(i + 2) & 15] ^ block->l[i & 15], 1)) | ||||
|  | ||||
| /* (R0+R1), R2, R3, R4 are the different operations used in SHA1 */ | ||||
| #define R0(v, w, x, y, z, i) \ | ||||
| 	z += ((w & (x ^ y)) ^ y) + blk0(i) + 0x5A827999 + rol(v, 5); \ | ||||
| 	w = rol(w, 30); | ||||
| #define R1(v, w, x, y, z, i) \ | ||||
| 	z += ((w & (x ^ y)) ^ y) + blk(i) + 0x5A827999 + rol(v, 5); \ | ||||
| 	w = rol(w, 30); | ||||
| #define R2(v, w, x, y, z, i) \ | ||||
| 	z += (w ^ x ^ y) + blk(i) + 0x6ED9EBA1 + rol(v, 5); \ | ||||
| 	w = rol(w, 30); | ||||
| #define R3(v, w, x, y, z, i) \ | ||||
| 	z += (((w | x) & y) | (w & x)) + blk(i) + 0x8F1BBCDC + rol(v, 5); \ | ||||
| 	w = rol(w, 30); | ||||
| #define R4(v, w, x, y, z, i) \ | ||||
| 	z += (w ^ x ^ y) + blk(i) + 0xCA62C1D6 + rol(v, 5); \ | ||||
| 	w = rol(w, 30); | ||||
|  | ||||
| #ifdef VERBOSE /* SAK */ | ||||
| void SHAPrintContext(SHA1_CTX* context, char* msg) | ||||
| { | ||||
| 	printf("%s (%d,%d) %x %x %x %x %x\n", msg, context->count[0], context->count[1], context->state[0], context->state[1], context->state[2], context->state[3], context->state[4]); | ||||
| } | ||||
| #endif | ||||
|  | ||||
| /* Hash a single 512-bit block. This is the core of the algorithm. */ | ||||
|  | ||||
| void SHA1Transform(uint32_t state[5], const unsigned char buffer[64]) | ||||
| { | ||||
| 	uint32_t a, b, c, d, e; | ||||
| 	typedef union | ||||
| 	{ | ||||
| 		unsigned char c[64]; | ||||
| 		uint32_t l[16]; | ||||
| 	} CHAR64LONG16; | ||||
| 	CHAR64LONG16* block; | ||||
| #ifdef SHA1HANDSOFF | ||||
| 	CHAR64LONG16 workspace; | ||||
| 	block = &workspace; | ||||
| 	memcpy(block, buffer, 64); | ||||
| #else | ||||
| 	block = (CHAR64LONG16*)buffer; | ||||
| #endif | ||||
| 	/* Copy context->state[] to working vars */ | ||||
| 	a = state[0]; | ||||
| 	b = state[1]; | ||||
| 	c = state[2]; | ||||
| 	d = state[3]; | ||||
| 	e = state[4]; | ||||
| 	/* 4 rounds of 20 operations each. Loop unrolled. */ | ||||
| 	R0(a, b, c, d, e, 0); | ||||
| 	R0(e, a, b, c, d, 1); | ||||
| 	R0(d, e, a, b, c, 2); | ||||
| 	R0(c, d, e, a, b, 3); | ||||
| 	R0(b, c, d, e, a, 4); | ||||
| 	R0(a, b, c, d, e, 5); | ||||
| 	R0(e, a, b, c, d, 6); | ||||
| 	R0(d, e, a, b, c, 7); | ||||
| 	R0(c, d, e, a, b, 8); | ||||
| 	R0(b, c, d, e, a, 9); | ||||
| 	R0(a, b, c, d, e, 10); | ||||
| 	R0(e, a, b, c, d, 11); | ||||
| 	R0(d, e, a, b, c, 12); | ||||
| 	R0(c, d, e, a, b, 13); | ||||
| 	R0(b, c, d, e, a, 14); | ||||
| 	R0(a, b, c, d, e, 15); | ||||
| 	R1(e, a, b, c, d, 16); | ||||
| 	R1(d, e, a, b, c, 17); | ||||
| 	R1(c, d, e, a, b, 18); | ||||
| 	R1(b, c, d, e, a, 19); | ||||
| 	R2(a, b, c, d, e, 20); | ||||
| 	R2(e, a, b, c, d, 21); | ||||
| 	R2(d, e, a, b, c, 22); | ||||
| 	R2(c, d, e, a, b, 23); | ||||
| 	R2(b, c, d, e, a, 24); | ||||
| 	R2(a, b, c, d, e, 25); | ||||
| 	R2(e, a, b, c, d, 26); | ||||
| 	R2(d, e, a, b, c, 27); | ||||
| 	R2(c, d, e, a, b, 28); | ||||
| 	R2(b, c, d, e, a, 29); | ||||
| 	R2(a, b, c, d, e, 30); | ||||
| 	R2(e, a, b, c, d, 31); | ||||
| 	R2(d, e, a, b, c, 32); | ||||
| 	R2(c, d, e, a, b, 33); | ||||
| 	R2(b, c, d, e, a, 34); | ||||
| 	R2(a, b, c, d, e, 35); | ||||
| 	R2(e, a, b, c, d, 36); | ||||
| 	R2(d, e, a, b, c, 37); | ||||
| 	R2(c, d, e, a, b, 38); | ||||
| 	R2(b, c, d, e, a, 39); | ||||
| 	R3(a, b, c, d, e, 40); | ||||
| 	R3(e, a, b, c, d, 41); | ||||
| 	R3(d, e, a, b, c, 42); | ||||
| 	R3(c, d, e, a, b, 43); | ||||
| 	R3(b, c, d, e, a, 44); | ||||
| 	R3(a, b, c, d, e, 45); | ||||
| 	R3(e, a, b, c, d, 46); | ||||
| 	R3(d, e, a, b, c, 47); | ||||
| 	R3(c, d, e, a, b, 48); | ||||
| 	R3(b, c, d, e, a, 49); | ||||
| 	R3(a, b, c, d, e, 50); | ||||
| 	R3(e, a, b, c, d, 51); | ||||
| 	R3(d, e, a, b, c, 52); | ||||
| 	R3(c, d, e, a, b, 53); | ||||
| 	R3(b, c, d, e, a, 54); | ||||
| 	R3(a, b, c, d, e, 55); | ||||
| 	R3(e, a, b, c, d, 56); | ||||
| 	R3(d, e, a, b, c, 57); | ||||
| 	R3(c, d, e, a, b, 58); | ||||
| 	R3(b, c, d, e, a, 59); | ||||
| 	R4(a, b, c, d, e, 60); | ||||
| 	R4(e, a, b, c, d, 61); | ||||
| 	R4(d, e, a, b, c, 62); | ||||
| 	R4(c, d, e, a, b, 63); | ||||
| 	R4(b, c, d, e, a, 64); | ||||
| 	R4(a, b, c, d, e, 65); | ||||
| 	R4(e, a, b, c, d, 66); | ||||
| 	R4(d, e, a, b, c, 67); | ||||
| 	R4(c, d, e, a, b, 68); | ||||
| 	R4(b, c, d, e, a, 69); | ||||
| 	R4(a, b, c, d, e, 70); | ||||
| 	R4(e, a, b, c, d, 71); | ||||
| 	R4(d, e, a, b, c, 72); | ||||
| 	R4(c, d, e, a, b, 73); | ||||
| 	R4(b, c, d, e, a, 74); | ||||
| 	R4(a, b, c, d, e, 75); | ||||
| 	R4(e, a, b, c, d, 76); | ||||
| 	R4(d, e, a, b, c, 77); | ||||
| 	R4(c, d, e, a, b, 78); | ||||
| 	R4(b, c, d, e, a, 79); | ||||
| 	/* Add the working vars back into context.state[] */ | ||||
| 	state[0] += a; | ||||
| 	state[1] += b; | ||||
| 	state[2] += c; | ||||
| 	state[3] += d; | ||||
| 	state[4] += e; | ||||
| 	/* Wipe variables */ | ||||
| 	a = b = c = d = e = 0; | ||||
| #ifdef SHA1HANDSOFF | ||||
| 	memset(block, 0, 64); | ||||
| #endif | ||||
| } | ||||
|  | ||||
| /* SHA1Init - Initialize new context */ | ||||
|  | ||||
| void SHA1Init(SHA1_CTX* context) | ||||
| { | ||||
| 	/* SHA1 initialization constants */ | ||||
| 	context->state[0] = 0x67452301; | ||||
| 	context->state[1] = 0xEFCDAB89; | ||||
| 	context->state[2] = 0x98BADCFE; | ||||
| 	context->state[3] = 0x10325476; | ||||
| 	context->state[4] = 0xC3D2E1F0; | ||||
| 	context->count[0] = context->count[1] = 0; | ||||
| } | ||||
|  | ||||
| /* Run your data through this. */ | ||||
|  | ||||
| void SHA1Update(SHA1_CTX* context, const void* _data, uint32_t len) | ||||
| { | ||||
| 	uint32_t i, j; | ||||
| 	const unsigned char* data = _data; | ||||
|  | ||||
| #ifdef VERBOSE | ||||
| 	SHAPrintContext(context, "before"); | ||||
| #endif | ||||
| 	j = (context->count[0] >> 3) & 63; | ||||
| 	if ((context->count[0] += len << 3) < (len << 3)) | ||||
| 		context->count[1]++; | ||||
| 	context->count[1] += (len >> 29); | ||||
| 	if ((j + len) > 63) | ||||
| 	{ | ||||
| 		memcpy(&context->buffer[j], data, (i = 64 - j)); | ||||
| 		SHA1Transform(context->state, context->buffer); | ||||
| 		for (; i + 63 < len; i += 64) | ||||
| 		{ | ||||
| 			SHA1Transform(context->state, &data[i]); | ||||
| 		} | ||||
| 		j = 0; | ||||
| 	} | ||||
| 	else | ||||
| 		i = 0; | ||||
| 	memcpy(&context->buffer[j], &data[i], len - i); | ||||
| #ifdef VERBOSE | ||||
| 	SHAPrintContext(context, "after "); | ||||
| #endif | ||||
| } | ||||
|  | ||||
| /* Add padding and return the message digest. */ | ||||
|  | ||||
| void SHA1Final(unsigned char digest[20], SHA1_CTX* context) | ||||
| { | ||||
| 	uint32_t i; | ||||
| 	unsigned char finalcount[8]; | ||||
|  | ||||
| 	for (i = 0; i < 8; i++) | ||||
| 	{ | ||||
| 		/* Endian independent */ | ||||
| 		finalcount[i] = (unsigned char)((context->count[(i >= 4 ? 0 : 1)] >> ((3 - (i & 3)) * 8)) & 255); | ||||
| 	} | ||||
| 	SHA1Update(context, (unsigned char*)"\200", 1); | ||||
| 	while ((context->count[0] & 504) != 448) | ||||
| 	{ | ||||
| 		SHA1Update(context, (unsigned char*)"\0", 1); | ||||
| 	} | ||||
| 	/* Should cause a SHA1Transform() */ | ||||
| 	SHA1Update(context, finalcount, 8); | ||||
| 	for (i = 0; i < 20; i++) | ||||
| 	{ | ||||
| 		digest[i] = (unsigned char)((context->state[i >> 2] >> ((3 - (i & 3)) * 8)) & 255); | ||||
| 	} | ||||
| 	/* Wipe variables */ | ||||
| 	i = 0; | ||||
| 	memset(context->buffer, 0, 64); | ||||
| 	memset(context->state, 0, 20); | ||||
| 	memset(context->count, 0, 8); | ||||
| 	memset(finalcount, 0, 8); | ||||
| } | ||||
|  | ||||
| /* ===== end - public domain SHA1 implementation ===== */ | ||||
							
								
								
									
										71
									
								
								src/sha1.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										71
									
								
								src/sha1.h
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,71 @@ | ||||
| /* | ||||
|  * SHA1 internal definitions | ||||
|  * Copyright (c) 2003-2005, Jouni Malinen <j@w1.fi> | ||||
|  * | ||||
|  * This software may be distributed under the terms of the BSD license. | ||||
|  * See README for more details. | ||||
|  */ | ||||
|  | ||||
| /** | ||||
| ** \defgroup sha1 SHA1 | ||||
| ** SHA1 API. | ||||
| ** Adapted from | ||||
| ** https://web.mit.edu/freebsd/head/contrib/wpa/src/crypto/sha1_i.h by Cory | ||||
| ** McWilliams 2025-09-28. | ||||
| ** @{ | ||||
| */ | ||||
|  | ||||
| #ifndef SHA1_I_H | ||||
| #define SHA1_I_H | ||||
|  | ||||
| #include <inttypes.h> | ||||
|  | ||||
| /** | ||||
| ** SHA1 context struct. | ||||
| */ | ||||
| struct SHA1Context | ||||
| { | ||||
| 	/** SHA1 state. */ | ||||
| 	uint32_t state[5]; | ||||
| 	/** SHA1 count. */ | ||||
| 	uint32_t count[2]; | ||||
| 	/** SHA1 buffer. */ | ||||
| 	unsigned char buffer[64]; | ||||
| }; | ||||
|  | ||||
| /** | ||||
| ** SHA1 context. | ||||
| */ | ||||
| typedef struct SHA1Context SHA1_CTX; | ||||
|  | ||||
| /** | ||||
| ** Initialize a SHA1 context. | ||||
| ** @param context The context. | ||||
| */ | ||||
| void SHA1Init(struct SHA1Context* context); | ||||
|  | ||||
| /** | ||||
| ** Calculate an ongoing hash for a block of data. | ||||
| ** @param context The SHA1 context. | ||||
| ** @param data The data to hash. | ||||
| ** @param len The length of data. | ||||
| */ | ||||
| void SHA1Update(struct SHA1Context* context, const void* data, uint32_t len); | ||||
|  | ||||
| /** | ||||
| ** Calculate the final hash digest. | ||||
| ** @param digest Populated with the digest. | ||||
| ** @param context The SHA1 context. | ||||
| */ | ||||
| void SHA1Final(unsigned char digest[20], struct SHA1Context* context); | ||||
|  | ||||
| /** | ||||
| ** Perform a SHA1 transformation. | ||||
| ** @param state The SHA1 state. | ||||
| ** @param buffer The data. | ||||
| */ | ||||
| void SHA1Transform(uint32_t state[5], const unsigned char buffer[64]); | ||||
|  | ||||
| #endif /* SHA1_I_H */ | ||||
|  | ||||
| /** @} */ | ||||
							
								
								
									
										1165
									
								
								src/socket.js.c
									
									
									
									
									
								
							
							
						
						
									
										1165
									
								
								src/socket.js.c
									
									
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @@ -1,30 +0,0 @@ | ||||
| #pragma once | ||||
|  | ||||
| /** | ||||
| ** \defgroup socket_js Socket Interface | ||||
| ** Exposes network sockets to script. | ||||
| ** @{ | ||||
| */ | ||||
|  | ||||
| #include "quickjs.h" | ||||
|  | ||||
| /** | ||||
| ** Register the socket script interface. | ||||
| ** @param context The JS context. | ||||
| ** @return The Socket constructor. | ||||
| */ | ||||
| JSValue tf_socket_register(JSContext* context); | ||||
|  | ||||
| /** | ||||
| ** Get the number of active socket objects. | ||||
| ** @return The count. | ||||
| */ | ||||
| int tf_socket_get_count(); | ||||
|  | ||||
| /** | ||||
| ** Get the number of connected socket objects. | ||||
| ** @return the count. | ||||
| */ | ||||
| int tf_socket_get_open_count(); | ||||
|  | ||||
| /** @} */ | ||||
| @@ -1,5 +1,6 @@ | ||||
| #include "ssb.h" | ||||
|  | ||||
| #include "http.h" | ||||
| #include "log.h" | ||||
| #include "mem.h" | ||||
| #include "ssb.connections.h" | ||||
| @@ -2711,6 +2712,7 @@ void tf_ssb_destroy(tf_ssb_t* ssb) | ||||
| 			tf_printf("--\n"); | ||||
| 			uv_print_all_handles(ssb->loop, stdout); | ||||
| 		} | ||||
| 		tf_http_debug_destroy(); | ||||
| 		uv_run(ssb->loop, UV_RUN_ONCE); | ||||
| 	} | ||||
|  | ||||
|   | ||||
							
								
								
									
										26
									
								
								src/ssb.db.c
									
									
									
									
									
								
							
							
						
						
									
										26
									
								
								src/ssb.db.c
									
									
									
									
									
								
							| @@ -2465,6 +2465,32 @@ bool tf_ssb_db_get_global_setting_string(sqlite3* db, const char* name, char* ou | ||||
| 	return result; | ||||
| } | ||||
|  | ||||
| const char* tf_ssb_db_get_global_setting_string_alloc(sqlite3* db, const char* name) | ||||
| { | ||||
| 	const char* result = NULL; | ||||
| 	sqlite3_stmt* statement; | ||||
| 	if (sqlite3_prepare_v2(db, "SELECT json_extract(value, '$.' || ?) FROM properties WHERE id = 'core' AND key = 'settings'", -1, &statement, NULL) == SQLITE_OK) | ||||
| 	{ | ||||
| 		if (sqlite3_bind_text(statement, 1, name, -1, NULL) == SQLITE_OK) | ||||
| 		{ | ||||
| 			if (sqlite3_step(statement) == SQLITE_ROW && sqlite3_column_type(statement, 0) != SQLITE_NULL) | ||||
| 			{ | ||||
| 				result = tf_strdup((const char*)sqlite3_column_text(statement, 0)); | ||||
| 			} | ||||
| 		} | ||||
| 		sqlite3_finalize(statement); | ||||
| 	} | ||||
| 	else | ||||
| 	{ | ||||
| 		tf_printf("prepare failed: %s\n", sqlite3_errmsg(db)); | ||||
| 	} | ||||
| 	if (!result) | ||||
| 	{ | ||||
| 		result = tf_strdup(tf_util_get_default_global_setting_string(name)); | ||||
| 	} | ||||
| 	return result; | ||||
| } | ||||
|  | ||||
| bool tf_ssb_db_set_global_setting_from_string(sqlite3* db, const char* name, char* value) | ||||
| { | ||||
| 	tf_setting_kind_t kind = tf_util_get_global_setting_kind(name); | ||||
|   | ||||
| @@ -500,6 +500,14 @@ bool tf_ssb_db_get_global_setting_int64(sqlite3* db, const char* name, int64_t* | ||||
| */ | ||||
| bool tf_ssb_db_get_global_setting_string(sqlite3* db, const char* name, char* out_value, size_t size); | ||||
|  | ||||
| /** | ||||
| ** Get a string global setting value. | ||||
| ** @param db The database. | ||||
| ** @param name The setting name. | ||||
| ** @return The setting if found. | ||||
| */ | ||||
| const char* tf_ssb_db_get_global_setting_string_alloc(sqlite3* db, const char* name); | ||||
|  | ||||
| /** | ||||
| ** Set a global setting from a string representation of its value. | ||||
| ** @param db The database. | ||||
|   | ||||
							
								
								
									
										195
									
								
								src/ssb.js.c
									
									
									
									
									
								
							
							
						
						
									
										195
									
								
								src/ssb.js.c
									
									
									
									
									
								
							| @@ -363,85 +363,6 @@ static JSValue _tf_ssb_swap_with_server_identity(JSContext* context, JSValueCons | ||||
| 	return result; | ||||
| } | ||||
|  | ||||
| typedef struct _identities_visit_t | ||||
| { | ||||
| 	JSContext* context; | ||||
| 	JSValue promise[2]; | ||||
| 	const char** identities; | ||||
| 	int count; | ||||
| 	char user[]; | ||||
| } identities_visit_t; | ||||
|  | ||||
| static void _tf_ssb_getIdentities_visit(const char* identity, void* user_data) | ||||
| { | ||||
| 	identities_visit_t* work = user_data; | ||||
| 	work->identities = tf_resize_vec(work->identities, (work->count + 1) * sizeof(const char*)); | ||||
| 	char id[k_id_base64_len]; | ||||
| 	snprintf(id, sizeof(id), "@%s", identity); | ||||
| 	work->identities[work->count++] = tf_strdup(id); | ||||
| } | ||||
|  | ||||
| static void _tf_ssb_get_identities_work(tf_ssb_t* ssb, void* user_data) | ||||
| { | ||||
| 	identities_visit_t* work = user_data; | ||||
| 	if (tf_ssb_db_user_has_permission(ssb, NULL, work->user, "administration")) | ||||
| 	{ | ||||
| 		char id[k_id_base64_len] = ""; | ||||
| 		if (tf_ssb_whoami(ssb, id, sizeof(id))) | ||||
| 		{ | ||||
| 			_tf_ssb_getIdentities_visit(*id == '@' ? id + 1 : id, work); | ||||
| 		} | ||||
| 	} | ||||
| 	tf_ssb_db_identity_visit(ssb, work->user, _tf_ssb_getIdentities_visit, user_data); | ||||
| } | ||||
|  | ||||
| static void _tf_ssb_get_all_identities_work(tf_ssb_t* ssb, void* user_data) | ||||
| { | ||||
| 	tf_ssb_db_identity_visit_all(ssb, _tf_ssb_getIdentities_visit, user_data); | ||||
| } | ||||
|  | ||||
| static void _tf_ssb_get_identities_after_work(tf_ssb_t* ssb, int status, void* user_data) | ||||
| { | ||||
| 	identities_visit_t* work = user_data; | ||||
| 	JSContext* context = tf_ssb_get_context(ssb); | ||||
| 	JSValue result = JS_NewArray(context); | ||||
| 	for (int i = 0; i < work->count; i++) | ||||
| 	{ | ||||
| 		JS_SetPropertyUint32(context, result, i, JS_NewString(context, work->identities[i])); | ||||
| 		tf_free((void*)work->identities[i]); | ||||
| 	} | ||||
| 	tf_free(work->identities); | ||||
|  | ||||
| 	JSValue error = JS_Call(context, work->promise[0], JS_UNDEFINED, 1, &result); | ||||
| 	JS_FreeValue(context, result); | ||||
| 	tf_util_report_error(context, error); | ||||
| 	JS_FreeValue(context, error); | ||||
| 	JS_FreeValue(context, work->promise[0]); | ||||
| 	JS_FreeValue(context, work->promise[1]); | ||||
| 	tf_free(work); | ||||
| } | ||||
|  | ||||
| static JSValue _tf_ssb_getIdentities(JSContext* context, JSValueConst this_val, int argc, JSValueConst* argv) | ||||
| { | ||||
| 	JSValue result = JS_UNDEFINED; | ||||
| 	tf_ssb_t* ssb = JS_GetOpaque(this_val, _tf_ssb_classId); | ||||
| 	if (ssb) | ||||
| 	{ | ||||
| 		size_t user_length = 0; | ||||
| 		const char* user = JS_ToCStringLen(context, &user_length, argv[0]); | ||||
| 		identities_visit_t* work = tf_malloc(sizeof(identities_visit_t) + user_length + 1); | ||||
| 		*work = (identities_visit_t) { | ||||
| 			.context = context, | ||||
| 		}; | ||||
| 		memcpy(work->user, user, user_length + 1); | ||||
| 		JS_FreeCString(context, user); | ||||
|  | ||||
| 		result = JS_NewPromiseCapability(context, work->promise); | ||||
| 		tf_ssb_run_work(ssb, _tf_ssb_get_identities_work, _tf_ssb_get_identities_after_work, work); | ||||
| 	} | ||||
| 	return result; | ||||
| } | ||||
|  | ||||
| typedef struct _get_private_key_t | ||||
| { | ||||
| 	JSContext* context; | ||||
| @@ -513,109 +434,6 @@ static JSValue _tf_ssb_getServerIdentity(JSContext* context, JSValueConst this_v | ||||
| 	return result; | ||||
| } | ||||
|  | ||||
| static JSValue _tf_ssb_getAllIdentities(JSContext* context, JSValueConst this_val, int argc, JSValueConst* argv) | ||||
| { | ||||
| 	JSValue result = JS_UNDEFINED; | ||||
| 	tf_ssb_t* ssb = JS_GetOpaque(this_val, _tf_ssb_classId); | ||||
| 	if (ssb) | ||||
| 	{ | ||||
| 		identities_visit_t* work = tf_malloc(sizeof(identities_visit_t)); | ||||
| 		*work = (identities_visit_t) { | ||||
| 			.context = context, | ||||
| 		}; | ||||
|  | ||||
| 		result = JS_NewPromiseCapability(context, work->promise); | ||||
| 		tf_ssb_run_work(ssb, _tf_ssb_get_all_identities_work, _tf_ssb_get_identities_after_work, work); | ||||
| 	} | ||||
| 	return result; | ||||
| } | ||||
|  | ||||
| typedef struct _active_identity_work_t | ||||
| { | ||||
| 	JSContext* context; | ||||
| 	const char* name; | ||||
| 	const char* package_owner; | ||||
| 	const char* package_name; | ||||
| 	char identity[k_id_base64_len]; | ||||
| 	int result; | ||||
| 	JSValue promise[2]; | ||||
| } active_identity_work_t; | ||||
|  | ||||
| static void _tf_ssb_getActiveIdentity_visit(const char* identity, void* user_data) | ||||
| { | ||||
| 	active_identity_work_t* request = user_data; | ||||
| 	if (!*request->identity) | ||||
| 	{ | ||||
| 		snprintf(request->identity, sizeof(request->identity), "@%s", identity); | ||||
| 	} | ||||
| } | ||||
|  | ||||
| static void _tf_ssb_getActiveIdentity_work(tf_ssb_t* ssb, void* user_data) | ||||
| { | ||||
| 	active_identity_work_t* request = user_data; | ||||
| 	sqlite3* db = tf_ssb_acquire_db_reader(ssb); | ||||
| 	tf_ssb_db_identity_get_active(db, request->name, request->package_owner, request->package_name, request->identity, sizeof(request->identity)); | ||||
| 	tf_ssb_release_db_reader(ssb, db); | ||||
|  | ||||
| 	if (!*request->identity) | ||||
| 	{ | ||||
| 		tf_ssb_db_identity_visit(ssb, request->name, _tf_ssb_getActiveIdentity_visit, request); | ||||
| 	} | ||||
|  | ||||
| 	if (!*request->identity && tf_ssb_db_user_has_permission(ssb, NULL, request->name, "administration")) | ||||
| 	{ | ||||
| 		tf_ssb_whoami(ssb, request->identity, sizeof(request->identity)); | ||||
| 	} | ||||
| } | ||||
|  | ||||
| static void _tf_ssb_getActiveIdentity_after_work(tf_ssb_t* ssb, int status, void* user_data) | ||||
| { | ||||
| 	active_identity_work_t* request = user_data; | ||||
| 	JSContext* context = request->context; | ||||
| 	if (request->result == 0) | ||||
| 	{ | ||||
| 		JSValue identity = JS_NewString(context, request->identity); | ||||
| 		JSValue error = JS_Call(context, request->promise[0], JS_UNDEFINED, 1, &identity); | ||||
| 		JS_FreeValue(context, identity); | ||||
| 		tf_util_report_error(context, error); | ||||
| 		JS_FreeValue(context, error); | ||||
| 	} | ||||
| 	else | ||||
| 	{ | ||||
| 		JSValue error = JS_Call(context, request->promise[1], JS_UNDEFINED, 0, NULL); | ||||
| 		tf_util_report_error(context, error); | ||||
| 		JS_FreeValue(context, error); | ||||
| 	} | ||||
| 	JS_FreeValue(context, request->promise[0]); | ||||
| 	JS_FreeValue(context, request->promise[1]); | ||||
| 	tf_free((void*)request->name); | ||||
| 	tf_free((void*)request->package_owner); | ||||
| 	tf_free((void*)request->package_name); | ||||
| 	tf_free(request); | ||||
| } | ||||
|  | ||||
| static JSValue _tf_ssb_getActiveIdentity(JSContext* context, JSValueConst this_val, int argc, JSValueConst* argv) | ||||
| { | ||||
| 	tf_ssb_t* ssb = JS_GetOpaque(this_val, _tf_ssb_classId); | ||||
| 	const char* name = JS_ToCString(context, argv[0]); | ||||
| 	const char* package_owner = JS_ToCString(context, argv[1]); | ||||
| 	const char* package_name = JS_ToCString(context, argv[2]); | ||||
| 	active_identity_work_t* work = tf_malloc(sizeof(active_identity_work_t)); | ||||
| 	*work = (active_identity_work_t) { | ||||
| 		.context = context, | ||||
| 		.name = tf_strdup(name), | ||||
| 		.package_owner = tf_strdup(package_owner), | ||||
| 		.package_name = tf_strdup(package_name), | ||||
| 	}; | ||||
| 	JSValue result = JS_NewPromiseCapability(context, work->promise); | ||||
| 	JS_FreeCString(context, name); | ||||
| 	JS_FreeCString(context, package_owner); | ||||
| 	JS_FreeCString(context, package_name); | ||||
|  | ||||
| 	tf_ssb_run_work(ssb, _tf_ssb_getActiveIdentity_work, _tf_ssb_getActiveIdentity_after_work, work); | ||||
| 	return result; | ||||
| } | ||||
|  | ||||
| typedef struct _identity_info_work_t | ||||
| { | ||||
| 	JSContext* context; | ||||
| @@ -2370,12 +2188,15 @@ void tf_ssb_register(JSContext* context, tf_ssb_t* ssb) | ||||
| 	JS_SetPropertyStr(context, global, "ssb", object); | ||||
| 	JS_SetOpaque(object, ssb); | ||||
|  | ||||
| 	JSValue object_internal = JS_NewObjectClass(context, _tf_ssb_classId); | ||||
| 	JS_SetPropertyStr(context, global, "ssb_internal", object_internal); | ||||
| 	JS_SetOpaque(object_internal, ssb); | ||||
|  | ||||
| 	/* Requires an identity. */ | ||||
| 	JS_SetPropertyStr(context, object, "createIdentity", JS_NewCFunction(context, _tf_ssb_createIdentity, "createIdentity", 1)); | ||||
| 	JS_SetPropertyStr(context, object, "addIdentity", JS_NewCFunction(context, _tf_ssb_addIdentity, "addIdentity", 2)); | ||||
| 	JS_SetPropertyStr(context, object, "deleteIdentity", JS_NewCFunction(context, _tf_ssb_deleteIdentity, "deleteIdentity", 2)); | ||||
| 	JS_SetPropertyStr(context, object, "swapWithServerIdentity", JS_NewCFunction(context, _tf_ssb_swap_with_server_identity, "swapWithServerIdentity", 2)); | ||||
| 	JS_SetPropertyStr(context, object, "getIdentities", JS_NewCFunction(context, _tf_ssb_getIdentities, "getIdentities", 1)); | ||||
| 	JS_SetPropertyStr(context, object, "getPrivateKey", JS_NewCFunction(context, _tf_ssb_getPrivateKey, "getPrivateKey", 2)); | ||||
| 	JS_SetPropertyStr(context, object, "privateMessageEncrypt", JS_NewCFunction(context, _tf_ssb_private_message_encrypt, "privateMessageEncrypt", 4)); | ||||
| 	JS_SetPropertyStr(context, object, "privateMessageDecrypt", JS_NewCFunction(context, _tf_ssb_private_message_decrypt, "privateMessageDecrypt", 3)); | ||||
| @@ -2385,9 +2206,6 @@ void tf_ssb_register(JSContext* context, tf_ssb_t* ssb) | ||||
|  | ||||
| 	/* Does not require an identity. */ | ||||
| 	JS_SetPropertyStr(context, object, "getServerIdentity", JS_NewCFunction(context, _tf_ssb_getServerIdentity, "getServerIdentity", 0)); | ||||
| 	JS_SetPropertyStr(context, object, "getAllIdentities", JS_NewCFunction(context, _tf_ssb_getAllIdentities, "getAllIdentities", 0)); | ||||
| 	JS_SetPropertyStr(context, object, "getActiveIdentity", JS_NewCFunction(context, _tf_ssb_getActiveIdentity, "getActiveIdentity", 3)); | ||||
| 	JS_SetPropertyStr(context, object, "getIdentityInfo", JS_NewCFunction(context, _tf_ssb_getIdentityInfo, "getIdentityInfo", 3)); | ||||
| 	JS_SetPropertyStr(context, object, "blobGet", JS_NewCFunction(context, _tf_ssb_blobGet, "blobGet", 1)); | ||||
| 	JS_SetPropertyStr(context, object, "connections", JS_NewCFunction(context, _tf_ssb_connections, "connections", 0)); | ||||
| 	JS_SetPropertyStr(context, object, "storedConnections", JS_NewCFunction(context, _tf_ssb_storedConnections, "storedConnections", 0)); | ||||
| @@ -2406,8 +2224,9 @@ void tf_ssb_register(JSContext* context, tf_ssb_t* ssb) | ||||
| 	JS_SetPropertyStr(context, object, "blobStore", JS_NewCFunction(context, _tf_ssb_blobStore, "blobStore", 1)); | ||||
|  | ||||
| 	/* Trusted only. */ | ||||
| 	JS_SetPropertyStr(context, object, "addEventListener", JS_NewCFunction(context, _tf_ssb_add_event_listener, "addEventListener", 2)); | ||||
| 	JS_SetPropertyStr(context, object, "removeEventListener", JS_NewCFunction(context, _tf_ssb_remove_event_listener, "removeEventListener", 2)); | ||||
| 	JS_SetPropertyStr(context, object_internal, "getIdentityInfo", JS_NewCFunction(context, _tf_ssb_getIdentityInfo, "getIdentityInfo", 3)); | ||||
| 	JS_SetPropertyStr(context, object_internal, "addEventListener", JS_NewCFunction(context, _tf_ssb_add_event_listener, "addEventListener", 2)); | ||||
| 	JS_SetPropertyStr(context, object_internal, "removeEventListener", JS_NewCFunction(context, _tf_ssb_remove_event_listener, "removeEventListener", 2)); | ||||
|  | ||||
| 	JS_FreeValue(context, global); | ||||
| } | ||||
|   | ||||
							
								
								
									
										198
									
								
								src/ssb.tests.c
									
									
									
									
									
								
							
							
						
						
									
										198
									
								
								src/ssb.tests.c
									
									
									
									
									
								
							| @@ -923,7 +923,7 @@ static void _write_file(const char* path, const char* contents) | ||||
| 	fclose(file); | ||||
| } | ||||
|  | ||||
| #define TEST_ARGS " --args=ssb_port=0,http_port=0,https_port=0" | ||||
| #define TEST_ARGS " --args=ssb_port=0,http_port=0" | ||||
|  | ||||
| void tf_ssb_test_encrypt(const tf_test_options_t* options) | ||||
| { | ||||
| @@ -1220,7 +1220,181 @@ void tf_ssb_test_replicate(const tf_test_options_t* options) | ||||
| 	tf_printf("done\n"); | ||||
|  | ||||
| 	tf_printf("Waiting for blob.\n"); | ||||
| 	while (!tf_ssb_db_blob_get(ssb0, blob_id, NULL, NULL)) | ||||
| 	while (!tf_ssb_db_blob_get(ssb1, blob_id, NULL, NULL)) | ||||
| 	{ | ||||
| 		tf_ssb_set_main_thread(ssb1, true); | ||||
| 		uv_run(&loop, UV_RUN_ONCE); | ||||
| 		tf_ssb_set_main_thread(ssb1, false); | ||||
| 	} | ||||
| 	tf_printf("done\n"); | ||||
|  | ||||
| 	tf_ssb_send_close(ssb1); | ||||
|  | ||||
| 	uv_close((uv_handle_t*)&idle0, NULL); | ||||
| 	uv_close((uv_handle_t*)&idle1, NULL); | ||||
|  | ||||
| 	tf_printf("final run\n"); | ||||
| 	tf_ssb_set_main_thread(ssb0, true); | ||||
| 	tf_ssb_set_main_thread(ssb1, true); | ||||
| 	uv_run(&loop, UV_RUN_DEFAULT); | ||||
| 	tf_ssb_set_main_thread(ssb0, false); | ||||
| 	tf_ssb_set_main_thread(ssb1, false); | ||||
| 	tf_printf("done\n"); | ||||
|  | ||||
| 	tf_printf("destroy 0\n"); | ||||
| 	tf_ssb_destroy(ssb0); | ||||
| 	tf_printf("destroy 1\n"); | ||||
| 	tf_ssb_destroy(ssb1); | ||||
|  | ||||
| 	tf_printf("close\n"); | ||||
| 	uv_loop_close(&loop); | ||||
| } | ||||
|  | ||||
| void tf_ssb_test_replicate_blob(const tf_test_options_t* options) | ||||
| { | ||||
| 	tf_printf("Testing blob replication.\n"); | ||||
|  | ||||
| 	uv_loop_t loop = { 0 }; | ||||
| 	uv_loop_init(&loop); | ||||
|  | ||||
| 	unlink("out/test_db0.sqlite"); | ||||
| 	tf_ssb_t* ssb0 = tf_ssb_create(&loop, NULL, "file:out/test_db0.sqlite", NULL); | ||||
| 	tf_ssb_register(tf_ssb_get_context(ssb0), ssb0); | ||||
| 	unlink("out/test_db1.sqlite"); | ||||
| 	tf_ssb_t* ssb1 = tf_ssb_create(&loop, NULL, "file:out/test_db1.sqlite", NULL); | ||||
| 	tf_ssb_register(tf_ssb_get_context(ssb1), ssb1); | ||||
|  | ||||
| 	uv_idle_t idle0 = { .data = ssb0 }; | ||||
| 	uv_idle_init(&loop, &idle0); | ||||
| 	uv_idle_start(&idle0, _ssb_test_idle); | ||||
|  | ||||
| 	uv_idle_t idle1 = { .data = ssb1 }; | ||||
| 	uv_idle_init(&loop, &idle1); | ||||
| 	uv_idle_start(&idle1, _ssb_test_idle); | ||||
|  | ||||
| 	test_t test = { | ||||
| 		.ssb0 = ssb0, | ||||
| 		.ssb1 = ssb1, | ||||
| 	}; | ||||
|  | ||||
| 	tf_ssb_add_connections_changed_callback(ssb0, _ssb_test_connections_changed, NULL, &test); | ||||
| 	tf_ssb_add_connections_changed_callback(ssb1, _ssb_test_connections_changed, NULL, &test); | ||||
|  | ||||
| 	tf_ssb_generate_keys(ssb0); | ||||
| 	tf_ssb_generate_keys(ssb1); | ||||
|  | ||||
| 	uint8_t priv0[crypto_sign_SECRETKEYBYTES] = { 0 }; | ||||
| 	uint8_t priv1[crypto_sign_SECRETKEYBYTES] = { 0 }; | ||||
| 	tf_ssb_get_private_key(ssb0, priv0, sizeof(priv0)); | ||||
| 	tf_ssb_get_private_key(ssb1, priv1, sizeof(priv1)); | ||||
|  | ||||
| 	char id0[k_id_base64_len] = { 0 }; | ||||
| 	char id1[k_id_base64_len] = { 0 }; | ||||
| 	bool b = tf_ssb_whoami(ssb0, id0, sizeof(id0)); | ||||
| 	(void)b; | ||||
| 	assert(b); | ||||
| 	b = tf_ssb_whoami(ssb1, id1, sizeof(id1)); | ||||
| 	assert(b); | ||||
| 	tf_printf("ID %s and %s\n", id0, id1); | ||||
|  | ||||
| 	char priv0_str[512] = { 0 }; | ||||
| 	char priv1_str[512] = { 0 }; | ||||
| 	tf_base64_encode(priv0, sizeof(priv0), priv0_str, sizeof(priv0_str)); | ||||
| 	tf_base64_encode(priv1, sizeof(priv0), priv1_str, sizeof(priv1_str)); | ||||
| 	tf_ssb_db_identity_add(ssb0, "test", id0 + 1, priv0_str); | ||||
| 	tf_ssb_db_identity_add(ssb1, "test", id1 + 1, priv1_str); | ||||
|  | ||||
| 	static const int k_key_count = 5; | ||||
| 	char public[k_key_count][k_id_base64_len - 1]; | ||||
| 	char private[k_key_count][512]; | ||||
| 	for (int i = 0; i < k_key_count; i++) | ||||
| 	{ | ||||
| 		tf_ssb_generate_keys_buffer(public[i], sizeof(public[i]), private[i], sizeof(private[i])); | ||||
| 		bool added = tf_ssb_db_identity_add(ssb0, "test", public[i], private[i]); | ||||
| 		tf_printf("%s user %d = %s private=%s\n", added ? "added" : "failed", i, public[i], private[i]); | ||||
| 	} | ||||
|  | ||||
| 	tf_printf("ssb0\n"); | ||||
| 	tf_ssb_db_identity_visit_all(ssb0, _test_print_identity, ssb0); | ||||
| 	tf_printf("ssb1\n"); | ||||
| 	tf_ssb_db_identity_visit_all(ssb1, _test_print_identity, ssb1); | ||||
|  | ||||
| 	tf_ssb_server_open(ssb0, 12347); | ||||
|  | ||||
| 	uint8_t id0bin[k_id_bin_len]; | ||||
| 	tf_ssb_id_str_to_bin(id0bin, id0); | ||||
| 	tf_ssb_connect(ssb1, "127.0.0.1", 12347, id0bin, 0, NULL, NULL); | ||||
|  | ||||
| 	tf_printf("Waiting for connection.\n"); | ||||
| 	while (test.connection_count0 != 1 || test.connection_count1 != 1) | ||||
| 	{ | ||||
| 		tf_ssb_set_main_thread(ssb0, true); | ||||
| 		tf_ssb_set_main_thread(ssb1, true); | ||||
| 		uv_run(&loop, UV_RUN_ONCE); | ||||
| 		tf_ssb_set_main_thread(ssb0, false); | ||||
| 		tf_ssb_set_main_thread(ssb1, false); | ||||
| 	} | ||||
| 	tf_ssb_server_close(ssb0); | ||||
|  | ||||
| 	char blob_id[k_id_base64_len] = { 0 }; | ||||
| 	const char* k_blob = "Hello, new blob!"; | ||||
| 	b = tf_ssb_db_blob_store(ssb0, (const uint8_t*)k_blob, strlen(k_blob), blob_id, sizeof(blob_id), NULL); | ||||
| 	assert(b); | ||||
|  | ||||
| 	JSContext* context0 = tf_ssb_get_context(ssb0); | ||||
| 	for (int i = 0; i < k_key_count - 1; i++) | ||||
| 	{ | ||||
| 		JSValue obj = JS_NewObject(context0); | ||||
| 		JS_SetPropertyStr(context0, obj, "type", JS_NewString(context0, "contact")); | ||||
| 		char self[k_id_base64_len]; | ||||
| 		snprintf(self, sizeof(self), "@%s", public[i]); | ||||
| 		char contact[k_id_base64_len]; | ||||
| 		snprintf(contact, sizeof(contact), "@%s", public[i + 1]); | ||||
| 		JS_SetPropertyStr(context0, obj, "contact", JS_NewString(context0, contact)); | ||||
| 		JS_SetPropertyStr(context0, obj, "following", JS_TRUE); | ||||
| 		bool stored = false; | ||||
| 		uint8_t private_bin[512] = { 0 }; | ||||
| 		tf_base64_decode(private[i], strlen(private[i]) - strlen(".ed25519"), private_bin, sizeof(private_bin)); | ||||
| 		tf_printf("ssb0 %s following %s\n", self, contact); | ||||
| 		JSValue signed_message = tf_ssb_sign_message(ssb0, self, private_bin, obj, NULL, 0); | ||||
| 		tf_ssb_verify_strip_and_store_message(ssb0, signed_message, _message_stored, &stored); | ||||
| 		JS_FreeValue(context0, signed_message); | ||||
| 		_wait_stored(ssb0, &stored); | ||||
| 		JS_FreeValue(context0, obj); | ||||
|  | ||||
| 		obj = JS_NewObject(context0); | ||||
| 		JS_SetPropertyStr(context0, obj, "type", JS_NewString(context0, "post")); | ||||
| 		JS_SetPropertyStr(context0, obj, "text", JS_NewString(context0, "Hello, world!")); | ||||
| 		JS_SetPropertyStr(context0, obj, "arbitrary_reference", JS_NewString(context0, blob_id)); | ||||
| 		stored = false; | ||||
| 		signed_message = tf_ssb_sign_message(ssb0, self, private_bin, obj, NULL, 0); | ||||
| 		tf_ssb_verify_strip_and_store_message(ssb0, signed_message, _message_stored, &stored); | ||||
| 		JS_FreeValue(context0, signed_message); | ||||
| 		_wait_stored(ssb0, &stored); | ||||
| 		JS_FreeValue(context0, obj); | ||||
| 	} | ||||
|  | ||||
| 	JSContext* context1 = tf_ssb_get_context(ssb1); | ||||
| 	{ | ||||
| 		JSValue obj = JS_NewObject(context1); | ||||
| 		JS_SetPropertyStr(context1, obj, "type", JS_NewString(context1, "contact")); | ||||
| 		char self[k_id_base64_len]; | ||||
| 		tf_string_set(self, sizeof(self), id1); | ||||
| 		char contact[k_id_base64_len]; | ||||
| 		snprintf(contact, sizeof(contact), "@%s", public[0]); | ||||
| 		JS_SetPropertyStr(context1, obj, "contact", JS_NewString(context1, contact)); | ||||
| 		JS_SetPropertyStr(context1, obj, "following", JS_TRUE); | ||||
| 		bool stored = false; | ||||
| 		tf_printf("ssb1 %s following %s\n", self, contact); | ||||
| 		JSValue signed_message = tf_ssb_sign_message(ssb1, self, priv1, obj, NULL, 0); | ||||
| 		tf_ssb_verify_strip_and_store_message(ssb1, signed_message, _message_stored, &stored); | ||||
| 		JS_FreeValue(context1, signed_message); | ||||
| 		_wait_stored(ssb1, &stored); | ||||
| 		JS_FreeValue(context1, obj); | ||||
| 	} | ||||
|  | ||||
| 	tf_printf("Waiting for blob.\n"); | ||||
| 	while (!tf_ssb_db_blob_get(ssb1, blob_id, NULL, NULL)) | ||||
| 	{ | ||||
| 		tf_ssb_set_main_thread(ssb1, true); | ||||
| 		uv_run(&loop, UV_RUN_ONCE); | ||||
| @@ -1379,12 +1553,6 @@ void tf_ssb_test_invite(const tf_test_options_t* options) | ||||
| 	tf_ssb_release_db_writer(ssb0, writer); | ||||
| 	tf_printf("invite: %s\n", invite); | ||||
|  | ||||
| 	int count0 = 0; | ||||
| 	int count1 = 0; | ||||
|  | ||||
| 	tf_ssb_add_message_added_callback(ssb0, _message_added, NULL, &count0); | ||||
| 	tf_ssb_add_message_added_callback(ssb1, _message_added, NULL, &count1); | ||||
|  | ||||
| 	tf_ssb_connect_str(ssb1, invite, 0, NULL, NULL); | ||||
|  | ||||
| 	tf_printf("Waiting for connection.\n"); | ||||
| @@ -1400,11 +1568,19 @@ void tf_ssb_test_invite(const tf_test_options_t* options) | ||||
| 	tf_printf("waiting for messages\n"); | ||||
| 	tf_ssb_set_main_thread(ssb0, true); | ||||
| 	tf_ssb_set_main_thread(ssb1, true); | ||||
| 	while (count0 != 3 || count1 != 3) | ||||
|  | ||||
| 	int32_t sequence0 = 0; | ||||
| 	int32_t sequence1 = 0; | ||||
| 	while (sequence0 != 1 || sequence1 != 2) | ||||
| 	{ | ||||
| 		uv_run(&loop, UV_RUN_ONCE); | ||||
|  | ||||
| 		tf_printf("count0=%d count1=%d\n", count0, count1); | ||||
| 		tf_ssb_set_main_thread(ssb0, false); | ||||
| 		tf_ssb_set_main_thread(ssb1, false); | ||||
| 		tf_ssb_db_get_latest_message_by_author(ssb0, id0, &sequence0, NULL, 0); | ||||
| 		tf_ssb_db_get_latest_message_by_author(ssb1, id1, &sequence1, NULL, 0); | ||||
| 		tf_ssb_set_main_thread(ssb0, true); | ||||
| 		tf_ssb_set_main_thread(ssb1, true); | ||||
| 		tf_printf("sequence0=%d sequence1=%d\n", sequence0, sequence1); | ||||
| 	} | ||||
| 	tf_ssb_set_main_thread(ssb0, false); | ||||
| 	tf_ssb_set_main_thread(ssb1, false); | ||||
|   | ||||
| @@ -66,13 +66,19 @@ void tf_ssb_test_peer_exchange(const tf_test_options_t* options); | ||||
| void tf_ssb_test_publish(const tf_test_options_t* options); | ||||
|  | ||||
| /** | ||||
| ** Test connecting by string. | ||||
| ** Test message and replication. | ||||
| ** @param options The test options. | ||||
| */ | ||||
| void tf_ssb_test_replicate(const tf_test_options_t* options); | ||||
|  | ||||
| /** | ||||
| ** Test invites. | ||||
| ** Test blob replication for a message received while already connected. | ||||
| ** @param options The test options. | ||||
| */ | ||||
| void tf_ssb_test_replicate_blob(const tf_test_options_t* options); | ||||
|  | ||||
| /** | ||||
| ** Test connecting by string. | ||||
| ** @param options The test options. | ||||
| */ | ||||
| void tf_ssb_test_connect_str(const tf_test_options_t* options); | ||||
|   | ||||
							
								
								
									
										19
									
								
								src/task.c
									
									
									
									
									
								
							
							
						
						
									
										19
									
								
								src/task.c
									
									
									
									
									
								
							| @@ -1,20 +1,18 @@ | ||||
| #include "task.h" | ||||
|  | ||||
| #include "api.js.h" | ||||
| #include "bcrypt.js.h" | ||||
| #include "database.js.h" | ||||
| #include "file.js.h" | ||||
| #include "http.h" | ||||
| #include "httpd.js.h" | ||||
| #include "log.h" | ||||
| #include "mem.h" | ||||
| #include "packetstream.h" | ||||
| #include "serialize.h" | ||||
| #include "socket.js.h" | ||||
| #include "ssb.db.h" | ||||
| #include "ssb.h" | ||||
| #include "ssb.js.h" | ||||
| #include "taskstub.js.h" | ||||
| #include "tlscontext.js.h" | ||||
| #include "trace.h" | ||||
| #include "util.js.h" | ||||
| #include "version.h" | ||||
| @@ -28,8 +26,6 @@ | ||||
| #include "uv.h" | ||||
| #include "zlib.h" | ||||
|  | ||||
| #include <openssl/crypto.h> | ||||
|  | ||||
| #include <assert.h> | ||||
| #include <stdio.h> | ||||
| #include <string.h> | ||||
| @@ -709,11 +705,6 @@ static JSValue _tf_task_version(JSContext* context, JSValueConst this_val, int a | ||||
| 	JS_SetPropertyStr(context, version, "name", JS_NewString(context, VERSION_NAME)); | ||||
| 	JS_SetPropertyStr(context, version, "libuv", JS_NewString(context, uv_version_string())); | ||||
| 	JS_SetPropertyStr(context, version, "sqlite", JS_NewString(context, sqlite3_libversion())); | ||||
| #if defined(OPENSSL_VERSION_STRING) | ||||
| 	JS_SetPropertyStr(context, version, "openssl", JS_NewString(context, OpenSSL_version(OPENSSL_VERSION_STRING))); | ||||
| #else | ||||
| 	JS_SetPropertyStr(context, version, "openssl", JS_NewString(context, OpenSSL_version(OPENSSL_VERSION))); | ||||
| #endif | ||||
| 	const char* sodium_version_string(); | ||||
| 	JS_SetPropertyStr(context, version, "c-ares", JS_NewString(context, ares_version(NULL))); | ||||
| 	JS_SetPropertyStr(context, version, "libsodium", JS_NewString(context, sodium_version_string())); | ||||
| @@ -824,12 +815,8 @@ static JSValue _tf_task_getStats(JSContext* context, JSValueConst this_val, int | ||||
| 	JS_SetPropertyStr(context, result, "sqlite3_memory_percent", JS_NewFloat64(context, 100.0 * tf_mem_get_sqlite_malloc_size() / total_memory)); | ||||
| 	JS_SetPropertyStr(context, result, "js_malloc_percent", JS_NewFloat64(context, 100.0 * tf_mem_get_js_malloc_size() / total_memory)); | ||||
| 	JS_SetPropertyStr(context, result, "uv_malloc_percent", JS_NewFloat64(context, 100.0 * tf_mem_get_uv_malloc_size() / total_memory)); | ||||
| 	JS_SetPropertyStr(context, result, "tls_malloc_percent", JS_NewFloat64(context, 100.0 * tf_mem_get_tls_malloc_size() / total_memory)); | ||||
| 	JS_SetPropertyStr(context, result, "tf_malloc_percent", JS_NewFloat64(context, 100.0 * tf_mem_get_tf_malloc_size() / total_memory)); | ||||
|  | ||||
| 	JS_SetPropertyStr(context, result, "socket_count", JS_NewInt32(context, tf_socket_get_count())); | ||||
| 	JS_SetPropertyStr(context, result, "socket_open_count", JS_NewInt32(context, tf_socket_get_open_count())); | ||||
|  | ||||
| 	if (task->_ssb) | ||||
| 	{ | ||||
| 		tf_ssb_stats_t ssb_stats = { 0 }; | ||||
| @@ -1666,8 +1653,6 @@ void tf_task_activate(tf_task_t* task) | ||||
| 		sqlite3_open(task->_db_path, &task->_db); | ||||
|  | ||||
| 		JS_SetPropertyStr(context, global, "Task", tf_taskstub_register(context)); | ||||
| 		JS_SetPropertyStr(context, global, "Socket", tf_socket_register(context)); | ||||
| 		JS_SetPropertyStr(context, global, "TlsContext", tf_tls_context_register(context)); | ||||
| 		tf_file_register(context); | ||||
| 		tf_database_register(context); | ||||
|  | ||||
| @@ -1728,7 +1713,6 @@ void tf_task_activate(tf_task_t* task) | ||||
| 		tf_trace_set_write_callback(task->_trace, _tf_task_trace_to_parent, task); | ||||
| 	} | ||||
|  | ||||
| 	tf_bcrypt_register(context); | ||||
| 	tf_util_register(context); | ||||
| 	JS_SetPropertyStr(context, global, "exit", JS_NewCFunction(context, _tf_task_exit, "exit", 1)); | ||||
| 	JS_SetPropertyStr(context, global, "version", JS_NewCFunction(context, _tf_task_version, "version", 0)); | ||||
| @@ -1884,6 +1868,7 @@ void tf_task_destroy(tf_task_t* task) | ||||
| 			tf_printf("--\n"); | ||||
| 			uv_print_all_handles(&task->_loop, stdout); | ||||
| 		} | ||||
| 		tf_http_debug_destroy(); | ||||
| 		uv_run(&task->_loop, UV_RUN_ONCE); | ||||
| 	} | ||||
| 	if (task->_trace) | ||||
|   | ||||
							
								
								
									
										95
									
								
								src/tests.c
									
									
									
									
									
								
							
							
						
						
									
										95
									
								
								src/tests.c
									
									
									
									
									
								
							| @@ -32,7 +32,7 @@ | ||||
| #include <TargetConditionals.h> | ||||
| #endif | ||||
|  | ||||
| #define TEST_ARGS " --args=ssb_port=0,http_port=0,https_port=0" | ||||
| #define TEST_ARGS " --args=ssb_port=0,http_port=0" | ||||
|  | ||||
| #if !TARGET_OS_IPHONE | ||||
| static void _write_file(const char* path, const char* contents) | ||||
| @@ -549,93 +549,6 @@ static void _test_float(const tf_test_options_t* options) | ||||
| 	unlink("out/child.js"); | ||||
| } | ||||
|  | ||||
| static void _test_socket(const tf_test_options_t* options) | ||||
| { | ||||
| 	_write_file("out/test.js", | ||||
| 		"'use strict';\n" | ||||
| 		"\n" | ||||
| 		"var s = new Socket();\n" | ||||
| 		"print('connecting');\n" | ||||
| 		"print('before connect', s.isConnected);\n" | ||||
| 		"s.onError(function(e) {\n" | ||||
| 		"	print(e);\n" | ||||
| 		"});\n" | ||||
| 		"print('noDelay', s.noDelay);\n" | ||||
| 		"s.noDelay = true;\n" | ||||
| 		"s.connect('www.unprompted.com', 80).then(function() {\n" | ||||
| 		"	print('connected', 'www.unprompted.com', 80, s.isConnected);\n" | ||||
| 		"	print(s.peerName);\n" | ||||
| 		"	s.read(function(data) {\n" | ||||
| 		"		print('read', data ? data.length : null);\n" | ||||
| 		"	});\n" | ||||
| 		"	s.write('GET / HTTP/1.0\\r\\n\\r\\n');\n" | ||||
| 		"}).then(function(e) {\n" | ||||
| 		"	print('closed 1');\n" | ||||
| 		"});\n" | ||||
| 		"\n" | ||||
| 		"var s2 = new Socket();\n" | ||||
| 		"print('connecting');\n" | ||||
| 		"print('before connect', s2.isConnected);\n" | ||||
| 		"s2.onError(function(e) {\n" | ||||
| 		"	print('error');\n" | ||||
| 		"	print(e);\n" | ||||
| 		"});\n" | ||||
| 		"print('noDelay', s2.noDelay);\n" | ||||
| 		"s2.noDelay = true;\n" | ||||
| 		"s2.connect('www.unprompted.com', 443).then(function() {\n" | ||||
| 		"	print('connected', 'www.unprompted.com', 443);\n" | ||||
| 		"	s2.read(function(data) {\n" | ||||
| 		"		print('read', data ? data.length : null);\n" | ||||
| 		"	});\n" | ||||
| 		"	return s2.startTls();\n" | ||||
| 		"}).then(function() {\n" | ||||
| 		"	print('ready');\n" | ||||
| 		"	print(s2.peerName);\n" | ||||
| 		"	s2.write('GET / HTTP/1.0\\r\\nConnection: close\\r\\n\\r\\n').then(function() {\n" | ||||
| 		"		s2.shutdown();\n" | ||||
| 		"	});\n" | ||||
| 		"}).catch(function(e) {\n" | ||||
| 		"	print('caught');\n" | ||||
| 		"	print(e);\n" | ||||
| 		"});\n" | ||||
| 		"var s3 = new Socket();\n" | ||||
| 		"print('connecting s3');\n" | ||||
| 		"print('before connect', s3.isConnected);\n" | ||||
| 		"s3.onError(function(e) {\n" | ||||
| 		"	print('error');\n" | ||||
| 		"	print(e);\n" | ||||
| 		"});\n" | ||||
| 		"print('noDelay', s3.noDelay);\n" | ||||
| 		"s3.noDelay = true;\n" | ||||
| 		"s3.connect('0.0.0.0', 443).then(function() {\n" | ||||
| 		"	print('connected', '0.0.0.0', 443);\n" | ||||
| 		"	s3.read(function(data) {\n" | ||||
| 		"		print('read', data ? data.length : null);\n" | ||||
| 		"	});\n" | ||||
| 		"	return s3.startTls();\n" | ||||
| 		"}).then(function() {\n" | ||||
| 		"	print('ready');\n" | ||||
| 		"	print(s3.peerName);\n" | ||||
| 		"	s3.write('GET / HTTP/1.0\\r\\nConnection: close\\r\\n\\r\\n').then(function() {\n" | ||||
| 		"		s3.shutdown();\n" | ||||
| 		"	});\n" | ||||
| 		"}).catch(function(e) {\n" | ||||
| 		"	print('caught');\n" | ||||
| 		"	print(e);\n" | ||||
| 		"});\n"); | ||||
|  | ||||
| 	char command[256]; | ||||
| 	unlink("out/test_db0.sqlite"); | ||||
| 	snprintf(command, sizeof(command), "%s run --db-path=out/test_db0.sqlite -s out/test.js" TEST_ARGS, options->exe_path); | ||||
| 	tf_printf("%s\n", command); | ||||
| 	int result = system(command); | ||||
| 	tf_printf("returned %d\n", WEXITSTATUS(result)); | ||||
| 	assert(WIFEXITED(result)); | ||||
| 	assert(WEXITSTATUS(result) == 0); | ||||
|  | ||||
| 	unlink("out/test.js"); | ||||
| } | ||||
|  | ||||
| static void _test_file(const tf_test_options_t* options) | ||||
| { | ||||
| 	_write_file("out/test.js", | ||||
| @@ -781,7 +694,7 @@ static void _test_http(const tf_test_options_t* options) | ||||
| 	tf_http_t* http = tf_http_create(&loop); | ||||
| 	tf_http_add_handler(http, "/hello", _test_http_handler, NULL, NULL); | ||||
| 	tf_http_add_handler(http, "/post", _test_http_handler_post, NULL, NULL); | ||||
| 	tf_http_listen(http, 23456, true, NULL, NULL, NULL); | ||||
| 	tf_http_listen(http, 23456, true, NULL, NULL); | ||||
|  | ||||
| 	test_http_t test = { .loop = &loop }; | ||||
| 	uv_async_init(&loop, &test.async, _test_http_async); | ||||
| @@ -873,7 +786,7 @@ static void _test_httpd(const tf_test_options_t* options) | ||||
| 	uv_spawn(&loop, &process, | ||||
| 		&(uv_process_options_t) { | ||||
| 			.file = options->exe_path, | ||||
| 			.args = (char*[]) { (char*)options->exe_path, "run", "--db-path=out/test_db0.sqlite", "--args=ssb_port=0,http_port=8080,https_port=0", NULL }, | ||||
| 			.args = (char*[]) { (char*)options->exe_path, "run", "--db-path=out/test_db0.sqlite", "--args=ssb_port=0,http_port=8080", NULL }, | ||||
| 			.stdio_count = sizeof(stdio) / sizeof(*stdio), | ||||
| 			.stdio = stdio, | ||||
| 		}); | ||||
| @@ -1065,7 +978,6 @@ void tf_tests(const tf_test_options_t* options) | ||||
| 	_tf_test_run(options, "icu", _test_icu, false); | ||||
| 	_tf_test_run(options, "uint8array", _test_uint8array, false); | ||||
| 	_tf_test_run(options, "float", _test_float, false); | ||||
| 	_tf_test_run(options, "socket", _test_socket, false); | ||||
| 	_tf_test_run(options, "file", _test_file, false); | ||||
| 	_tf_test_run(options, "b64", _test_b64, false); | ||||
| 	_tf_test_run(options, "rooms", tf_ssb_test_rooms, false); | ||||
| @@ -1076,6 +988,7 @@ void tf_tests(const tf_test_options_t* options) | ||||
| 	_tf_test_run(options, "peer_exchange", tf_ssb_test_peer_exchange, false); | ||||
| 	_tf_test_run(options, "publish", tf_ssb_test_publish, false); | ||||
| 	_tf_test_run(options, "replicate", tf_ssb_test_replicate, false); | ||||
| 	_tf_test_run(options, "replicate_blob", tf_ssb_test_replicate_blob, false); | ||||
| 	_tf_test_run(options, "connect_str", tf_ssb_test_connect_str, false); | ||||
| 	_tf_test_run(options, "invite", tf_ssb_test_invite, false); | ||||
| 	_tf_test_run(options, "triggers", tf_ssb_test_triggers, false); | ||||
|   | ||||
							
								
								
									
										384
									
								
								src/tls.c
									
									
									
									
									
								
							
							
						
						
									
										384
									
								
								src/tls.c
									
									
									
									
									
								
							| @@ -1,384 +0,0 @@ | ||||
| #include "tls.h" | ||||
|  | ||||
| #include "mem.h" | ||||
|  | ||||
| #include <string.h> | ||||
|  | ||||
| #include <openssl/bio.h> | ||||
| #include <openssl/err.h> | ||||
| #include <openssl/pem.h> | ||||
| #include <openssl/ssl.h> | ||||
| #include <openssl/x509v3.h> | ||||
|  | ||||
| typedef enum _direction_t | ||||
| { | ||||
| 	k_direction_undetermined, | ||||
| 	k_direction_accept, | ||||
| 	k_direction_connect, | ||||
| } direction_t; | ||||
|  | ||||
| typedef struct _tf_tls_context_t | ||||
| { | ||||
| 	SSL_CTX* context; | ||||
| } tf_tls_context_t; | ||||
|  | ||||
| typedef struct _tf_tls_session_t | ||||
| { | ||||
| 	tf_tls_context_t* context; | ||||
| 	BIO* bio_in; | ||||
| 	BIO* bio_out; | ||||
| 	SSL* ssl; | ||||
| 	const char* hostname; | ||||
| 	direction_t direction; | ||||
| } tf_tls_session_t; | ||||
|  | ||||
| tf_tls_context_t* tf_tls_context_create() | ||||
| { | ||||
| 	tf_tls_context_t* context = tf_malloc(sizeof(tf_tls_context_t)); | ||||
| 	memset(context, 0, sizeof(*context)); | ||||
| 	OPENSSL_init_ssl(0, NULL); | ||||
| 	context->context = SSL_CTX_new(SSLv23_method()); | ||||
| 	SSL_CTX_set_default_verify_paths(context->context); | ||||
| 	return context; | ||||
| } | ||||
|  | ||||
| bool tf_tls_context_set_certificate(tf_tls_context_t* context, const char* certificate) | ||||
| { | ||||
| 	int result = 0; | ||||
| 	BIO* bio = BIO_new(BIO_s_mem()); | ||||
| 	BIO_puts(bio, certificate); | ||||
| 	X509* x509 = PEM_read_bio_X509(bio, 0, 0, 0); | ||||
| 	result = SSL_CTX_use_certificate(context->context, x509); | ||||
| 	X509_free(x509); | ||||
| 	while (true) | ||||
| 	{ | ||||
| 		x509 = PEM_read_bio_X509(bio, 0, 0, 0); | ||||
| 		if (x509) | ||||
| 		{ | ||||
| 			SSL_CTX_add_extra_chain_cert(context->context, x509); | ||||
| 			/* Docs say don't x509_free: https://www.openssl.org/docs/man3.2/man3/SSL_CTX_add_extra_chain_cert.html. */ | ||||
| 		} | ||||
| 		else | ||||
| 		{ | ||||
| 			break; | ||||
| 		} | ||||
| 	} | ||||
| 	BIO_free(bio); | ||||
| 	return result == 1; | ||||
| } | ||||
|  | ||||
| bool tf_tls_context_set_private_key(tf_tls_context_t* context, const char* private_key) | ||||
| { | ||||
| 	int result = 0; | ||||
| 	BIO* bio = BIO_new(BIO_s_mem()); | ||||
| 	BIO_puts(bio, private_key); | ||||
| 	EVP_PKEY* key = PEM_read_bio_PrivateKey(bio, 0, 0, 0); | ||||
| 	result = SSL_CTX_use_PrivateKey(context->context, key); | ||||
| 	EVP_PKEY_free(key); | ||||
| 	BIO_free(bio); | ||||
| 	return result == 1; | ||||
| } | ||||
|  | ||||
| bool tf_tls_context_add_trusted_certificate(tf_tls_context_t* context, const char* certificate) | ||||
| { | ||||
| 	bool result = false; | ||||
| 	BIO* bio = BIO_new_mem_buf(certificate, -1); | ||||
| 	X509* x509 = PEM_read_bio_X509(bio, 0, 0, 0); | ||||
| 	BIO_free(bio); | ||||
|  | ||||
| 	if (x509) | ||||
| 	{ | ||||
| 		X509_STORE* store = SSL_CTX_get_cert_store(context->context); | ||||
| 		if (store && X509_STORE_add_cert(store, x509) == 1) | ||||
| 		{ | ||||
| 			result = true; | ||||
| 		} | ||||
| 		X509_free(x509); | ||||
| 	} | ||||
|  | ||||
| 	return result; | ||||
| } | ||||
|  | ||||
| tf_tls_session_t* tf_tls_context_create_session(tf_tls_context_t* context) | ||||
| { | ||||
| 	tf_tls_session_t* session = tf_malloc(sizeof(tf_tls_session_t)); | ||||
| 	memset(session, 0, sizeof(*session)); | ||||
| 	session->context = context; | ||||
| 	session->bio_in = BIO_new(BIO_s_mem()); | ||||
| 	session->bio_out = BIO_new(BIO_s_mem()); | ||||
| 	return session; | ||||
| } | ||||
|  | ||||
| void tf_tls_context_destroy(tf_tls_context_t* context) | ||||
| { | ||||
| 	SSL_CTX_free(context->context); | ||||
| 	OPENSSL_cleanup(); | ||||
| 	tf_free(context); | ||||
| } | ||||
|  | ||||
| void tf_tls_session_destroy(tf_tls_session_t* session) | ||||
| { | ||||
| 	if (session->ssl) | ||||
| 	{ | ||||
| 		SSL_free(session->ssl); | ||||
| 	} | ||||
| 	if (session->hostname) | ||||
| 	{ | ||||
| 		tf_free((void*)session->hostname); | ||||
| 	} | ||||
| 	tf_free(session); | ||||
| } | ||||
|  | ||||
| void tf_tls_session_set_hostname(tf_tls_session_t* session, const char* hostname) | ||||
| { | ||||
| 	if (session->hostname) | ||||
| 	{ | ||||
| 		tf_free((void*)session->hostname); | ||||
| 		session->hostname = NULL; | ||||
| 	} | ||||
| 	if (hostname) | ||||
| 	{ | ||||
| 		session->hostname = tf_strdup(hostname); | ||||
| 	} | ||||
| } | ||||
|  | ||||
| void tf_tls_session_start_accept(tf_tls_session_t* session) | ||||
| { | ||||
| 	session->direction = k_direction_accept; | ||||
| 	session->ssl = SSL_new(session->context->context); | ||||
| 	SSL_set_bio(session->ssl, session->bio_in, session->bio_out); | ||||
| 	SSL_accept(session->ssl); | ||||
| 	tf_tls_session_handshake(session); | ||||
| } | ||||
|  | ||||
| void tf_tls_session_start_connect(tf_tls_session_t* session) | ||||
| { | ||||
| 	session->direction = k_direction_connect; | ||||
| 	session->ssl = SSL_new(session->context->context); | ||||
| 	X509_VERIFY_PARAM* param = SSL_get0_param(session->ssl); | ||||
| 	X509_VERIFY_PARAM_set_hostflags(param, X509_CHECK_FLAG_NO_PARTIAL_WILDCARDS); | ||||
| 	X509_VERIFY_PARAM_set1_host(param, session->hostname, 0); | ||||
| 	SSL_set_tlsext_host_name(session->ssl, session->hostname); | ||||
| 	SSL_set_bio(session->ssl, session->bio_in, session->bio_out); | ||||
| 	SSL_connect(session->ssl); | ||||
| 	tf_tls_session_handshake(session); | ||||
| } | ||||
|  | ||||
| void tf_tls_session_shutdown(tf_tls_session_t* session) | ||||
| { | ||||
| 	SSL_shutdown(session->ssl); | ||||
| } | ||||
|  | ||||
| int tf_tls_session_get_peer_certificate(tf_tls_session_t* session, char* buffer, size_t bytes) | ||||
| { | ||||
| 	int result = -1; | ||||
| #if OPENSSL_VERSION_NUMBER < 0x30000000L | ||||
| 	X509* certificate = SSL_get_peer_certificate(session->ssl); | ||||
| #else | ||||
| 	X509* certificate = SSL_get1_peer_certificate(session->ssl); | ||||
| #endif | ||||
| 	BIO* bio = BIO_new(BIO_s_mem()); | ||||
| 	PEM_write_bio_X509(bio, certificate); | ||||
| 	X509_free(certificate); | ||||
| 	BUF_MEM* mem; | ||||
| 	BIO_get_mem_ptr(bio, &mem); | ||||
| 	if (mem->length <= bytes) | ||||
| 	{ | ||||
| 		memcpy(buffer, mem->data, mem->length); | ||||
| 		result = mem->length; | ||||
| 	} | ||||
| 	BIO_free(bio); | ||||
| 	return result; | ||||
| } | ||||
|  | ||||
| #if OPENSSL_VERSION_NUMBER < 0x10100000L | ||||
| static bool _tls_session_wildcard_match(const char* pattern, size_t pattern_length, const char* name) | ||||
| { | ||||
| 	const char* it = pattern; | ||||
| 	while (it - pattern < pattern_length && *name) | ||||
| 	{ | ||||
| 		if (*it == '*') | ||||
| 		{ | ||||
| 			for (const char* p = name; *p; ++p) | ||||
| 			{ | ||||
| 				if (_tls_session_wildcard_match(it + 1, pattern_length - 1, p)) | ||||
| 				{ | ||||
| 					return true; | ||||
| 				} | ||||
| 			} | ||||
| 			return false; | ||||
| 		} | ||||
| 		else if (tolower(*it) == tolower(*name)) | ||||
| 		{ | ||||
| 			++it; | ||||
| 			++name; | ||||
| 		} | ||||
| 		else | ||||
| 		{ | ||||
| 			break; | ||||
| 		} | ||||
| 	} | ||||
| 	return it - pattern <= pattern_length && *name == 0; | ||||
| } | ||||
|  | ||||
| static bool _tls_session_verify_hostname(X509* certificate, const char* hostname) | ||||
| { | ||||
| 	bool verified = false; | ||||
| 	void* names = X509_get_ext_d2i(certificate, NID_subject_alt_name, 0, 0); | ||||
| 	if (names) | ||||
| 	{ | ||||
| 		int count = sk_GENERAL_NAME_num(names); | ||||
| 		for (int i = 0; i < count; ++i) | ||||
| 		{ | ||||
| 			const GENERAL_NAME* check = sk_GENERAL_NAME_value(names, i); | ||||
| 			if (!verified) | ||||
| 			{ | ||||
| #if OPENSSL_VERSION_NUMBER <= 0x1000211fL | ||||
| 				const unsigned char* name = ASN1_STRING_data(check->d.ia5); | ||||
| #else | ||||
| 				const char* name = ASN1_STRING_get0_data(check->d.ia5); | ||||
| #endif | ||||
| 				size_t length = ASN1_STRING_length(check->d.ia5); | ||||
| 				if (_tls_session_wildcard_match((const char*)name, length, hostname)) | ||||
| 				{ | ||||
| 					verified = true; | ||||
| 				} | ||||
| 			} | ||||
| 		} | ||||
| 		sk_GENERAL_NAMES_free(names); | ||||
| 	} | ||||
|  | ||||
| 	if (!verified) | ||||
| 	{ | ||||
| 		int index = X509_NAME_get_index_by_NID(X509_get_subject_name(certificate), NID_commonName, -1); | ||||
| 		if (index >= 0) | ||||
| 		{ | ||||
| 			X509_NAME_ENTRY* entry = X509_NAME_get_entry(X509_get_subject_name(certificate), index); | ||||
| 			if (entry) | ||||
| 			{ | ||||
| 				ASN1_STRING* asn1 = X509_NAME_ENTRY_get_data(entry); | ||||
| 				if (asn1) | ||||
| 				{ | ||||
| #if OPENSSL_VERSION_NUMBER <= 0x1000211fL | ||||
| 					const unsigned char* commonName = ASN1_STRING_data(asn1); | ||||
| #else | ||||
| 					const char* commonName = ASN1_STRING_get0_data(asn1); | ||||
| #endif | ||||
| 					if ((size_t)(ASN1_STRING_length(asn1)) == strlen((const char*)commonName)) | ||||
| 					{ | ||||
| 						verified = _tls_session_wildcard_match((const char*)commonName, ASN1_STRING_length(asn1), hostname); | ||||
| 					} | ||||
| 				} | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	return verified; | ||||
| } | ||||
| #endif | ||||
|  | ||||
| static bool _tls_session_verify_peer_certificate(tf_tls_session_t* session) | ||||
| { | ||||
| 	bool verified = false; | ||||
| #if OPENSSL_VERSION_NUMBER < 0x30000000L | ||||
| 	X509* certificate = SSL_get_peer_certificate(session->ssl); | ||||
| #else | ||||
| 	X509* certificate = SSL_get1_peer_certificate(session->ssl); | ||||
| #endif | ||||
| 	if (certificate) | ||||
| 	{ | ||||
| 		if (SSL_get_verify_result(session->ssl) == X509_V_OK) | ||||
| 		{ | ||||
| #if OPENSSL_VERSION_NUMBER < 0x10100000L | ||||
| 			if (_tls_session_verify_hostname(certificate, session->hostname)) | ||||
| 			{ | ||||
| 				verified = true; | ||||
| 			} | ||||
| #else | ||||
| 			verified = true; | ||||
| #endif | ||||
| 		} | ||||
| 		X509_free(certificate); | ||||
| 	} | ||||
| 	return verified; | ||||
| } | ||||
|  | ||||
| tf_tls_handshake_t tf_tls_session_handshake(tf_tls_session_t* session) | ||||
| { | ||||
| 	tf_tls_handshake_t result = k_tls_handshake_done; | ||||
| 	if (!SSL_is_init_finished(session->ssl)) | ||||
| 	{ | ||||
| 		int value = SSL_do_handshake(session->ssl); | ||||
| 		if (value <= 0) | ||||
| 		{ | ||||
| 			int error = SSL_get_error(session->ssl, value); | ||||
| 			if (error != SSL_ERROR_WANT_READ && error != SSL_ERROR_WANT_WRITE) | ||||
| 			{ | ||||
| 				result = k_tls_handshake_failed; | ||||
| 			} | ||||
| 			else | ||||
| 			{ | ||||
| 				result = k_tls_handshake_more; | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| 	if (result == k_tls_handshake_done && session->direction == k_direction_connect && !_tls_session_verify_peer_certificate(session)) | ||||
| 	{ | ||||
| 		result = k_tls_handshake_failed; | ||||
| 	} | ||||
| 	return result; | ||||
| } | ||||
|  | ||||
| int tf_tls_session_read_plain(tf_tls_session_t* session, char* buffer, size_t bytes) | ||||
| { | ||||
| 	int result = SSL_read(session->ssl, buffer, bytes); | ||||
| 	if (result <= 0) | ||||
| 	{ | ||||
| 		int error = SSL_get_error(session->ssl, result); | ||||
| 		if (error == SSL_ERROR_WANT_READ || error == SSL_ERROR_WANT_WRITE) | ||||
| 		{ | ||||
| 			result = 0; | ||||
| 		} | ||||
| 		else if (error == SSL_ERROR_ZERO_RETURN) | ||||
| 		{ | ||||
| 			if ((SSL_get_shutdown(session->ssl) & SSL_RECEIVED_SHUTDOWN) != 0) | ||||
| 			{ | ||||
| 				result = k_tls_read_zero; | ||||
| 			} | ||||
| 			else | ||||
| 			{ | ||||
| 				result = 0; | ||||
| 			} | ||||
| 		} | ||||
| 		else | ||||
| 		{ | ||||
| 			result = k_tls_read_failed; | ||||
| 		} | ||||
| 	} | ||||
| 	return result; | ||||
| } | ||||
|  | ||||
| int tf_tls_session_write_plain(tf_tls_session_t* session, const char* buffer, size_t bytes) | ||||
| { | ||||
| 	return SSL_write(session->ssl, buffer, bytes); | ||||
| } | ||||
|  | ||||
| int tf_tls_session_read_encrypted(tf_tls_session_t* session, char* buffer, size_t bytes) | ||||
| { | ||||
| 	return BIO_read(session->bio_out, buffer, bytes); | ||||
| } | ||||
|  | ||||
| int tf_tls_session_write_encrypted(tf_tls_session_t* session, const char* buffer, size_t bytes) | ||||
| { | ||||
| 	return BIO_write(session->bio_in, buffer, bytes); | ||||
| } | ||||
|  | ||||
| bool tf_tls_session_get_error(tf_tls_session_t* session, char* buffer, size_t bytes) | ||||
| { | ||||
| 	unsigned long error = ERR_get_error(); | ||||
| 	if (error != 0) | ||||
| 	{ | ||||
| 		ERR_error_string_n(error, buffer, bytes); | ||||
| 	} | ||||
| 	return error != 0; | ||||
| } | ||||
							
								
								
									
										177
									
								
								src/tls.h
									
									
									
									
									
								
							
							
						
						
									
										177
									
								
								src/tls.h
									
									
									
									
									
								
							| @@ -1,177 +0,0 @@ | ||||
| #pragma once | ||||
|  | ||||
| /** | ||||
| ** \defgroup tls TLS | ||||
| ** A minimal wrapper around OpenSSL. | ||||
| ** @{ | ||||
| */ | ||||
|  | ||||
| #include <stdbool.h> | ||||
| #include <stddef.h> | ||||
|  | ||||
| /** | ||||
| ** A TLS context.  May have many tf_tls_session_t instances. | ||||
| */ | ||||
| typedef struct _tf_tls_context_t tf_tls_context_t; | ||||
|  | ||||
| /** | ||||
| ** A TLS session.  Belongs to one tf_tls_context_t and represents a single connection. | ||||
| */ | ||||
| typedef struct _tf_tls_session_t tf_tls_session_t; | ||||
|  | ||||
| /** | ||||
| ** The state of a TLS handshake. | ||||
| */ | ||||
| typedef enum _tf_tls_handshake_t | ||||
| { | ||||
| 	k_tls_handshake_done, | ||||
| 	k_tls_handshake_more, | ||||
| 	k_tls_handshake_failed, | ||||
| } tf_tls_handshake_t; | ||||
|  | ||||
| /** | ||||
| ** Possible error statuses from tf_tls_session_read_plain. | ||||
| */ | ||||
| typedef enum _tf_tls_read_t | ||||
| { | ||||
| 	k_tls_read_zero = -1, | ||||
| 	k_tls_read_failed = -2, | ||||
| } tf_tls_read_t; | ||||
|  | ||||
| /** | ||||
| ** Create a TLS context.  Clean up with tf_tls_context_destroy(). | ||||
| ** @return A new TLS context. | ||||
| */ | ||||
| tf_tls_context_t* tf_tls_context_create(); | ||||
|  | ||||
| /** | ||||
| ** Set the TLS context's server certificate. | ||||
| ** @param context The TLS context. | ||||
| ** @param certificate The certificate in PEM format. | ||||
| ** @return true if set successfully. | ||||
| */ | ||||
| bool tf_tls_context_set_certificate(tf_tls_context_t* context, const char* certificate); | ||||
|  | ||||
| /** | ||||
| ** Set the TLS context's server certificate's private key. | ||||
| ** @param context The TLS context. | ||||
| ** @param private_key The private key in PEM format. | ||||
| ** @return true if set successfully. | ||||
| */ | ||||
| bool tf_tls_context_set_private_key(tf_tls_context_t* context, const char* private_key); | ||||
|  | ||||
| /** | ||||
| ** Add a trusted certificate. | ||||
| ** @param context The TLS context. | ||||
| ** @param certificate The certificate in PEM format. | ||||
| ** @return true if the certificate was added to the trusted list successfully. | ||||
| */ | ||||
| bool tf_tls_context_add_trusted_certificate(tf_tls_context_t* context, const char* certificate); | ||||
|  | ||||
| /** | ||||
| ** Create a TLS session from a context.  Once created, call | ||||
| ** tf_tls_session_handshake() until it returns k_tls_handshake_done.  Call | ||||
| ** tf_tls_session_[read/write]_[plain/encrypted]() as data is available. | ||||
| ** @param context The TLS context.  Clean up with tf_tls_session_destroy(). | ||||
| ** @return A new TLS session. | ||||
| */ | ||||
| tf_tls_session_t* tf_tls_context_create_session(tf_tls_context_t* context); | ||||
|  | ||||
| /** | ||||
| ** Destroy a TLS context. | ||||
| ** @param context The TLS contextx created by tf_tls_context_create(). | ||||
| */ | ||||
| void tf_tls_context_destroy(tf_tls_context_t* context); | ||||
|  | ||||
| /** | ||||
| ** Destroy a TLS session. | ||||
| ** @param session A TLS sesssion created by tf_tls_context_create_session(). | ||||
| */ | ||||
| void tf_tls_session_destroy(tf_tls_session_t* session); | ||||
|  | ||||
| /** | ||||
| ** Set the remote hostname for a session. | ||||
| ** @param session The TLS session. | ||||
| ** @param hostname The hostname. | ||||
| */ | ||||
| void tf_tls_session_set_hostname(tf_tls_session_t* session, const char* hostname); | ||||
|  | ||||
| /** | ||||
| ** Begin an outgoing TLS session. | ||||
| ** @param session The TLS session. | ||||
| */ | ||||
| void tf_tls_session_start_accept(tf_tls_session_t* session); | ||||
|  | ||||
| /** | ||||
| ** Begin an incoming TLS session. | ||||
| ** @param session The TLS session. | ||||
| */ | ||||
| void tf_tls_session_start_connect(tf_tls_session_t* session); | ||||
|  | ||||
| /** | ||||
| ** Begin the clean shutdown process for a TLS session. | ||||
| ** @param session The TLS session. | ||||
| */ | ||||
| void tf_tls_session_shutdown(tf_tls_session_t* session); | ||||
|  | ||||
| /** | ||||
| ** Get the certificate from the remote end of a TLS session if available. | ||||
| ** @param session The TLS session. | ||||
| ** @param buffer A buffer to receive the certificate. | ||||
| ** @param bytes The size of the buffer. | ||||
| ** @return The size of the returned certificate, or -1 on failure. | ||||
| */ | ||||
| int tf_tls_session_get_peer_certificate(tf_tls_session_t* session, char* buffer, size_t bytes); | ||||
|  | ||||
| /** | ||||
| ** Update the TLS handshake.  Call repeatedly as new data is available until it returns done. | ||||
| ** @param session The TLS session. | ||||
| ** @return The current state of the handshake process. | ||||
| */ | ||||
| tf_tls_handshake_t tf_tls_session_handshake(tf_tls_session_t* session); | ||||
|  | ||||
| /** | ||||
| ** Read decrypted data from the TLS session. | ||||
| ** @param session The TLS session. | ||||
| ** @param buffer A buffer to receive the data. | ||||
| ** @param bytes The size of the buffer. | ||||
| ** @return The number of bytes returned. | ||||
| */ | ||||
| int tf_tls_session_read_plain(tf_tls_session_t* session, char* buffer, size_t bytes); | ||||
|  | ||||
| /** | ||||
| ** Write unencrypted data to the TLS session. | ||||
| ** @param session The TLS session. | ||||
| ** @param buffer The data to encrypt. | ||||
| ** @param bytes The size of the data. | ||||
| ** @return 1 on success, 0 on failure. | ||||
| */ | ||||
| int tf_tls_session_write_plain(tf_tls_session_t* session, const char* buffer, size_t bytes); | ||||
|  | ||||
| /** | ||||
| ** Read encrypted data from the TLS session that needs to be sent. | ||||
| ** @param session The TLS session. | ||||
| ** @param buffer A buffer to receive the data. | ||||
| ** @param bytes The size of the buffer. | ||||
| ** @return The number of bytes returned. | ||||
| */ | ||||
| int tf_tls_session_read_encrypted(tf_tls_session_t* session, char* buffer, size_t bytes); | ||||
|  | ||||
| /** | ||||
| ** Write encrypted data to the TLS session. | ||||
| ** @param session The TLS session. | ||||
| ** @param buffer The encrypted data. | ||||
| ** @param bytes The number of bytes written. | ||||
| */ | ||||
| int tf_tls_session_write_encrypted(tf_tls_session_t* session, const char* buffer, size_t bytes); | ||||
|  | ||||
| /** | ||||
| ** Retrieve the last error from a TLS session. | ||||
| ** @param session The TLS session. | ||||
| ** @param buffer A buffer to receive the error text. | ||||
| ** @param bytes The size of the buffer. | ||||
| ** @return true if an error was retrieved. | ||||
| */ | ||||
| bool tf_tls_session_get_error(tf_tls_session_t* session, char* buffer, size_t bytes); | ||||
|  | ||||
| /** @} */ | ||||
| @@ -1,105 +0,0 @@ | ||||
| #include "tlscontext.js.h" | ||||
|  | ||||
| #include "log.h" | ||||
| #include "mem.h" | ||||
| #include "task.h" | ||||
| #include "tls.h" | ||||
|  | ||||
| #include <stdlib.h> | ||||
| #include <string.h> | ||||
|  | ||||
| static JSClassID _classId; | ||||
| static int _count; | ||||
|  | ||||
| typedef struct _tf_tls_context_t | ||||
| { | ||||
| 	tf_tls_context_t* context; | ||||
| 	tf_task_t* task; | ||||
| 	JSValue object; | ||||
| } tf_tls_context_t; | ||||
|  | ||||
| static JSValue _tls_context_create(JSContext* context, JSValueConst this_val, int argc, JSValueConst* argv); | ||||
| static void _tls_context_finalizer(JSRuntime* runtime, JSValue value); | ||||
|  | ||||
| static JSValue _tls_context_set_certificate(JSContext* context, JSValueConst this_val, int argc, JSValueConst* argv) | ||||
| { | ||||
| 	tf_tls_context_t* tls = JS_GetOpaque(this_val, _classId); | ||||
| 	const char* value = JS_ToCString(context, argv[0]); | ||||
| 	tf_tls_context_set_certificate(tls->context, value); | ||||
| 	JS_FreeCString(context, value); | ||||
| 	return JS_UNDEFINED; | ||||
| } | ||||
|  | ||||
| static JSValue _tls_context_set_private_key(JSContext* context, JSValueConst this_val, int argc, JSValueConst* argv) | ||||
| { | ||||
| 	tf_tls_context_t* tls = JS_GetOpaque(this_val, _classId); | ||||
| 	const char* value = JS_ToCString(context, argv[0]); | ||||
| 	tf_tls_context_set_private_key(tls->context, value); | ||||
| 	JS_FreeCString(context, value); | ||||
| 	return JS_UNDEFINED; | ||||
| } | ||||
|  | ||||
| static JSValue _tls_context_add_trusted_certificate(JSContext* context, JSValueConst this_val, int argc, JSValueConst* argv) | ||||
| { | ||||
| 	tf_tls_context_t* tls = JS_GetOpaque(this_val, _classId); | ||||
| 	const char* value = JS_ToCString(context, argv[0]); | ||||
| 	tf_tls_context_add_trusted_certificate(tls->context, value); | ||||
| 	JS_FreeCString(context, value); | ||||
| 	return JS_UNDEFINED; | ||||
| } | ||||
|  | ||||
| JSValue tf_tls_context_register(JSContext* context) | ||||
| { | ||||
| 	JS_NewClassID(&_classId); | ||||
| 	JSClassDef def = { | ||||
| 		.class_name = "TlsContext", | ||||
| 		.finalizer = _tls_context_finalizer, | ||||
| 	}; | ||||
| 	if (JS_NewClass(JS_GetRuntime(context), _classId, &def) != 0) | ||||
| 	{ | ||||
| 		fprintf(stderr, "Failed to register TlsContext.\n"); | ||||
| 	} | ||||
| 	return JS_NewCFunction2(context, _tls_context_create, "TlsContext", 0, JS_CFUNC_constructor, 0); | ||||
| } | ||||
|  | ||||
| tf_tls_context_t* tf_tls_context_get(JSValue value) | ||||
| { | ||||
| 	tf_tls_context_t* tls = JS_GetOpaque(value, _classId); | ||||
| 	return tls ? tls->context : NULL; | ||||
| } | ||||
|  | ||||
| int tf_tls_context_get_count() | ||||
| { | ||||
| 	return _count; | ||||
| } | ||||
|  | ||||
| static JSValue _tls_context_create(JSContext* context, JSValueConst this_val, int argc, JSValueConst* argv) | ||||
| { | ||||
| 	tf_tls_context_t* tls = tf_malloc(sizeof(tf_tls_context_t)); | ||||
| 	memset(tls, 0, sizeof(*tls)); | ||||
|  | ||||
| 	++_count; | ||||
| 	tls->object = JS_NewObjectClass(context, _classId); | ||||
| 	JS_SetOpaque(tls->object, tls); | ||||
|  | ||||
| 	JS_SetPropertyStr(context, tls->object, "setCertificate", JS_NewCFunction(context, _tls_context_set_certificate, "setCertificate", 1)); | ||||
| 	JS_SetPropertyStr(context, tls->object, "setPrivateKey", JS_NewCFunction(context, _tls_context_set_private_key, "setPrivateKey", 1)); | ||||
| 	JS_SetPropertyStr(context, tls->object, "addTrustedCertificate", JS_NewCFunction(context, _tls_context_add_trusted_certificate, "addTrustedCertificate", 1)); | ||||
|  | ||||
| 	tls->context = tf_tls_context_create(); | ||||
| 	tls->task = tf_task_get(context); | ||||
|  | ||||
| 	return tls->object; | ||||
| } | ||||
|  | ||||
| static void _tls_context_finalizer(JSRuntime* runtime, JSValue value) | ||||
| { | ||||
| 	tf_tls_context_t* tls = JS_GetOpaque(value, _classId); | ||||
| 	if (tls->context) | ||||
| 	{ | ||||
| 		tf_tls_context_destroy(tls->context); | ||||
| 		tls->context = NULL; | ||||
| 	} | ||||
| 	--_count; | ||||
| 	tf_free(tls); | ||||
| } | ||||
| @@ -1,37 +0,0 @@ | ||||
| #pragma once | ||||
|  | ||||
| /** | ||||
| ** \defgroup tls_js TLS Interface | ||||
| ** Exposes \ref tls to JS. | ||||
| ** @{ | ||||
| */ | ||||
|  | ||||
| #include "quickjs.h" | ||||
|  | ||||
| /** | ||||
| ** A TLS context instance. | ||||
| */ | ||||
| typedef struct _tf_tls_context_t tf_tls_context_t; | ||||
|  | ||||
| /** | ||||
| ** Register TLS script interface. | ||||
| ** @param context The TLS context. | ||||
| ** @return the TlsContext constructor. | ||||
| */ | ||||
| JSValue tf_tls_context_register(JSContext* context); | ||||
|  | ||||
| /** | ||||
| ** Get a TLS context instance from its JS object. | ||||
| ** @param value A TlsContext JS object. | ||||
| ** @return The corresponding instance. | ||||
| */ | ||||
| tf_tls_context_t* tf_tls_context_get(JSValue value); | ||||
|  | ||||
| /** | ||||
| ** Get the number of active TLS context instances. | ||||
| ** @return The number of TlsContext objects created that have not been | ||||
| ** finalized. | ||||
| */ | ||||
| int tf_tls_context_get_count(); | ||||
|  | ||||
| /** @} */ | ||||
| @@ -253,66 +253,6 @@ bool tf_util_report_error(JSContext* context, JSValue value) | ||||
| 	return is_error; | ||||
| } | ||||
|  | ||||
| static JSValue _util_parseHttpResponse(JSContext* context, JSValueConst this_val, int argc, JSValueConst* argv) | ||||
| { | ||||
| 	JSValue result = JS_UNDEFINED; | ||||
| 	int status = 0; | ||||
| 	int minor_version = 0; | ||||
| 	const char* message = NULL; | ||||
| 	size_t message_length = 0; | ||||
| 	struct phr_header headers[100]; | ||||
| 	size_t header_count = sizeof(headers) / sizeof(*headers); | ||||
| 	int previous_length = 0; | ||||
| 	JS_ToInt32(context, &previous_length, argv[1]); | ||||
|  | ||||
| 	JSValue buffer = JS_UNDEFINED; | ||||
| 	size_t length; | ||||
| 	uint8_t* array = tf_util_try_get_array_buffer(context, &length, argv[0]); | ||||
| 	if (!array) | ||||
| 	{ | ||||
| 		size_t offset; | ||||
| 		size_t element_size; | ||||
| 		buffer = tf_util_try_get_typed_array_buffer(context, argv[0], &offset, &length, &element_size); | ||||
| 		if (!JS_IsException(buffer)) | ||||
| 		{ | ||||
| 			array = tf_util_try_get_array_buffer(context, &length, buffer); | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	if (array) | ||||
| 	{ | ||||
| 		int parse_result = phr_parse_response((const char*)array, length, &minor_version, &status, &message, &message_length, headers, &header_count, previous_length); | ||||
| 		if (parse_result > 0) | ||||
| 		{ | ||||
| 			result = JS_NewObject(context); | ||||
| 			JS_SetPropertyStr(context, result, "bytes_parsed", JS_NewInt32(context, parse_result)); | ||||
| 			JS_SetPropertyStr(context, result, "minor_version", JS_NewInt32(context, minor_version)); | ||||
| 			JS_SetPropertyStr(context, result, "status", JS_NewInt32(context, status)); | ||||
| 			JS_SetPropertyStr(context, result, "message", JS_NewStringLen(context, message, message_length)); | ||||
| 			JSValue header_object = JS_NewObject(context); | ||||
| 			for (int i = 0; i < (int)header_count; i++) | ||||
| 			{ | ||||
| 				char name[256]; | ||||
| 				snprintf(name, sizeof(name), "%.*s", (int)headers[i].name_len, headers[i].name); | ||||
| 				JS_SetPropertyStr(context, header_object, name, JS_NewStringLen(context, headers[i].value, headers[i].value_len)); | ||||
| 			} | ||||
| 			JS_SetPropertyStr(context, result, "headers", header_object); | ||||
| 		} | ||||
| 		else | ||||
| 		{ | ||||
| 			result = JS_NewInt32(context, parse_result); | ||||
| 		} | ||||
| 	} | ||||
| 	else | ||||
| 	{ | ||||
| 		result = JS_ThrowTypeError(context, "Could not convert argument to array."); | ||||
| 	} | ||||
|  | ||||
| 	JS_FreeValue(context, buffer); | ||||
|  | ||||
| 	return result; | ||||
| } | ||||
|  | ||||
| static const char* k_kind_name[] = { | ||||
| 	[k_kind_bool] = "bool", | ||||
| 	[k_kind_int] = "int", | ||||
| @@ -349,7 +289,6 @@ static const setting_t k_settings[] = { | ||||
| 		.description = "Whether to bind http(s) to the loopback address.  Otherwise any.", | ||||
| 		.default_value = { .kind = k_kind_bool, .bool_value = TF_IS_MOBILE ? true : false } }, | ||||
| 	{ .name = "http_port", .type = "integer", .description = "Port on which to listen for HTTP connections.", .default_value = { .kind = k_kind_int, .int_value = 12345 } }, | ||||
| 	{ .name = "https_port", .type = "integer", .description = "Port on which to listen for secure HTTP connections.", .default_value = { .kind = k_kind_int, .int_value = 0 } }, | ||||
| 	{ .name = "out_http_port_file", .type = "hidden", .description = "File to which to write bound HTTP port.", .default_value = { .kind = k_kind_string, .string_value = NULL } }, | ||||
| 	{ .name = "blob_fetch_age_seconds", | ||||
| 		.type = "integer", | ||||
| @@ -359,10 +298,6 @@ static const setting_t k_settings[] = { | ||||
| 		.type = "integer", | ||||
| 		.description = "Blobs older than this will be automatically deleted.", | ||||
| 		.default_value = { .kind = k_kind_int, .int_value = TF_IS_MOBILE ? (int)(1.0f * 365 * 24 * 60 * 60) : -1 } }, | ||||
| 	{ .name = "fetch_hosts", | ||||
| 		.type = "string", | ||||
| 		.description = "Comma-separated list of host names to which HTTP fetch requests are allowed.  None if empty.", | ||||
| 		.default_value = { .kind = k_kind_string, .string_value = NULL } }, | ||||
| 	{ .name = "http_redirect", | ||||
| 		.type = "string", | ||||
| 		.description = "If connecting by HTTP and HTTPS is configured, Location header prefix (ie, \"http://example.com\")", | ||||
| @@ -445,6 +380,31 @@ const char* tf_util_get_default_global_setting_string(const char* name) | ||||
| 	return setting && setting->default_value.string_value ? setting->default_value.string_value : ""; | ||||
| } | ||||
|  | ||||
| bool tf_util_get_global_setting_by_index(int index, const char** out_name, const char** out_type, tf_setting_kind_t* out_kind, const char** out_description) | ||||
| { | ||||
| 	if (index >= 0 && index < tf_countof(k_settings)) | ||||
| 	{ | ||||
| 		if (out_name) | ||||
| 		{ | ||||
| 			*out_name = k_settings[index].name; | ||||
| 		} | ||||
| 		if (out_type) | ||||
| 		{ | ||||
| 			*out_type = k_settings[index].type; | ||||
| 		} | ||||
| 		if (out_kind) | ||||
| 		{ | ||||
| 			*out_kind = k_settings[index].default_value.kind; | ||||
| 		} | ||||
| 		if (out_description) | ||||
| 		{ | ||||
| 			*out_description = k_settings[index].description; | ||||
| 		} | ||||
| 		return true; | ||||
| 	} | ||||
| 	return false; | ||||
| } | ||||
|  | ||||
| static JSValue _util_defaultGlobalSettings(JSContext* context, JSValueConst this_val, int argc, JSValueConst* argv) | ||||
| { | ||||
| 	JSValue settings = JS_NewObject(context); | ||||
| @@ -523,7 +483,6 @@ void tf_util_register(JSContext* context) | ||||
| 	JS_SetPropertyStr(context, global, "bip39Words", JS_NewCFunction(context, _util_bip39_words, "bip39Words", 1)); | ||||
| 	JS_SetPropertyStr(context, global, "bip39Bytes", JS_NewCFunction(context, _util_bip39_bytes, "bip39Bytes", 1)); | ||||
| 	JS_SetPropertyStr(context, global, "print", JS_NewCFunction(context, _util_print, "print", 1)); | ||||
| 	JS_SetPropertyStr(context, global, "parseHttpResponse", JS_NewCFunction(context, _util_parseHttpResponse, "parseHttpResponse", 2)); | ||||
| 	JS_SetPropertyStr(context, global, "defaultGlobalSettings", JS_NewCFunction(context, _util_defaultGlobalSettings, "defaultGlobalSettings", 2)); | ||||
| 	JS_FreeValue(context, global); | ||||
| } | ||||
|   | ||||
| @@ -212,6 +212,16 @@ const char* tf_util_get_default_global_setting_string(const char* name); | ||||
| */ | ||||
| tf_setting_kind_t tf_util_get_global_setting_kind(const char* name); | ||||
|  | ||||
| /** | ||||
| ** Get the index-th global setting. | ||||
| ** @param index The index. | ||||
| ** @param out_name Populated with the setting name. | ||||
| ** @param out_type Populated with the setting type. | ||||
| ** @param out_kind Populated with the setting kind. | ||||
| ** @param out_description Populated with the setting description. | ||||
| */ | ||||
| bool tf_util_get_global_setting_by_index(int index, const char** out_name, const char** out_type, tf_setting_kind_t* out_kind, const char** out_description); | ||||
|  | ||||
| /** | ||||
| ** Log documentation for the available settings. | ||||
| ** @param line_prefix Text to prefix each line with." | ||||
|   | ||||
| @@ -1,99 +0,0 @@ | ||||
| #!/usr/bin/env bash | ||||
|  | ||||
| if [[ -z $BUILD_PLATFORM ]]; then | ||||
| 	BUILD_PLATFORM=$(uname -s) | ||||
| fi | ||||
| if [[ -z $BUILD_TARGET ]]; then | ||||
| 	BUILD_TARGET=$(uname -m) | ||||
| 	WORK_DIR=out/openssl-local | ||||
| else | ||||
| 	WORK_DIR=out/openssl-$BUILD_PLATFORM-$BUILD_TARGET | ||||
| 	if [[ -z $SSL_TARGET ]]; then | ||||
| 		SSL_TARGET=linux-$BUILD_PLATFORM-$BUILD_TARGET | ||||
| 	fi | ||||
| fi | ||||
|  | ||||
| rm -rf $WORK_DIR | ||||
| mkdir -p out/ | ||||
| cp -aRf deps/openssl_src/ $WORK_DIR | ||||
|  | ||||
| echo "Building" | ||||
| pwd | ||||
| pushd $WORK_DIR || exit 128 | ||||
| rm -rf $DESTDIR | ||||
| echo $PATH | ||||
| export GLOBAL_OPTIONS=" \ | ||||
| no-apps \ | ||||
| no-asm \ | ||||
| no-async \ | ||||
| no-autoerrinit \ | ||||
| no-autoload-config \ | ||||
| no-cmp \ | ||||
| no-cms \ | ||||
| no-comp \ | ||||
| no-deprecated \ | ||||
| no-dgram \ | ||||
| no-docs \ | ||||
| no-dsa \ | ||||
| no-dso \ | ||||
| no-dtls \ | ||||
| no-dtls1 \ | ||||
| no-dtls1-method \ | ||||
| no-dynamic-engine \ | ||||
| no-ec2m \ | ||||
| no-egd \ | ||||
| no-engine \ | ||||
| no-err \ | ||||
| no-filenames \ | ||||
| no-gost \ | ||||
| no-http \ | ||||
| no-idea \ | ||||
| no-legacy \ | ||||
| no-md2 \ | ||||
| no-md4 \ | ||||
| no-module \ | ||||
| no-multiblock \ | ||||
| no-nextprotoneg \ | ||||
| no-ocsp \ | ||||
| no-psk \ | ||||
| no-shared \ | ||||
| no-sock \ | ||||
| no-srp \ | ||||
| no-ssl \ | ||||
| no-ssl3 \ | ||||
| no-ssl-trace \ | ||||
| no-stdio \ | ||||
| no-tests \ | ||||
| no-thread-pool \ | ||||
| no-threads \ | ||||
| no-tls1 \ | ||||
| no-tls1-method \ | ||||
| no-trace \ | ||||
| no-ui-console \ | ||||
| no-uplink \ | ||||
| no-weak-ssl-ciphers \ | ||||
| no-whirlpool \ | ||||
| no-zlib \ | ||||
| -Os \ | ||||
| -DOPENSSL_SMALL_FOOTPRINT \ | ||||
| -Wno-error \ | ||||
| -ffunction-sections \ | ||||
| -fdata-sections \ | ||||
| --static \ | ||||
| -static \ | ||||
| " | ||||
| pwd | ||||
| echo "./Configure $SSL_TARGET $OPTIONS $GLOBAL_OPTIONS" && \ | ||||
| ./Configure $SSL_TARGET $OPTIONS $GLOBAL_OPTIONS && \ | ||||
| make -s clean && \ | ||||
| make -s build_generated && \ | ||||
| make -s libcrypto.a libssl.a || exit 128 | ||||
| popd | ||||
| echo WORK_DIR=$WORK_DIR | ||||
| rm -rf out/openssl/$BUILD_PLATFORM/$BUILD_TARGET/ | ||||
| mkdir -p out/openssl/$BUILD_PLATFORM/$BUILD_TARGET/usr/local/include/ | ||||
| mkdir -p out/openssl/$BUILD_PLATFORM/$BUILD_TARGET/usr/local/lib/ | ||||
| cp -R $WORK_DIR/include/* out/openssl/$BUILD_PLATFORM/$BUILD_TARGET/usr/local/include/ | ||||
| cp $WORK_DIR/*.a out/openssl/$BUILD_PLATFORM/$BUILD_TARGET/usr/local/lib/ | ||||
|  | ||||
| echo Success | ||||
		Reference in New Issue
	
	Block a user