Merge branch 'refactor_server'
This commit is contained in:
		
							
								
								
									
										10
									
								
								android/.idea/deploymentTargetSelector.xml
									
									
									
										generated
									
									
									
										Normal file
									
								
							
							
						
						
									
										10
									
								
								android/.idea/deploymentTargetSelector.xml
									
									
									
										generated
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,10 @@ | |||||||
|  | <?xml version="1.0" encoding="UTF-8"?> | ||||||
|  | <project version="4"> | ||||||
|  |   <component name="deploymentTargetSelector"> | ||||||
|  |     <selectionStates> | ||||||
|  |       <SelectionState runConfigName="app"> | ||||||
|  |         <option name="selectionMode" value="DROPDOWN" /> | ||||||
|  |       </SelectionState> | ||||||
|  |     </selectionStates> | ||||||
|  |   </component> | ||||||
|  | </project> | ||||||
							
								
								
									
										263
									
								
								android/.idea/other.xml
									
									
									
										generated
									
									
									
										Normal file
									
								
							
							
						
						
									
										263
									
								
								android/.idea/other.xml
									
									
									
										generated
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,263 @@ | |||||||
|  | <?xml version="1.0" encoding="UTF-8"?> | ||||||
|  | <project version="4"> | ||||||
|  |   <component name="direct_access_persist.xml"> | ||||||
|  |     <option name="deviceSelectionList"> | ||||||
|  |       <list> | ||||||
|  |         <PersistentDeviceSelectionData> | ||||||
|  |           <option name="api" value="27" /> | ||||||
|  |           <option name="brand" value="DOCOMO" /> | ||||||
|  |           <option name="codename" value="F01L" /> | ||||||
|  |           <option name="id" value="F01L" /> | ||||||
|  |           <option name="manufacturer" value="FUJITSU" /> | ||||||
|  |           <option name="name" value="F-01L" /> | ||||||
|  |           <option name="screenDensity" value="360" /> | ||||||
|  |           <option name="screenX" value="720" /> | ||||||
|  |           <option name="screenY" value="1280" /> | ||||||
|  |         </PersistentDeviceSelectionData> | ||||||
|  |         <PersistentDeviceSelectionData> | ||||||
|  |           <option name="api" value="28" /> | ||||||
|  |           <option name="brand" value="DOCOMO" /> | ||||||
|  |           <option name="codename" value="SH-01L" /> | ||||||
|  |           <option name="id" value="SH-01L" /> | ||||||
|  |           <option name="manufacturer" value="SHARP" /> | ||||||
|  |           <option name="name" value="AQUOS sense2 SH-01L" /> | ||||||
|  |           <option name="screenDensity" value="480" /> | ||||||
|  |           <option name="screenX" value="1080" /> | ||||||
|  |           <option name="screenY" value="2160" /> | ||||||
|  |         </PersistentDeviceSelectionData> | ||||||
|  |         <PersistentDeviceSelectionData> | ||||||
|  |           <option name="api" value="31" /> | ||||||
|  |           <option name="brand" value="samsung" /> | ||||||
|  |           <option name="codename" value="a51" /> | ||||||
|  |           <option name="id" value="a51" /> | ||||||
|  |           <option name="manufacturer" value="Samsung" /> | ||||||
|  |           <option name="name" value="Galaxy A51" /> | ||||||
|  |           <option name="screenDensity" value="420" /> | ||||||
|  |           <option name="screenX" value="1080" /> | ||||||
|  |           <option name="screenY" value="2400" /> | ||||||
|  |         </PersistentDeviceSelectionData> | ||||||
|  |         <PersistentDeviceSelectionData> | ||||||
|  |           <option name="api" value="34" /> | ||||||
|  |           <option name="brand" value="google" /> | ||||||
|  |           <option name="codename" value="akita" /> | ||||||
|  |           <option name="id" value="akita" /> | ||||||
|  |           <option name="manufacturer" value="Google" /> | ||||||
|  |           <option name="name" value="Pixel 8a" /> | ||||||
|  |           <option name="screenDensity" value="420" /> | ||||||
|  |           <option name="screenX" value="1080" /> | ||||||
|  |           <option name="screenY" value="2400" /> | ||||||
|  |         </PersistentDeviceSelectionData> | ||||||
|  |         <PersistentDeviceSelectionData> | ||||||
|  |           <option name="api" value="33" /> | ||||||
|  |           <option name="brand" value="samsung" /> | ||||||
|  |           <option name="codename" value="b0q" /> | ||||||
|  |           <option name="id" value="b0q" /> | ||||||
|  |           <option name="manufacturer" value="Samsung" /> | ||||||
|  |           <option name="name" value="Galaxy S22 Ultra" /> | ||||||
|  |           <option name="screenDensity" value="600" /> | ||||||
|  |           <option name="screenX" value="1440" /> | ||||||
|  |           <option name="screenY" value="3088" /> | ||||||
|  |         </PersistentDeviceSelectionData> | ||||||
|  |         <PersistentDeviceSelectionData> | ||||||
|  |           <option name="api" value="32" /> | ||||||
|  |           <option name="brand" value="google" /> | ||||||
|  |           <option name="codename" value="bluejay" /> | ||||||
|  |           <option name="id" value="bluejay" /> | ||||||
|  |           <option name="manufacturer" value="Google" /> | ||||||
|  |           <option name="name" value="Pixel 6a" /> | ||||||
|  |           <option name="screenDensity" value="420" /> | ||||||
|  |           <option name="screenX" value="1080" /> | ||||||
|  |           <option name="screenY" value="2400" /> | ||||||
|  |         </PersistentDeviceSelectionData> | ||||||
|  |         <PersistentDeviceSelectionData> | ||||||
|  |           <option name="api" value="29" /> | ||||||
|  |           <option name="brand" value="samsung" /> | ||||||
|  |           <option name="codename" value="crownqlteue" /> | ||||||
|  |           <option name="id" value="crownqlteue" /> | ||||||
|  |           <option name="manufacturer" value="Samsung" /> | ||||||
|  |           <option name="name" value="Galaxy Note9" /> | ||||||
|  |           <option name="screenDensity" value="420" /> | ||||||
|  |           <option name="screenX" value="2220" /> | ||||||
|  |           <option name="screenY" value="1080" /> | ||||||
|  |         </PersistentDeviceSelectionData> | ||||||
|  |         <PersistentDeviceSelectionData> | ||||||
|  |           <option name="api" value="34" /> | ||||||
|  |           <option name="brand" value="samsung" /> | ||||||
|  |           <option name="codename" value="dm3q" /> | ||||||
|  |           <option name="id" value="dm3q" /> | ||||||
|  |           <option name="manufacturer" value="Samsung" /> | ||||||
|  |           <option name="name" value="Galaxy S23 Ultra" /> | ||||||
|  |           <option name="screenDensity" value="600" /> | ||||||
|  |           <option name="screenX" value="1440" /> | ||||||
|  |           <option name="screenY" value="3088" /> | ||||||
|  |         </PersistentDeviceSelectionData> | ||||||
|  |         <PersistentDeviceSelectionData> | ||||||
|  |           <option name="api" value="33" /> | ||||||
|  |           <option name="brand" value="google" /> | ||||||
|  |           <option name="codename" value="felix" /> | ||||||
|  |           <option name="id" value="felix" /> | ||||||
|  |           <option name="manufacturer" value="Google" /> | ||||||
|  |           <option name="name" value="Pixel Fold" /> | ||||||
|  |           <option name="screenDensity" value="420" /> | ||||||
|  |           <option name="screenX" value="2208" /> | ||||||
|  |           <option name="screenY" value="1840" /> | ||||||
|  |         </PersistentDeviceSelectionData> | ||||||
|  |         <PersistentDeviceSelectionData> | ||||||
|  |           <option name="api" value="33" /> | ||||||
|  |           <option name="brand" value="google" /> | ||||||
|  |           <option name="codename" value="felix_camera" /> | ||||||
|  |           <option name="id" value="felix_camera" /> | ||||||
|  |           <option name="manufacturer" value="Google" /> | ||||||
|  |           <option name="name" value="Pixel Fold (Camera-enabled)" /> | ||||||
|  |           <option name="screenDensity" value="420" /> | ||||||
|  |           <option name="screenX" value="2208" /> | ||||||
|  |           <option name="screenY" value="1840" /> | ||||||
|  |         </PersistentDeviceSelectionData> | ||||||
|  |         <PersistentDeviceSelectionData> | ||||||
|  |           <option name="api" value="33" /> | ||||||
|  |           <option name="brand" value="samsung" /> | ||||||
|  |           <option name="codename" value="gts8uwifi" /> | ||||||
|  |           <option name="id" value="gts8uwifi" /> | ||||||
|  |           <option name="manufacturer" value="Samsung" /> | ||||||
|  |           <option name="name" value="Galaxy Tab S8 Ultra" /> | ||||||
|  |           <option name="screenDensity" value="320" /> | ||||||
|  |           <option name="screenX" value="1848" /> | ||||||
|  |           <option name="screenY" value="2960" /> | ||||||
|  |         </PersistentDeviceSelectionData> | ||||||
|  |         <PersistentDeviceSelectionData> | ||||||
|  |           <option name="api" value="34" /> | ||||||
|  |           <option name="brand" value="google" /> | ||||||
|  |           <option name="codename" value="husky" /> | ||||||
|  |           <option name="id" value="husky" /> | ||||||
|  |           <option name="manufacturer" value="Google" /> | ||||||
|  |           <option name="name" value="Pixel 8 Pro" /> | ||||||
|  |           <option name="screenDensity" value="390" /> | ||||||
|  |           <option name="screenX" value="1008" /> | ||||||
|  |           <option name="screenY" value="2244" /> | ||||||
|  |         </PersistentDeviceSelectionData> | ||||||
|  |         <PersistentDeviceSelectionData> | ||||||
|  |           <option name="api" value="30" /> | ||||||
|  |           <option name="brand" value="motorola" /> | ||||||
|  |           <option name="codename" value="java" /> | ||||||
|  |           <option name="id" value="java" /> | ||||||
|  |           <option name="manufacturer" value="Motorola" /> | ||||||
|  |           <option name="name" value="G20" /> | ||||||
|  |           <option name="screenDensity" value="280" /> | ||||||
|  |           <option name="screenX" value="720" /> | ||||||
|  |           <option name="screenY" value="1600" /> | ||||||
|  |         </PersistentDeviceSelectionData> | ||||||
|  |         <PersistentDeviceSelectionData> | ||||||
|  |           <option name="api" value="33" /> | ||||||
|  |           <option name="brand" value="google" /> | ||||||
|  |           <option name="codename" value="lynx" /> | ||||||
|  |           <option name="id" value="lynx" /> | ||||||
|  |           <option name="manufacturer" value="Google" /> | ||||||
|  |           <option name="name" value="Pixel 7a" /> | ||||||
|  |           <option name="screenDensity" value="420" /> | ||||||
|  |           <option name="screenX" value="1080" /> | ||||||
|  |           <option name="screenY" value="2400" /> | ||||||
|  |         </PersistentDeviceSelectionData> | ||||||
|  |         <PersistentDeviceSelectionData> | ||||||
|  |           <option name="api" value="31" /> | ||||||
|  |           <option name="brand" value="google" /> | ||||||
|  |           <option name="codename" value="oriole" /> | ||||||
|  |           <option name="id" value="oriole" /> | ||||||
|  |           <option name="manufacturer" value="Google" /> | ||||||
|  |           <option name="name" value="Pixel 6" /> | ||||||
|  |           <option name="screenDensity" value="420" /> | ||||||
|  |           <option name="screenX" value="1080" /> | ||||||
|  |           <option name="screenY" value="2400" /> | ||||||
|  |         </PersistentDeviceSelectionData> | ||||||
|  |         <PersistentDeviceSelectionData> | ||||||
|  |           <option name="api" value="33" /> | ||||||
|  |           <option name="brand" value="google" /> | ||||||
|  |           <option name="codename" value="panther" /> | ||||||
|  |           <option name="id" value="panther" /> | ||||||
|  |           <option name="manufacturer" value="Google" /> | ||||||
|  |           <option name="name" value="Pixel 7" /> | ||||||
|  |           <option name="screenDensity" value="420" /> | ||||||
|  |           <option name="screenX" value="1080" /> | ||||||
|  |           <option name="screenY" value="2400" /> | ||||||
|  |         </PersistentDeviceSelectionData> | ||||||
|  |         <PersistentDeviceSelectionData> | ||||||
|  |           <option name="api" value="31" /> | ||||||
|  |           <option name="brand" value="samsung" /> | ||||||
|  |           <option name="codename" value="q2q" /> | ||||||
|  |           <option name="id" value="q2q" /> | ||||||
|  |           <option name="manufacturer" value="Samsung" /> | ||||||
|  |           <option name="name" value="Galaxy Z Fold3" /> | ||||||
|  |           <option name="screenDensity" value="420" /> | ||||||
|  |           <option name="screenX" value="1768" /> | ||||||
|  |           <option name="screenY" value="2208" /> | ||||||
|  |         </PersistentDeviceSelectionData> | ||||||
|  |         <PersistentDeviceSelectionData> | ||||||
|  |           <option name="api" value="34" /> | ||||||
|  |           <option name="brand" value="samsung" /> | ||||||
|  |           <option name="codename" value="q5q" /> | ||||||
|  |           <option name="id" value="q5q" /> | ||||||
|  |           <option name="manufacturer" value="Samsung" /> | ||||||
|  |           <option name="name" value="Galaxy Z Fold5" /> | ||||||
|  |           <option name="screenDensity" value="420" /> | ||||||
|  |           <option name="screenX" value="1812" /> | ||||||
|  |           <option name="screenY" value="2176" /> | ||||||
|  |         </PersistentDeviceSelectionData> | ||||||
|  |         <PersistentDeviceSelectionData> | ||||||
|  |           <option name="api" value="30" /> | ||||||
|  |           <option name="brand" value="google" /> | ||||||
|  |           <option name="codename" value="r11" /> | ||||||
|  |           <option name="id" value="r11" /> | ||||||
|  |           <option name="manufacturer" value="Google" /> | ||||||
|  |           <option name="name" value="Pixel Watch" /> | ||||||
|  |           <option name="screenDensity" value="320" /> | ||||||
|  |           <option name="screenX" value="384" /> | ||||||
|  |           <option name="screenY" value="384" /> | ||||||
|  |           <option name="type" value="WEAR_OS" /> | ||||||
|  |         </PersistentDeviceSelectionData> | ||||||
|  |         <PersistentDeviceSelectionData> | ||||||
|  |           <option name="api" value="30" /> | ||||||
|  |           <option name="brand" value="google" /> | ||||||
|  |           <option name="codename" value="redfin" /> | ||||||
|  |           <option name="id" value="redfin" /> | ||||||
|  |           <option name="manufacturer" value="Google" /> | ||||||
|  |           <option name="name" value="Pixel 5" /> | ||||||
|  |           <option name="screenDensity" value="440" /> | ||||||
|  |           <option name="screenX" value="1080" /> | ||||||
|  |           <option name="screenY" value="2340" /> | ||||||
|  |         </PersistentDeviceSelectionData> | ||||||
|  |         <PersistentDeviceSelectionData> | ||||||
|  |           <option name="api" value="34" /> | ||||||
|  |           <option name="brand" value="google" /> | ||||||
|  |           <option name="codename" value="shiba" /> | ||||||
|  |           <option name="id" value="shiba" /> | ||||||
|  |           <option name="manufacturer" value="Google" /> | ||||||
|  |           <option name="name" value="Pixel 8" /> | ||||||
|  |           <option name="screenDensity" value="420" /> | ||||||
|  |           <option name="screenX" value="1080" /> | ||||||
|  |           <option name="screenY" value="2400" /> | ||||||
|  |         </PersistentDeviceSelectionData> | ||||||
|  |         <PersistentDeviceSelectionData> | ||||||
|  |           <option name="api" value="33" /> | ||||||
|  |           <option name="brand" value="google" /> | ||||||
|  |           <option name="codename" value="tangorpro" /> | ||||||
|  |           <option name="id" value="tangorpro" /> | ||||||
|  |           <option name="manufacturer" value="Google" /> | ||||||
|  |           <option name="name" value="Pixel Tablet" /> | ||||||
|  |           <option name="screenDensity" value="320" /> | ||||||
|  |           <option name="screenX" value="1600" /> | ||||||
|  |           <option name="screenY" value="2560" /> | ||||||
|  |         </PersistentDeviceSelectionData> | ||||||
|  |         <PersistentDeviceSelectionData> | ||||||
|  |           <option name="api" value="29" /> | ||||||
|  |           <option name="brand" value="samsung" /> | ||||||
|  |           <option name="codename" value="x1q" /> | ||||||
|  |           <option name="id" value="x1q" /> | ||||||
|  |           <option name="manufacturer" value="Samsung" /> | ||||||
|  |           <option name="name" value="Galaxy S20" /> | ||||||
|  |           <option name="screenDensity" value="480" /> | ||||||
|  |           <option name="screenX" value="1440" /> | ||||||
|  |           <option name="screenY" value="3200" /> | ||||||
|  |         </PersistentDeviceSelectionData> | ||||||
|  |       </list> | ||||||
|  |     </option> | ||||||
|  |   </component> | ||||||
|  | </project> | ||||||
							
								
								
									
										
											BIN
										
									
								
								data/appicon_2.0.xcf
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								data/appicon_2.0.xcf
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										
											BIN
										
									
								
								data/function_graphic.ora
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								data/function_graphic.ora
									
									
									
									
									
										Executable file
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										
											BIN
										
									
								
								data/icon.ora
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								data/icon.ora
									
									
									
									
									
										Executable file
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										
											BIN
										
									
								
								data/icon_web.ora
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								data/icon_web.ora
									
									
									
									
									
										Executable file
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										
											BIN
										
									
								
								data/phone.ora
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								data/phone.ora
									
									
									
									
									
										Executable file
									
								
							
										
											Binary file not shown.
										
									
								
							| @@ -23,12 +23,26 @@ | |||||||
|  - [ ] Logout |  - [ ] Logout | ||||||
|  - [ ] Send-page |  - [ ] Send-page | ||||||
|  |  | ||||||
|  |  - [ ] Still @ERROR on scn-init, but no logs? - better persist error (write in SharedPrefs at error_$date=txt ?), also perhaps print first error line in scn-init notification? | ||||||
|  |  | ||||||
|  ----- |  ----- | ||||||
|  |  | ||||||
|  | # TODO iOS specific | ||||||
|  |  | ||||||
|  |  - [ ] payment / pro | ||||||
|  |  - [ ] show notifiactions (foreground/background/etc) | ||||||
|  |  - [ ] handle click-on-notifications should open message | ||||||
|  |  - [ ] share message | ||||||
|  |  - [ ] scan QR | ||||||
|  |  | ||||||
|  |  ----- | ||||||
|  |  | ||||||
|  | # TODO Server | ||||||
|  |  | ||||||
|  - [ ] Switch server to sq style from faby |  - [ ] Switch server to sq style from faby | ||||||
|         - [ ] switch from mattn to go-sqlite |         - [ ] switch from mattn to go-sqlite | ||||||
|         - [ ] Single struct for model/db/json |         - [ ] Single struct for model/db/json | ||||||
|  |         - [ ] use ginext | ||||||
|         - [ ] use sq.Query | sq.Update | sq.InsertAndQuery | .... |         - [ ] use sq.Query | sq.Update | sq.InsertAndQuery | .... | ||||||
|         - [ ] sq.DBOptions - enable CommentTrimmer and DefaultConverter |         - [ ] sq.DBOptions - enable CommentTrimmer and DefaultConverter | ||||||
|         - [ ] run unit-tests... |         - [ ] run unit-tests... | ||||||
|   | |||||||
| @@ -5,6 +5,8 @@ PORT=9090 | |||||||
| NAMESPACE=$(shell git rev-parse --abbrev-ref HEAD) | NAMESPACE=$(shell git rev-parse --abbrev-ref HEAD) | ||||||
| HASH=$(shell git rev-parse HEAD) | HASH=$(shell git rev-parse HEAD) | ||||||
|  |  | ||||||
|  | TAGS="timetzdata sqlite_fts5 sqlite_foreign_keys" | ||||||
|  |  | ||||||
| .PHONY: test swagger pygmentize docker migrate dgi pygmentize lint docker | .PHONY: test swagger pygmentize docker migrate dgi pygmentize lint docker | ||||||
|  |  | ||||||
| SWAGGO_VERSION=v1.8.12 | SWAGGO_VERSION=v1.8.12 | ||||||
| @@ -13,7 +15,7 @@ SWAGGO=github.com/swaggo/swag/cmd/swag@$(SWAGGO_VERSION) | |||||||
| build: ids enums swagger pygmentize fmt | build: ids enums swagger pygmentize fmt | ||||||
| 	mkdir -p _build | 	mkdir -p _build | ||||||
| 	rm -f ./_build/scn_backend | 	rm -f ./_build/scn_backend | ||||||
| 	CGO_ENABLED=1 go build -v -o _build/scn_backend -tags "timetzdata sqlite_fts5 sqlite_foreign_keys" ./cmd/scnserver | 	CGO_ENABLED=1 go build -v -o _build/scn_backend -tags $(TAGS) ./cmd/scnserver | ||||||
|  |  | ||||||
| enums: | enums: | ||||||
| 	go generate models/enums.go | 	go generate models/enums.go | ||||||
| @@ -27,7 +29,7 @@ run: build | |||||||
|  |  | ||||||
| gow: | gow: | ||||||
| 	which gow || go install github.com/mitranim/gow@latest | 	which gow || go install github.com/mitranim/gow@latest | ||||||
| 	gow -e "go,mod,html,css,json,yaml,js" run -tags "timetzdata sqlite_fts5 sqlite_foreign_keys" blackforestbytes.com/simplecloudnotifier/cmd/scnserver | 	gow -e "go,mod,html,css,json,yaml,js" run -tags $(TAGS) blackforestbytes.com/simplecloudnotifier/cmd/scnserver | ||||||
|  |  | ||||||
| dgi: | dgi: | ||||||
| 	[ ! -f "DOCKER_GIT_INFO" ] || rm DOCKER_GIT_INFO | 	[ ! -f "DOCKER_GIT_INFO" ] || rm DOCKER_GIT_INFO | ||||||
| @@ -99,10 +101,10 @@ fmt: swagger-setup | |||||||
|  |  | ||||||
| test: | test: | ||||||
| 	which gotestsum || go install gotest.tools/gotestsum@latest | 	which gotestsum || go install gotest.tools/gotestsum@latest | ||||||
| 	gotestsum --format "testname" -- -tags="timetzdata sqlite_fts5 sqlite_foreign_keys" "./test" | 	gotestsum --format "testname" -- -tags $(TAGS) "./test" | ||||||
|  |  | ||||||
| migrate: | migrate: | ||||||
| 	CGO_ENABLED=1 go build -v -o _build/scn_migrate -tags "timetzdata sqlite_fts5 sqlite_foreign_keys" ./cmd/migrate | 	CGO_ENABLED=1 go build -v -o _build/scn_migrate -tags $(TAGS) ./cmd/migrate | ||||||
| 	./_build/scn_migrate | 	./_build/scn_migrate | ||||||
|  |  | ||||||
| lint: | lint: | ||||||
|   | |||||||
| @@ -10,11 +10,7 @@ | |||||||
|  |  | ||||||
|  - ios purchase verification |  - ios purchase verification | ||||||
|  |  | ||||||
|  - (!) use goext.ginWrapper |  - exerr.New | exerr.Wrap | ||||||
|    |  | ||||||
|  - (!) use goext.exerr |  | ||||||
|  |  | ||||||
|  - use bfcodegen (enums+id) |  | ||||||
|  |  | ||||||
| #### UNSURE | #### UNSURE | ||||||
|  |  | ||||||
| @@ -57,19 +53,12 @@ | |||||||
|  |  | ||||||
|  - Pagination for ListChannels / ListSubscriptions / ListClients / ListChannelSubscriptions / ListUserSubscriptions |  - Pagination for ListChannels / ListSubscriptions / ListClients / ListChannelSubscriptions / ListUserSubscriptions | ||||||
|  |  | ||||||
|  - Use only single struct for DB|Model|JSON |  | ||||||
|      * needs sq.Converter implementation |  | ||||||
|      * needs to handle joined data |  | ||||||
|      * rfctime.Time... |  | ||||||
|  |  | ||||||
|  - use job superclass (copy from isi/bnet/?), reduce duplicate code |  - use job superclass (copy from isi/bnet/?), reduce duplicate code | ||||||
|  |  | ||||||
|  - admin panel (especially errors and requests) |  - admin panel (especially errors and requests) | ||||||
|  |  | ||||||
|  - cli app (?) |  - cli app (?) | ||||||
|  |  | ||||||
|  - Use "github.com/glebarez/go-sqlite" instead of mattn3 (see ai-sig alarmserver) |  | ||||||
|  |  | ||||||
| #### FUTURE | #### FUTURE | ||||||
|  |  | ||||||
|  - Remove compat, especially do not create compat id for every new message... |  - Remove compat, especially do not create compat id for every new message... | ||||||
| @@ -18,6 +18,7 @@ const ( | |||||||
| 	BINDFAIL_QUERY_PARAM  APIError = 1151 | 	BINDFAIL_QUERY_PARAM  APIError = 1151 | ||||||
| 	BINDFAIL_BODY_PARAM   APIError = 1152 | 	BINDFAIL_BODY_PARAM   APIError = 1152 | ||||||
| 	BINDFAIL_URI_PARAM    APIError = 1153 | 	BINDFAIL_URI_PARAM    APIError = 1153 | ||||||
|  | 	BINDFAIL_HEADER_PARAM APIError = 1152 | ||||||
| 	INVALID_BODY_PARAM    APIError = 1161 | 	INVALID_BODY_PARAM    APIError = 1161 | ||||||
| 	INVALID_ENUM_VALUE    APIError = 1171 | 	INVALID_ENUM_VALUE    APIError = 1171 | ||||||
|  |  | ||||||
|   | |||||||
| @@ -1,21 +0,0 @@ | |||||||
| package ginext |  | ||||||
|  |  | ||||||
| import ( |  | ||||||
| 	"github.com/gin-gonic/gin" |  | ||||||
| 	"net/http" |  | ||||||
| ) |  | ||||||
|  |  | ||||||
| func CorsMiddleware() gin.HandlerFunc { |  | ||||||
| 	return func(c *gin.Context) { |  | ||||||
| 		c.Writer.Header().Set("Access-Control-Allow-Origin", "*") |  | ||||||
| 		c.Writer.Header().Set("Access-Control-Allow-Credentials", "true") |  | ||||||
| 		c.Writer.Header().Set("Access-Control-Allow-Headers", "Content-Type, Content-Length, Accept-Encoding, X-CSRF-Token, Authorization, accept, origin, Cache-Control, X-Requested-With") |  | ||||||
| 		c.Writer.Header().Set("Access-Control-Allow-Methods", "OPTIONS, GET, POST, PUT, PATCH, DELETE") |  | ||||||
|  |  | ||||||
| 		if c.Request.Method == "OPTIONS" { |  | ||||||
| 			c.AbortWithStatus(http.StatusOK) |  | ||||||
| 		} else { |  | ||||||
| 			c.Next() |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
| } |  | ||||||
| @@ -1,31 +0,0 @@ | |||||||
| package ginext |  | ||||||
|  |  | ||||||
| import ( |  | ||||||
| 	scn "blackforestbytes.com/simplecloudnotifier" |  | ||||||
| 	"github.com/gin-gonic/gin" |  | ||||||
| ) |  | ||||||
|  |  | ||||||
| var SuppressGinLogs = false |  | ||||||
|  |  | ||||||
| func NewEngine(cfg scn.Config) *gin.Engine { |  | ||||||
| 	engine := gin.New() |  | ||||||
|  |  | ||||||
| 	engine.RedirectFixedPath = false |  | ||||||
| 	engine.RedirectTrailingSlash = false |  | ||||||
|  |  | ||||||
| 	if cfg.Cors { |  | ||||||
| 		engine.Use(CorsMiddleware()) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	if cfg.GinDebug { |  | ||||||
| 		ginlogger := gin.Logger() |  | ||||||
| 		engine.Use(func(context *gin.Context) { |  | ||||||
| 			if SuppressGinLogs { |  | ||||||
| 				return |  | ||||||
| 			} |  | ||||||
| 			ginlogger(context) |  | ||||||
| 		}) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	return engine |  | ||||||
| } |  | ||||||
| @@ -1,24 +0,0 @@ | |||||||
| package ginext |  | ||||||
|  |  | ||||||
| import ( |  | ||||||
| 	"github.com/gin-gonic/gin" |  | ||||||
| 	"net/http" |  | ||||||
| ) |  | ||||||
|  |  | ||||||
| func RedirectFound(newuri string) gin.HandlerFunc { |  | ||||||
| 	return func(g *gin.Context) { |  | ||||||
| 		g.Redirect(http.StatusFound, newuri) |  | ||||||
| 	} |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func RedirectTemporary(newuri string) gin.HandlerFunc { |  | ||||||
| 	return func(g *gin.Context) { |  | ||||||
| 		g.Redirect(http.StatusTemporaryRedirect, newuri) |  | ||||||
| 	} |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func RedirectPermanent(newuri string) gin.HandlerFunc { |  | ||||||
| 	return func(g *gin.Context) { |  | ||||||
| 		g.Redirect(http.StatusPermanentRedirect, newuri) |  | ||||||
| 	} |  | ||||||
| } |  | ||||||
| @@ -7,114 +7,43 @@ import ( | |||||||
| 	"fmt" | 	"fmt" | ||||||
| 	"github.com/gin-gonic/gin" | 	"github.com/gin-gonic/gin" | ||||||
| 	"github.com/rs/zerolog/log" | 	"github.com/rs/zerolog/log" | ||||||
|  | 	"gogs.mikescher.com/BlackForestBytes/goext/ginext" | ||||||
| 	json "gogs.mikescher.com/BlackForestBytes/goext/gojson" | 	json "gogs.mikescher.com/BlackForestBytes/goext/gojson" | ||||||
| 	"gogs.mikescher.com/BlackForestBytes/goext/langext" | 	"gogs.mikescher.com/BlackForestBytes/goext/langext" | ||||||
| 	"runtime/debug" | 	"runtime/debug" | ||||||
| 	"strings" | 	"strings" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| type HTTPResponse interface { | type cookieval struct { | ||||||
| 	Write(g *gin.Context) | 	name     string | ||||||
| 	Statuscode() int | 	value    string | ||||||
| 	BodyString() *string | 	maxAge   int | ||||||
| 	ContentType() string | 	path     string | ||||||
|  | 	domain   string | ||||||
|  | 	secure   bool | ||||||
|  | 	httpOnly bool | ||||||
| } | } | ||||||
|  |  | ||||||
| type jsonHTTPResponse struct { | type headerval struct { | ||||||
| 	statusCode int | 	Key string | ||||||
| 	data       any | 	Val string | ||||||
| } |  | ||||||
|  |  | ||||||
| func (j jsonHTTPResponse) Write(g *gin.Context) { |  | ||||||
| 	g.Render(j.statusCode, json.GoJsonRender{Data: j.data, NilSafeSlices: true, NilSafeMaps: true}) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (j jsonHTTPResponse) Statuscode() int { |  | ||||||
| 	return j.statusCode |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (j jsonHTTPResponse) BodyString() *string { |  | ||||||
| 	v, err := json.Marshal(j.data) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return nil |  | ||||||
| 	} |  | ||||||
| 	return langext.Ptr(string(v)) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (j jsonHTTPResponse) ContentType() string { |  | ||||||
| 	return "application/json" |  | ||||||
| } |  | ||||||
|  |  | ||||||
| type emptyHTTPResponse struct { |  | ||||||
| 	statusCode int |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (j emptyHTTPResponse) Write(g *gin.Context) { |  | ||||||
| 	g.Status(j.statusCode) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (j emptyHTTPResponse) Statuscode() int { |  | ||||||
| 	return j.statusCode |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (j emptyHTTPResponse) BodyString() *string { |  | ||||||
| 	return nil |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (j emptyHTTPResponse) ContentType() string { |  | ||||||
| 	return "" |  | ||||||
| } |  | ||||||
|  |  | ||||||
| type textHTTPResponse struct { |  | ||||||
| 	statusCode int |  | ||||||
| 	data       string |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (j textHTTPResponse) Write(g *gin.Context) { |  | ||||||
| 	g.String(j.statusCode, "%s", j.data) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (j textHTTPResponse) Statuscode() int { |  | ||||||
| 	return j.statusCode |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (j textHTTPResponse) BodyString() *string { |  | ||||||
| 	return langext.Ptr(j.data) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (j textHTTPResponse) ContentType() string { |  | ||||||
| 	return "text/plain" |  | ||||||
| } |  | ||||||
|  |  | ||||||
| type dataHTTPResponse struct { |  | ||||||
| 	statusCode  int |  | ||||||
| 	data        []byte |  | ||||||
| 	contentType string |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (j dataHTTPResponse) Write(g *gin.Context) { |  | ||||||
| 	g.Data(j.statusCode, j.contentType, j.data) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (j dataHTTPResponse) Statuscode() int { |  | ||||||
| 	return j.statusCode |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (j dataHTTPResponse) BodyString() *string { |  | ||||||
| 	return langext.Ptr(string(j.data)) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (j dataHTTPResponse) ContentType() string { |  | ||||||
| 	return j.contentType |  | ||||||
| } | } | ||||||
|  |  | ||||||
| type errorHTTPResponse struct { | type errorHTTPResponse struct { | ||||||
| 	statusCode int | 	statusCode int | ||||||
| 	data       any | 	data       any | ||||||
| 	error      error | 	error      error | ||||||
|  | 	headers    []headerval | ||||||
|  | 	cookies    []cookieval | ||||||
| } | } | ||||||
|  |  | ||||||
| func (j errorHTTPResponse) Write(g *gin.Context) { | func (j errorHTTPResponse) Write(g *gin.Context) { | ||||||
|  | 	for _, v := range j.headers { | ||||||
|  | 		g.Header(v.Key, v.Val) | ||||||
|  | 	} | ||||||
|  | 	for _, v := range j.cookies { | ||||||
|  | 		g.SetCookie(v.name, v.value, v.maxAge, v.path, v.domain, v.secure, v.httpOnly) | ||||||
|  | 	} | ||||||
| 	g.JSON(j.statusCode, j.data) | 	g.JSON(j.statusCode, j.data) | ||||||
| } | } | ||||||
|  |  | ||||||
| @@ -122,7 +51,7 @@ func (j errorHTTPResponse) Statuscode() int { | |||||||
| 	return j.statusCode | 	return j.statusCode | ||||||
| } | } | ||||||
|  |  | ||||||
| func (j errorHTTPResponse) BodyString() *string { | func (j errorHTTPResponse) BodyString(g *gin.Context) *string { | ||||||
| 	v, err := json.Marshal(j.data) | 	v, err := json.Marshal(j.data) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil | 		return nil | ||||||
| @@ -134,39 +63,41 @@ func (j errorHTTPResponse) ContentType() string { | |||||||
| 	return "application/json" | 	return "application/json" | ||||||
| } | } | ||||||
|  |  | ||||||
| func Status(sc int) HTTPResponse { | func (j errorHTTPResponse) WithHeader(k string, v string) ginext.HTTPResponse { | ||||||
| 	return &emptyHTTPResponse{statusCode: sc} | 	j.headers = append(j.headers, headerval{k, v}) | ||||||
|  | 	return j | ||||||
| } | } | ||||||
|  |  | ||||||
| func JSON(sc int, data any) HTTPResponse { | func (j errorHTTPResponse) WithCookie(name string, value string, maxAge int, path string, domain string, secure bool, httpOnly bool) ginext.HTTPResponse { | ||||||
| 	return &jsonHTTPResponse{statusCode: sc, data: data} | 	j.cookies = append(j.cookies, cookieval{name, value, maxAge, path, domain, secure, httpOnly}) | ||||||
|  | 	return j | ||||||
| } | } | ||||||
|  |  | ||||||
| func Data(sc int, contentType string, data []byte) HTTPResponse { | func (j errorHTTPResponse) IsSuccess() bool { | ||||||
| 	return &dataHTTPResponse{statusCode: sc, contentType: contentType, data: data} | 	return false | ||||||
| } | } | ||||||
|  |  | ||||||
| func Text(sc int, data string) HTTPResponse { | func (j errorHTTPResponse) Headers() []string { | ||||||
| 	return &textHTTPResponse{statusCode: sc, data: data} | 	return langext.ArrMap(j.headers, func(v headerval) string { return v.Key + "=" + v.Val }) | ||||||
| } | } | ||||||
|  |  | ||||||
| func InternalError(e error) HTTPResponse { | func (j errorHTTPResponse) Unwrap() error { | ||||||
|  | 	return j.error | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func InternalError(e error) ginext.HTTPResponse { | ||||||
| 	return createApiError(nil, "InternalError", 500, apierr.INTERNAL_EXCEPTION, 0, e.Error(), e) | 	return createApiError(nil, "InternalError", 500, apierr.INTERNAL_EXCEPTION, 0, e.Error(), e) | ||||||
| } | } | ||||||
|  |  | ||||||
| func APIError(g *gin.Context, status int, errorid apierr.APIError, msg string, e error) HTTPResponse { | func APIError(g *gin.Context, status int, errorid apierr.APIError, msg string, e error) ginext.HTTPResponse { | ||||||
| 	return createApiError(g, "APIError", status, errorid, 0, msg, e) | 	return createApiError(g, "APIError", status, errorid, 0, msg, e) | ||||||
| } | } | ||||||
|  |  | ||||||
| func SendAPIError(g *gin.Context, status int, errorid apierr.APIError, highlight apihighlight.ErrHighlight, msg string, e error) HTTPResponse { | func SendAPIError(g *gin.Context, status int, errorid apierr.APIError, highlight apihighlight.ErrHighlight, msg string, e error) ginext.HTTPResponse { | ||||||
| 	return createApiError(g, "SendAPIError", status, errorid, highlight, msg, e) | 	return createApiError(g, "SendAPIError", status, errorid, highlight, msg, e) | ||||||
| } | } | ||||||
|  |  | ||||||
| func NotImplemented(g *gin.Context) HTTPResponse { | func createApiError(g *gin.Context, ident string, status int, errorid apierr.APIError, highlight apihighlight.ErrHighlight, msg string, e error) ginext.HTTPResponse { | ||||||
| 	return createApiError(g, "NotImplemented", 500, apierr.NOT_IMPLEMENTED, 0, "Not Implemented", nil) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func createApiError(g *gin.Context, ident string, status int, errorid apierr.APIError, highlight apihighlight.ErrHighlight, msg string, e error) HTTPResponse { |  | ||||||
| 	reqUri := "" | 	reqUri := "" | ||||||
| 	if g != nil && g.Request != nil { | 	if g != nil && g.Request != nil { | ||||||
| 		reqUri = g.Request.Method + " :: " + g.Request.RequestURI | 		reqUri = g.Request.Method + " :: " + g.Request.RequestURI | ||||||
| @@ -207,6 +138,6 @@ func createApiError(g *gin.Context, ident string, status int, errorid apierr.API | |||||||
| 	} | 	} | ||||||
| } | } | ||||||
|  |  | ||||||
| func CompatAPIError(errid int, msg string) HTTPResponse { | func CompatAPIError(errid int, msg string) ginext.HTTPResponse { | ||||||
| 	return &jsonHTTPResponse{statusCode: 200, data: compatAPIError{Success: false, ErrorID: errid, Message: msg}} | 	return ginext.JSON(200, compatAPIError{Success: false, ErrorID: errid, Message: msg}) | ||||||
| } | } | ||||||
|   | |||||||
| @@ -1,191 +0,0 @@ | |||||||
| package ginresp |  | ||||||
|  |  | ||||||
| import ( |  | ||||||
| 	scn "blackforestbytes.com/simplecloudnotifier" |  | ||||||
| 	"blackforestbytes.com/simplecloudnotifier/api/apierr" |  | ||||||
| 	"blackforestbytes.com/simplecloudnotifier/models" |  | ||||||
| 	"errors" |  | ||||||
| 	"fmt" |  | ||||||
| 	"github.com/gin-gonic/gin" |  | ||||||
| 	"github.com/mattn/go-sqlite3" |  | ||||||
| 	"github.com/rs/zerolog/log" |  | ||||||
| 	"gogs.mikescher.com/BlackForestBytes/goext/dataext" |  | ||||||
| 	"gogs.mikescher.com/BlackForestBytes/goext/langext" |  | ||||||
| 	"math/rand" |  | ||||||
| 	"runtime/debug" |  | ||||||
| 	"time" |  | ||||||
| ) |  | ||||||
|  |  | ||||||
| type WHandlerFunc func(*gin.Context) HTTPResponse |  | ||||||
|  |  | ||||||
| type RequestLogAcceptor interface { |  | ||||||
| 	InsertRequestLog(data models.RequestLog) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func Wrap(rlacc RequestLogAcceptor, fn WHandlerFunc) gin.HandlerFunc { |  | ||||||
|  |  | ||||||
| 	maxRetry := scn.Conf.RequestMaxRetry |  | ||||||
| 	retrySleep := scn.Conf.RequestRetrySleep |  | ||||||
|  |  | ||||||
| 	return func(g *gin.Context) { |  | ||||||
|  |  | ||||||
| 		reqctx := g.Request.Context() |  | ||||||
|  |  | ||||||
| 		if g.Request.Body != nil { |  | ||||||
| 			g.Request.Body = dataext.NewBufferedReadCloser(g.Request.Body) |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		t0 := time.Now() |  | ||||||
|  |  | ||||||
| 		for ctr := 1; ; ctr++ { |  | ||||||
|  |  | ||||||
| 			wrap, stackTrace, panicObj := callPanicSafe(fn, g) |  | ||||||
| 			if panicObj != nil { |  | ||||||
| 				log.Error().Interface("panicObj", panicObj).Msg("Panic occured (in gin handler)") |  | ||||||
| 				log.Error().Msg(stackTrace) |  | ||||||
| 				wrap = APIError(g, 500, apierr.PANIC, "A panic occured in the HTTP handler", errors.New(fmt.Sprintf("%+v\n\n@:\n%s", panicObj, stackTrace))) |  | ||||||
| 			} |  | ||||||
|  |  | ||||||
| 			if g.Writer.Written() { |  | ||||||
| 				if scn.Conf.ReqLogEnabled { |  | ||||||
| 					rlacc.InsertRequestLog(createRequestLog(g, t0, ctr, nil, langext.Ptr("Writing in WrapperFunc is not supported"))) |  | ||||||
| 				} |  | ||||||
| 				panic("Writing in WrapperFunc is not supported") |  | ||||||
| 			} |  | ||||||
|  |  | ||||||
| 			if ctr < maxRetry && isSqlite3Busy(wrap) { |  | ||||||
| 				log.Warn().Int("counter", ctr).Str("url", g.Request.URL.String()).Msg("Retry request (ErrBusy)") |  | ||||||
|  |  | ||||||
| 				err := resetBody(g) |  | ||||||
| 				if err != nil { |  | ||||||
| 					panic(err) |  | ||||||
| 				} |  | ||||||
|  |  | ||||||
| 				time.Sleep(time.Duration(int64(float64(retrySleep) * (0.5 + rand.Float64())))) |  | ||||||
| 				continue |  | ||||||
| 			} |  | ||||||
|  |  | ||||||
| 			if reqctx.Err() == nil { |  | ||||||
| 				if scn.Conf.ReqLogEnabled { |  | ||||||
| 					rlacc.InsertRequestLog(createRequestLog(g, t0, ctr, wrap, nil)) |  | ||||||
| 				} |  | ||||||
|  |  | ||||||
| 				statuscode := wrap.Statuscode() |  | ||||||
| 				if statuscode/100 != 2 { |  | ||||||
| 					log.Warn().Str("url", g.Request.Method+"::"+g.Request.URL.String()).Msg(fmt.Sprintf("Request failed with statuscode %d", statuscode)) |  | ||||||
| 				} |  | ||||||
|  |  | ||||||
| 				wrap.Write(g) |  | ||||||
| 			} |  | ||||||
|  |  | ||||||
| 			return |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func createRequestLog(g *gin.Context, t0 time.Time, ctr int, resp HTTPResponse, panicstr *string) models.RequestLog { |  | ||||||
|  |  | ||||||
| 	t1 := time.Now() |  | ||||||
|  |  | ||||||
| 	ua := g.Request.UserAgent() |  | ||||||
| 	auth := g.Request.Header.Get("Authorization") |  | ||||||
| 	ct := g.Request.Header.Get("Content-Type") |  | ||||||
|  |  | ||||||
| 	var reqbody []byte = nil |  | ||||||
| 	if g.Request.Body != nil { |  | ||||||
| 		brcbody, err := g.Request.Body.(dataext.BufferedReadCloser).BufferedAll() |  | ||||||
| 		if err == nil { |  | ||||||
| 			reqbody = brcbody |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
| 	var strreqbody *string = nil |  | ||||||
| 	if len(reqbody) < scn.Conf.ReqLogMaxBodySize { |  | ||||||
| 		strreqbody = langext.Ptr(string(reqbody)) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	var respbody *string = nil |  | ||||||
|  |  | ||||||
| 	var strrespbody *string = nil |  | ||||||
| 	if resp != nil { |  | ||||||
| 		respbody = resp.BodyString() |  | ||||||
| 		if respbody != nil && len(*respbody) < scn.Conf.ReqLogMaxBodySize { |  | ||||||
| 			strrespbody = respbody |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	permObj, hasPerm := g.Get("perm") |  | ||||||
|  |  | ||||||
| 	hasTok := false |  | ||||||
| 	if hasPerm { |  | ||||||
| 		hasTok = permObj.(models.PermissionSet).Token != nil |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	return models.RequestLog{ |  | ||||||
| 		Method:              g.Request.Method, |  | ||||||
| 		URI:                 g.Request.URL.String(), |  | ||||||
| 		UserAgent:           langext.Conditional(ua == "", nil, &ua), |  | ||||||
| 		Authentication:      langext.Conditional(auth == "", nil, &auth), |  | ||||||
| 		RequestBody:         strreqbody, |  | ||||||
| 		RequestBodySize:     int64(len(reqbody)), |  | ||||||
| 		RequestContentType:  ct, |  | ||||||
| 		RemoteIP:            g.RemoteIP(), |  | ||||||
| 		KeyID:               langext.ConditionalFn10(hasTok, func() *models.KeyTokenID { return langext.Ptr(permObj.(models.PermissionSet).Token.KeyTokenID) }, nil), |  | ||||||
| 		UserID:              langext.ConditionalFn10(hasTok, func() *models.UserID { return langext.Ptr(permObj.(models.PermissionSet).Token.OwnerUserID) }, nil), |  | ||||||
| 		Permissions:         langext.ConditionalFn10(hasTok, func() *string { return langext.Ptr(permObj.(models.PermissionSet).Token.Permissions.String()) }, nil), |  | ||||||
| 		ResponseStatuscode:  langext.ConditionalFn10(resp != nil, func() *int64 { return langext.Ptr(int64(resp.Statuscode())) }, nil), |  | ||||||
| 		ResponseBodySize:    langext.ConditionalFn10(strrespbody != nil, func() *int64 { return langext.Ptr(int64(len(*respbody))) }, nil), |  | ||||||
| 		ResponseBody:        strrespbody, |  | ||||||
| 		ResponseContentType: langext.ConditionalFn10(resp != nil, func() string { return resp.ContentType() }, ""), |  | ||||||
| 		RetryCount:          int64(ctr), |  | ||||||
| 		Panicked:            panicstr != nil, |  | ||||||
| 		PanicStr:            panicstr, |  | ||||||
| 		ProcessingTime:      t1.Sub(t0), |  | ||||||
| 		TimestampStart:      t0, |  | ||||||
| 		TimestampFinish:     t1, |  | ||||||
| 	} |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func callPanicSafe(fn WHandlerFunc, g *gin.Context) (res HTTPResponse, stackTrace string, panicObj any) { |  | ||||||
| 	defer func() { |  | ||||||
| 		if rec := recover(); rec != nil { |  | ||||||
| 			res = nil |  | ||||||
| 			stackTrace = string(debug.Stack()) |  | ||||||
| 			panicObj = rec |  | ||||||
| 		} |  | ||||||
| 	}() |  | ||||||
|  |  | ||||||
| 	res = fn(g) |  | ||||||
| 	return res, "", nil |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func resetBody(g *gin.Context) error { |  | ||||||
| 	if g.Request.Body == nil { |  | ||||||
| 		return nil |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	err := g.Request.Body.(dataext.BufferedReadCloser).Reset() |  | ||||||
| 	if err != nil { |  | ||||||
| 		return err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	return nil |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func isSqlite3Busy(r HTTPResponse) bool { |  | ||||||
| 	if errwrap, ok := r.(*errorHTTPResponse); ok && errwrap != nil { |  | ||||||
|  |  | ||||||
| 		if errors.Is(errwrap.error, sqlite3.ErrBusy) { |  | ||||||
| 			return true |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		var s3err sqlite3.Error |  | ||||||
| 		if errors.As(errwrap.error, &s3err) { |  | ||||||
| 			if errors.Is(s3err.Code, sqlite3.ErrBusy) { |  | ||||||
| 				return true |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
| 	return false |  | ||||||
| } |  | ||||||
| @@ -4,11 +4,12 @@ import ( | |||||||
| 	"blackforestbytes.com/simplecloudnotifier/api/apierr" | 	"blackforestbytes.com/simplecloudnotifier/api/apierr" | ||||||
| 	"blackforestbytes.com/simplecloudnotifier/api/ginresp" | 	"blackforestbytes.com/simplecloudnotifier/api/ginresp" | ||||||
| 	ct "blackforestbytes.com/simplecloudnotifier/db/cursortoken" | 	ct "blackforestbytes.com/simplecloudnotifier/db/cursortoken" | ||||||
|  | 	"blackforestbytes.com/simplecloudnotifier/logic" | ||||||
| 	"blackforestbytes.com/simplecloudnotifier/models" | 	"blackforestbytes.com/simplecloudnotifier/models" | ||||||
| 	"database/sql" | 	"database/sql" | ||||||
| 	"errors" | 	"errors" | ||||||
| 	"fmt" | 	"fmt" | ||||||
| 	"github.com/gin-gonic/gin" | 	"gogs.mikescher.com/BlackForestBytes/goext/ginext" | ||||||
| 	"gogs.mikescher.com/BlackForestBytes/goext/langext" | 	"gogs.mikescher.com/BlackForestBytes/goext/langext" | ||||||
| 	"gogs.mikescher.com/BlackForestBytes/goext/mathext" | 	"gogs.mikescher.com/BlackForestBytes/goext/mathext" | ||||||
| 	"net/http" | 	"net/http" | ||||||
| @@ -37,7 +38,7 @@ import ( | |||||||
| //	@Failure		500			{object}	ginresp.apiError	"internal server error" | //	@Failure		500			{object}	ginresp.apiError	"internal server error" | ||||||
| // | // | ||||||
| //	@Router			/api/v2/users/{uid}/channels [GET] | //	@Router			/api/v2/users/{uid}/channels [GET] | ||||||
| func (h APIHandler) ListChannels(g *gin.Context) ginresp.HTTPResponse { | func (h APIHandler) ListChannels(pctx ginext.PreContext) ginext.HTTPResponse { | ||||||
| 	type uri struct { | 	type uri struct { | ||||||
| 		UserID models.UserID `uri:"uid" binding:"entityid"` | 		UserID models.UserID `uri:"uid" binding:"entityid"` | ||||||
| 	} | 	} | ||||||
| @@ -45,32 +46,32 @@ func (h APIHandler) ListChannels(g *gin.Context) ginresp.HTTPResponse { | |||||||
| 		Selector *string `json:"selector" form:"selector"  enums:"owned,subscribed_any,all_any,subscribed,all"` | 		Selector *string `json:"selector" form:"selector"  enums:"owned,subscribed_any,all_any,subscribed,all"` | ||||||
| 	} | 	} | ||||||
| 	type response struct { | 	type response struct { | ||||||
| 		Channels []models.ChannelWithSubscriptionJSON `json:"channels"` | 		Channels []models.ChannelWithSubscription `json:"channels"` | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	var u uri | 	var u uri | ||||||
| 	var q query | 	var q query | ||||||
| 	ctx, errResp := h.app.StartRequest(g, &u, &q, nil, nil) | 	ctx, g, errResp := pctx.URI(&u).Query(&q).Start() | ||||||
| 	if errResp != nil { | 	if errResp != nil { | ||||||
| 		return *errResp | 		return *errResp | ||||||
| 	} | 	} | ||||||
| 	defer ctx.Cancel() | 	defer ctx.Cancel() | ||||||
|  |  | ||||||
|  | 	return h.app.DoRequest(ctx, g, models.TLockRead, func(ctx *logic.AppContext, finishSuccess func(r ginext.HTTPResponse) ginext.HTTPResponse) ginext.HTTPResponse { | ||||||
|  |  | ||||||
| 		if permResp := ctx.CheckPermissionUserRead(u.UserID); permResp != nil { | 		if permResp := ctx.CheckPermissionUserRead(u.UserID); permResp != nil { | ||||||
| 			return *permResp | 			return *permResp | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		sel := strings.ToLower(langext.Coalesce(q.Selector, "owned")) | 		sel := strings.ToLower(langext.Coalesce(q.Selector, "owned")) | ||||||
|  |  | ||||||
| 	var res []models.ChannelWithSubscriptionJSON |  | ||||||
|  |  | ||||||
| 		if sel == "owned" { | 		if sel == "owned" { | ||||||
|  |  | ||||||
| 			channels, err := h.database.ListChannelsByOwner(ctx, u.UserID, u.UserID) | 			channels, err := h.database.ListChannelsByOwner(ctx, u.UserID, u.UserID) | ||||||
| 			if err != nil { | 			if err != nil { | ||||||
| 				return ginresp.APIError(g, 500, apierr.DATABASE_ERROR, "Failed to query channels", err) | 				return ginresp.APIError(g, 500, apierr.DATABASE_ERROR, "Failed to query channels", err) | ||||||
| 			} | 			} | ||||||
| 		res = langext.ArrMap(channels, func(v models.ChannelWithSubscription) models.ChannelWithSubscriptionJSON { return v.JSON(true) }) | 			return finishSuccess(ginext.JSONWithFilter(http.StatusOK, response{Channels: channels}, "INCLUDE_KEY")) | ||||||
|  |  | ||||||
| 		} else if sel == "subscribed_any" { | 		} else if sel == "subscribed_any" { | ||||||
|  |  | ||||||
| @@ -78,7 +79,7 @@ func (h APIHandler) ListChannels(g *gin.Context) ginresp.HTTPResponse { | |||||||
| 			if err != nil { | 			if err != nil { | ||||||
| 				return ginresp.APIError(g, 500, apierr.DATABASE_ERROR, "Failed to query channels", err) | 				return ginresp.APIError(g, 500, apierr.DATABASE_ERROR, "Failed to query channels", err) | ||||||
| 			} | 			} | ||||||
| 		res = langext.ArrMap(channels, func(v models.ChannelWithSubscription) models.ChannelWithSubscriptionJSON { return v.JSON(false) }) | 			return finishSuccess(ginext.JSON(http.StatusOK, response{Channels: channels})) | ||||||
|  |  | ||||||
| 		} else if sel == "all_any" { | 		} else if sel == "all_any" { | ||||||
|  |  | ||||||
| @@ -86,7 +87,7 @@ func (h APIHandler) ListChannels(g *gin.Context) ginresp.HTTPResponse { | |||||||
| 			if err != nil { | 			if err != nil { | ||||||
| 				return ginresp.APIError(g, 500, apierr.DATABASE_ERROR, "Failed to query channels", err) | 				return ginresp.APIError(g, 500, apierr.DATABASE_ERROR, "Failed to query channels", err) | ||||||
| 			} | 			} | ||||||
| 		res = langext.ArrMap(channels, func(v models.ChannelWithSubscription) models.ChannelWithSubscriptionJSON { return v.JSON(false) }) | 			return finishSuccess(ginext.JSON(http.StatusOK, response{Channels: channels})) | ||||||
|  |  | ||||||
| 		} else if sel == "subscribed" { | 		} else if sel == "subscribed" { | ||||||
|  |  | ||||||
| @@ -94,7 +95,7 @@ func (h APIHandler) ListChannels(g *gin.Context) ginresp.HTTPResponse { | |||||||
| 			if err != nil { | 			if err != nil { | ||||||
| 				return ginresp.APIError(g, 500, apierr.DATABASE_ERROR, "Failed to query channels", err) | 				return ginresp.APIError(g, 500, apierr.DATABASE_ERROR, "Failed to query channels", err) | ||||||
| 			} | 			} | ||||||
| 		res = langext.ArrMap(channels, func(v models.ChannelWithSubscription) models.ChannelWithSubscriptionJSON { return v.JSON(false) }) | 			return finishSuccess(ginext.JSON(http.StatusOK, response{Channels: channels})) | ||||||
|  |  | ||||||
| 		} else if sel == "all" { | 		} else if sel == "all" { | ||||||
|  |  | ||||||
| @@ -102,7 +103,7 @@ func (h APIHandler) ListChannels(g *gin.Context) ginresp.HTTPResponse { | |||||||
| 			if err != nil { | 			if err != nil { | ||||||
| 				return ginresp.APIError(g, 500, apierr.DATABASE_ERROR, "Failed to query channels", err) | 				return ginresp.APIError(g, 500, apierr.DATABASE_ERROR, "Failed to query channels", err) | ||||||
| 			} | 			} | ||||||
| 		res = langext.ArrMap(channels, func(v models.ChannelWithSubscription) models.ChannelWithSubscriptionJSON { return v.JSON(false) }) | 			return finishSuccess(ginext.JSON(http.StatusOK, response{Channels: channels})) | ||||||
|  |  | ||||||
| 		} else { | 		} else { | ||||||
|  |  | ||||||
| @@ -110,7 +111,7 @@ func (h APIHandler) ListChannels(g *gin.Context) ginresp.HTTPResponse { | |||||||
|  |  | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 	return ctx.FinishSuccess(ginresp.JSON(http.StatusOK, response{Channels: res})) | 	}) | ||||||
| } | } | ||||||
|  |  | ||||||
| // GetChannel swaggerdoc | // GetChannel swaggerdoc | ||||||
| @@ -122,26 +123,28 @@ func (h APIHandler) ListChannels(g *gin.Context) ginresp.HTTPResponse { | |||||||
| //	@Param		uid	path		string	true	"UserID" | //	@Param		uid	path		string	true	"UserID" | ||||||
| //	@Param		cid	path		string	true	"ChannelID" | //	@Param		cid	path		string	true	"ChannelID" | ||||||
| // | // | ||||||
| //	@Success	200	{object}	models.ChannelWithSubscriptionJSON | //	@Success	200	{object}	models.ChannelWithSubscription | ||||||
| //	@Failure	400	{object}	ginresp.apiError	"supplied values/parameters cannot be parsed / are invalid" | //	@Failure	400	{object}	ginresp.apiError	"supplied values/parameters cannot be parsed / are invalid" | ||||||
| //	@Failure	401	{object}	ginresp.apiError	"user is not authorized / has missing permissions" | //	@Failure	401	{object}	ginresp.apiError	"user is not authorized / has missing permissions" | ||||||
| //	@Failure	404	{object}	ginresp.apiError	"channel not found" | //	@Failure	404	{object}	ginresp.apiError	"channel not found" | ||||||
| //	@Failure	500	{object}	ginresp.apiError	"internal server error" | //	@Failure	500	{object}	ginresp.apiError	"internal server error" | ||||||
| // | // | ||||||
| //	@Router		/api/v2/users/{uid}/channels/{cid} [GET] | //	@Router		/api/v2/users/{uid}/channels/{cid} [GET] | ||||||
| func (h APIHandler) GetChannel(g *gin.Context) ginresp.HTTPResponse { | func (h APIHandler) GetChannel(pctx ginext.PreContext) ginext.HTTPResponse { | ||||||
| 	type uri struct { | 	type uri struct { | ||||||
| 		UserID    models.UserID    `uri:"uid" binding:"entityid"` | 		UserID    models.UserID    `uri:"uid" binding:"entityid"` | ||||||
| 		ChannelID models.ChannelID `uri:"cid" binding:"entityid"` | 		ChannelID models.ChannelID `uri:"cid" binding:"entityid"` | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	var u uri | 	var u uri | ||||||
| 	ctx, errResp := h.app.StartRequest(g, &u, nil, nil, nil) | 	ctx, g, errResp := pctx.URI(&u).Start() | ||||||
| 	if errResp != nil { | 	if errResp != nil { | ||||||
| 		return *errResp | 		return *errResp | ||||||
| 	} | 	} | ||||||
| 	defer ctx.Cancel() | 	defer ctx.Cancel() | ||||||
|  |  | ||||||
|  | 	return h.app.DoRequest(ctx, g, models.TLockRead, func(ctx *logic.AppContext, finishSuccess func(r ginext.HTTPResponse) ginext.HTTPResponse) ginext.HTTPResponse { | ||||||
|  |  | ||||||
| 		if permResp := ctx.CheckPermissionUserRead(u.UserID); permResp != nil { | 		if permResp := ctx.CheckPermissionUserRead(u.UserID); permResp != nil { | ||||||
| 			return *permResp | 			return *permResp | ||||||
| 		} | 		} | ||||||
| @@ -154,7 +157,9 @@ func (h APIHandler) GetChannel(g *gin.Context) ginresp.HTTPResponse { | |||||||
| 			return ginresp.APIError(g, 500, apierr.DATABASE_ERROR, "Failed to query channel", err) | 			return ginresp.APIError(g, 500, apierr.DATABASE_ERROR, "Failed to query channel", err) | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 	return ctx.FinishSuccess(ginresp.JSON(http.StatusOK, channel.JSON(true))) | 		return finishSuccess(ginext.JSONWithFilter(http.StatusOK, channel, "INCLUDE_KEY")) | ||||||
|  |  | ||||||
|  | 	}) | ||||||
| } | } | ||||||
|  |  | ||||||
| // CreateChannel swaggerdoc | // CreateChannel swaggerdoc | ||||||
| @@ -166,14 +171,14 @@ func (h APIHandler) GetChannel(g *gin.Context) ginresp.HTTPResponse { | |||||||
| //	@Param		uid			path		string						true	"UserID" | //	@Param		uid			path		string						true	"UserID" | ||||||
| //	@Param		post_body	body		handler.CreateChannel.body	false	" " | //	@Param		post_body	body		handler.CreateChannel.body	false	" " | ||||||
| // | // | ||||||
| //	@Success	200			{object}	models.ChannelWithSubscriptionJSON | //	@Success	200			{object}	models.ChannelWithSubscription | ||||||
| //	@Failure	400			{object}	ginresp.apiError	"supplied values/parameters cannot be parsed / are invalid" | //	@Failure	400			{object}	ginresp.apiError	"supplied values/parameters cannot be parsed / are invalid" | ||||||
| //	@Failure	401			{object}	ginresp.apiError	"user is not authorized / has missing permissions" | //	@Failure	401			{object}	ginresp.apiError	"user is not authorized / has missing permissions" | ||||||
| //	@Failure	409			{object}	ginresp.apiError	"channel already exists" | //	@Failure	409			{object}	ginresp.apiError	"channel already exists" | ||||||
| //	@Failure	500			{object}	ginresp.apiError	"internal server error" | //	@Failure	500			{object}	ginresp.apiError	"internal server error" | ||||||
| // | // | ||||||
| //	@Router		/api/v2/users/{uid}/channels [POST] | //	@Router		/api/v2/users/{uid}/channels [POST] | ||||||
| func (h APIHandler) CreateChannel(g *gin.Context) ginresp.HTTPResponse { | func (h APIHandler) CreateChannel(pctx ginext.PreContext) ginext.HTTPResponse { | ||||||
| 	type uri struct { | 	type uri struct { | ||||||
| 		UserID models.UserID `uri:"uid" binding:"entityid"` | 		UserID models.UserID `uri:"uid" binding:"entityid"` | ||||||
| 	} | 	} | ||||||
| @@ -186,12 +191,14 @@ func (h APIHandler) CreateChannel(g *gin.Context) ginresp.HTTPResponse { | |||||||
|  |  | ||||||
| 	var u uri | 	var u uri | ||||||
| 	var b body | 	var b body | ||||||
| 	ctx, errResp := h.app.StartRequest(g, &u, nil, &b, nil) | 	ctx, g, errResp := pctx.URI(&u).Body(&b).Start() | ||||||
| 	if errResp != nil { | 	if errResp != nil { | ||||||
| 		return *errResp | 		return *errResp | ||||||
| 	} | 	} | ||||||
| 	defer ctx.Cancel() | 	defer ctx.Cancel() | ||||||
|  |  | ||||||
|  | 	return h.app.DoRequest(ctx, g, models.TLockReadWrite, func(ctx *logic.AppContext, finishSuccess func(r ginext.HTTPResponse) ginext.HTTPResponse) ginext.HTTPResponse { | ||||||
|  |  | ||||||
| 		if permResp := ctx.CheckPermissionUserAdmin(u.UserID); permResp != nil { | 		if permResp := ctx.CheckPermissionUserAdmin(u.UserID); permResp != nil { | ||||||
| 			return *permResp | 			return *permResp | ||||||
| 		} | 		} | ||||||
| @@ -247,14 +254,15 @@ func (h APIHandler) CreateChannel(g *gin.Context) ginresp.HTTPResponse { | |||||||
| 				return ginresp.APIError(g, 500, apierr.DATABASE_ERROR, "Failed to create subscription", err) | 				return ginresp.APIError(g, 500, apierr.DATABASE_ERROR, "Failed to create subscription", err) | ||||||
| 			} | 			} | ||||||
|  |  | ||||||
| 		return ctx.FinishSuccess(ginresp.JSON(http.StatusOK, channel.WithSubscription(langext.Ptr(sub)).JSON(true))) | 			return finishSuccess(ginext.JSONWithFilter(http.StatusOK, channel.WithSubscription(langext.Ptr(sub)), "INCLUDE_KEY")) | ||||||
|  |  | ||||||
| 		} else { | 		} else { | ||||||
|  |  | ||||||
| 		return ctx.FinishSuccess(ginresp.JSON(http.StatusOK, channel.WithSubscription(nil).JSON(true))) | 			return finishSuccess(ginext.JSONWithFilter(http.StatusOK, channel.WithSubscription(nil), "INCLUDE_KEY")) | ||||||
|  |  | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
|  | 	}) | ||||||
| } | } | ||||||
|  |  | ||||||
| // UpdateChannel swaggerdoc | // UpdateChannel swaggerdoc | ||||||
| @@ -270,14 +278,14 @@ func (h APIHandler) CreateChannel(g *gin.Context) ginresp.HTTPResponse { | |||||||
| //	@Param		send_key		body		string	false	"Send `true` to create a new send_key" | //	@Param		send_key		body		string	false	"Send `true` to create a new send_key" | ||||||
| //	@Param		display_name	body		string	false	"Change the cahnnel display-name (only chnages to lowercase/uppercase are allowed - internal_name must stay the same)" | //	@Param		display_name	body		string	false	"Change the cahnnel display-name (only chnages to lowercase/uppercase are allowed - internal_name must stay the same)" | ||||||
| // | // | ||||||
| //	@Success	200				{object}	models.ChannelWithSubscriptionJSON | //	@Success	200				{object}	models.ChannelWithSubscription | ||||||
| //	@Failure	400				{object}	ginresp.apiError	"supplied values/parameters cannot be parsed / are invalid" | //	@Failure	400				{object}	ginresp.apiError	"supplied values/parameters cannot be parsed / are invalid" | ||||||
| //	@Failure	401				{object}	ginresp.apiError	"user is not authorized / has missing permissions" | //	@Failure	401				{object}	ginresp.apiError	"user is not authorized / has missing permissions" | ||||||
| //	@Failure	404				{object}	ginresp.apiError	"channel not found" | //	@Failure	404				{object}	ginresp.apiError	"channel not found" | ||||||
| //	@Failure	500				{object}	ginresp.apiError	"internal server error" | //	@Failure	500				{object}	ginresp.apiError	"internal server error" | ||||||
| // | // | ||||||
| //	@Router		/api/v2/users/{uid}/channels/{cid} [PATCH] | //	@Router		/api/v2/users/{uid}/channels/{cid} [PATCH] | ||||||
| func (h APIHandler) UpdateChannel(g *gin.Context) ginresp.HTTPResponse { | func (h APIHandler) UpdateChannel(pctx ginext.PreContext) ginext.HTTPResponse { | ||||||
| 	type uri struct { | 	type uri struct { | ||||||
| 		UserID    models.UserID    `uri:"uid" binding:"entityid"` | 		UserID    models.UserID    `uri:"uid" binding:"entityid"` | ||||||
| 		ChannelID models.ChannelID `uri:"cid" binding:"entityid"` | 		ChannelID models.ChannelID `uri:"cid" binding:"entityid"` | ||||||
| @@ -290,12 +298,14 @@ func (h APIHandler) UpdateChannel(g *gin.Context) ginresp.HTTPResponse { | |||||||
|  |  | ||||||
| 	var u uri | 	var u uri | ||||||
| 	var b body | 	var b body | ||||||
| 	ctx, errResp := h.app.StartRequest(g, &u, nil, &b, nil) | 	ctx, g, errResp := pctx.URI(&u).Body(&b).Start() | ||||||
| 	if errResp != nil { | 	if errResp != nil { | ||||||
| 		return *errResp | 		return *errResp | ||||||
| 	} | 	} | ||||||
| 	defer ctx.Cancel() | 	defer ctx.Cancel() | ||||||
|  |  | ||||||
|  | 	return h.app.DoRequest(ctx, g, models.TLockReadWrite, func(ctx *logic.AppContext, finishSuccess func(r ginext.HTTPResponse) ginext.HTTPResponse) ginext.HTTPResponse { | ||||||
|  |  | ||||||
| 		if permResp := ctx.CheckPermissionUserAdmin(u.UserID); permResp != nil { | 		if permResp := ctx.CheckPermissionUserAdmin(u.UserID); permResp != nil { | ||||||
| 			return *permResp | 			return *permResp | ||||||
| 		} | 		} | ||||||
| @@ -367,7 +377,9 @@ func (h APIHandler) UpdateChannel(g *gin.Context) ginresp.HTTPResponse { | |||||||
| 			return ginresp.APIError(g, 500, apierr.DATABASE_ERROR, "Failed to query (updated) channel", err) | 			return ginresp.APIError(g, 500, apierr.DATABASE_ERROR, "Failed to query (updated) channel", err) | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 	return ctx.FinishSuccess(ginresp.JSON(http.StatusOK, channel.JSON(true))) | 		return finishSuccess(ginext.JSONWithFilter(http.StatusOK, channel, "INCLUDE_KEY")) | ||||||
|  |  | ||||||
|  | 	}) | ||||||
| } | } | ||||||
|  |  | ||||||
| // ListChannelMessages swaggerdoc | // ListChannelMessages swaggerdoc | ||||||
| @@ -391,7 +403,7 @@ func (h APIHandler) UpdateChannel(g *gin.Context) ginresp.HTTPResponse { | |||||||
| //	@Failure		500			{object}	ginresp.apiError	"internal server error" | //	@Failure		500			{object}	ginresp.apiError	"internal server error" | ||||||
| // | // | ||||||
| //	@Router			/api/v2/users/{uid}/channels/{cid}/messages [GET] | //	@Router			/api/v2/users/{uid}/channels/{cid}/messages [GET] | ||||||
| func (h APIHandler) ListChannelMessages(g *gin.Context) ginresp.HTTPResponse { | func (h APIHandler) ListChannelMessages(pctx ginext.PreContext) ginext.HTTPResponse { | ||||||
| 	type uri struct { | 	type uri struct { | ||||||
| 		ChannelUserID models.UserID    `uri:"uid" binding:"entityid"` | 		ChannelUserID models.UserID    `uri:"uid" binding:"entityid"` | ||||||
| 		ChannelID     models.ChannelID `uri:"cid" binding:"entityid"` | 		ChannelID     models.ChannelID `uri:"cid" binding:"entityid"` | ||||||
| @@ -403,19 +415,21 @@ func (h APIHandler) ListChannelMessages(g *gin.Context) ginresp.HTTPResponse { | |||||||
| 		Trimmed       *bool   `json:"trimmed"         form:"trimmed"` | 		Trimmed       *bool   `json:"trimmed"         form:"trimmed"` | ||||||
| 	} | 	} | ||||||
| 	type response struct { | 	type response struct { | ||||||
| 		Messages      []models.MessageJSON `json:"messages"` | 		Messages      []models.Message `json:"messages"` | ||||||
| 		NextPageToken string           `json:"next_page_token"` | 		NextPageToken string           `json:"next_page_token"` | ||||||
| 		PageSize      int              `json:"page_size"` | 		PageSize      int              `json:"page_size"` | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	var u uri | 	var u uri | ||||||
| 	var q query | 	var q query | ||||||
| 	ctx, errResp := h.app.StartRequest(g, &u, &q, nil, nil) | 	ctx, g, errResp := pctx.URI(&u).Query(&q).Start() | ||||||
| 	if errResp != nil { | 	if errResp != nil { | ||||||
| 		return *errResp | 		return *errResp | ||||||
| 	} | 	} | ||||||
| 	defer ctx.Cancel() | 	defer ctx.Cancel() | ||||||
|  |  | ||||||
|  | 	return h.app.DoRequest(ctx, g, models.TLockRead, func(ctx *logic.AppContext, finishSuccess func(r ginext.HTTPResponse) ginext.HTTPResponse) ginext.HTTPResponse { | ||||||
|  |  | ||||||
| 		trimmed := langext.Coalesce(q.Trimmed, true) | 		trimmed := langext.Coalesce(q.Trimmed, true) | ||||||
|  |  | ||||||
| 		maxPageSize := langext.Conditional(trimmed, 16, 256) | 		maxPageSize := langext.Conditional(trimmed, 16, 256) | ||||||
| @@ -448,12 +462,12 @@ func (h APIHandler) ListChannelMessages(g *gin.Context) ginresp.HTTPResponse { | |||||||
| 			return ginresp.APIError(g, 500, apierr.DATABASE_ERROR, "Failed to query messages", err) | 			return ginresp.APIError(g, 500, apierr.DATABASE_ERROR, "Failed to query messages", err) | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 	var res []models.MessageJSON |  | ||||||
| 		if trimmed { | 		if trimmed { | ||||||
| 		res = langext.ArrMap(messages, func(v models.Message) models.MessageJSON { return v.TrimmedJSON() }) | 			res := langext.ArrMap(messages, func(v models.Message) models.Message { return v.Trim() }) | ||||||
|  | 			return finishSuccess(ginext.JSON(http.StatusOK, response{Messages: res, NextPageToken: npt.Token(), PageSize: pageSize})) | ||||||
| 		} else { | 		} else { | ||||||
| 		res = langext.ArrMap(messages, func(v models.Message) models.MessageJSON { return v.FullJSON() }) | 			return finishSuccess(ginext.JSON(http.StatusOK, response{Messages: messages, NextPageToken: npt.Token(), PageSize: pageSize})) | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 	return ctx.FinishSuccess(ginresp.JSON(http.StatusOK, response{Messages: res, NextPageToken: npt.Token(), PageSize: pageSize})) | 	}) | ||||||
| } | } | ||||||
|   | |||||||
| @@ -3,10 +3,11 @@ package handler | |||||||
| import ( | import ( | ||||||
| 	"blackforestbytes.com/simplecloudnotifier/api/apierr" | 	"blackforestbytes.com/simplecloudnotifier/api/apierr" | ||||||
| 	"blackforestbytes.com/simplecloudnotifier/api/ginresp" | 	"blackforestbytes.com/simplecloudnotifier/api/ginresp" | ||||||
|  | 	"blackforestbytes.com/simplecloudnotifier/logic" | ||||||
| 	"blackforestbytes.com/simplecloudnotifier/models" | 	"blackforestbytes.com/simplecloudnotifier/models" | ||||||
| 	"database/sql" | 	"database/sql" | ||||||
| 	"errors" | 	"errors" | ||||||
| 	"github.com/gin-gonic/gin" | 	"gogs.mikescher.com/BlackForestBytes/goext/ginext" | ||||||
| 	"gogs.mikescher.com/BlackForestBytes/goext/langext" | 	"gogs.mikescher.com/BlackForestBytes/goext/langext" | ||||||
| 	"net/http" | 	"net/http" | ||||||
| ) | ) | ||||||
| @@ -25,21 +26,23 @@ import ( | |||||||
| //	@Failure	500	{object}	ginresp.apiError	"internal server error" | //	@Failure	500	{object}	ginresp.apiError	"internal server error" | ||||||
| // | // | ||||||
| //	@Router		/api/v2/users/{uid}/clients [GET] | //	@Router		/api/v2/users/{uid}/clients [GET] | ||||||
| func (h APIHandler) ListClients(g *gin.Context) ginresp.HTTPResponse { | func (h APIHandler) ListClients(pctx ginext.PreContext) ginext.HTTPResponse { | ||||||
| 	type uri struct { | 	type uri struct { | ||||||
| 		UserID models.UserID `uri:"uid" binding:"entityid"` | 		UserID models.UserID `uri:"uid" binding:"entityid"` | ||||||
| 	} | 	} | ||||||
| 	type response struct { | 	type response struct { | ||||||
| 		Clients []models.ClientJSON `json:"clients"` | 		Clients []models.Client `json:"clients"` | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	var u uri | 	var u uri | ||||||
| 	ctx, errResp := h.app.StartRequest(g, &u, nil, nil, nil) | 	ctx, g, errResp := pctx.URI(&u).Start() | ||||||
| 	if errResp != nil { | 	if errResp != nil { | ||||||
| 		return *errResp | 		return *errResp | ||||||
| 	} | 	} | ||||||
| 	defer ctx.Cancel() | 	defer ctx.Cancel() | ||||||
|  |  | ||||||
|  | 	return h.app.DoRequest(ctx, g, models.TLockRead, func(ctx *logic.AppContext, finishSuccess func(r ginext.HTTPResponse) ginext.HTTPResponse) ginext.HTTPResponse { | ||||||
|  |  | ||||||
| 		if permResp := ctx.CheckPermissionUserRead(u.UserID); permResp != nil { | 		if permResp := ctx.CheckPermissionUserRead(u.UserID); permResp != nil { | ||||||
| 			return *permResp | 			return *permResp | ||||||
| 		} | 		} | ||||||
| @@ -49,9 +52,9 @@ func (h APIHandler) ListClients(g *gin.Context) ginresp.HTTPResponse { | |||||||
| 			return ginresp.APIError(g, 500, apierr.DATABASE_ERROR, "Failed to query clients", err) | 			return ginresp.APIError(g, 500, apierr.DATABASE_ERROR, "Failed to query clients", err) | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 	res := langext.ArrMap(clients, func(v models.Client) models.ClientJSON { return v.JSON() }) | 		return finishSuccess(ginext.JSON(http.StatusOK, response{Clients: clients})) | ||||||
|  |  | ||||||
| 	return ctx.FinishSuccess(ginresp.JSON(http.StatusOK, response{Clients: res})) | 	}) | ||||||
| } | } | ||||||
|  |  | ||||||
| // GetClient swaggerdoc | // GetClient swaggerdoc | ||||||
| @@ -63,26 +66,28 @@ func (h APIHandler) ListClients(g *gin.Context) ginresp.HTTPResponse { | |||||||
| //	@Param		uid	path		string	true	"UserID" | //	@Param		uid	path		string	true	"UserID" | ||||||
| //	@Param		cid	path		string	true	"ClientID" | //	@Param		cid	path		string	true	"ClientID" | ||||||
| // | // | ||||||
| //	@Success	200	{object}	models.ClientJSON | //	@Success	200	{object}	models.Client | ||||||
| //	@Failure	400	{object}	ginresp.apiError	"supplied values/parameters cannot be parsed / are invalid" | //	@Failure	400	{object}	ginresp.apiError	"supplied values/parameters cannot be parsed / are invalid" | ||||||
| //	@Failure	401	{object}	ginresp.apiError	"user is not authorized / has missing permissions" | //	@Failure	401	{object}	ginresp.apiError	"user is not authorized / has missing permissions" | ||||||
| //	@Failure	404	{object}	ginresp.apiError	"client not found" | //	@Failure	404	{object}	ginresp.apiError	"client not found" | ||||||
| //	@Failure	500	{object}	ginresp.apiError	"internal server error" | //	@Failure	500	{object}	ginresp.apiError	"internal server error" | ||||||
| // | // | ||||||
| //	@Router		/api/v2/users/{uid}/clients/{cid} [GET] | //	@Router		/api/v2/users/{uid}/clients/{cid} [GET] | ||||||
| func (h APIHandler) GetClient(g *gin.Context) ginresp.HTTPResponse { | func (h APIHandler) GetClient(pctx ginext.PreContext) ginext.HTTPResponse { | ||||||
| 	type uri struct { | 	type uri struct { | ||||||
| 		UserID   models.UserID   `uri:"uid" binding:"entityid"` | 		UserID   models.UserID   `uri:"uid" binding:"entityid"` | ||||||
| 		ClientID models.ClientID `uri:"cid" binding:"entityid"` | 		ClientID models.ClientID `uri:"cid" binding:"entityid"` | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	var u uri | 	var u uri | ||||||
| 	ctx, errResp := h.app.StartRequest(g, &u, nil, nil, nil) | 	ctx, g, errResp := pctx.URI(&u).Start() | ||||||
| 	if errResp != nil { | 	if errResp != nil { | ||||||
| 		return *errResp | 		return *errResp | ||||||
| 	} | 	} | ||||||
| 	defer ctx.Cancel() | 	defer ctx.Cancel() | ||||||
|  |  | ||||||
|  | 	return h.app.DoRequest(ctx, g, models.TLockRead, func(ctx *logic.AppContext, finishSuccess func(r ginext.HTTPResponse) ginext.HTTPResponse) ginext.HTTPResponse { | ||||||
|  |  | ||||||
| 		if permResp := ctx.CheckPermissionUserRead(u.UserID); permResp != nil { | 		if permResp := ctx.CheckPermissionUserRead(u.UserID); permResp != nil { | ||||||
| 			return *permResp | 			return *permResp | ||||||
| 		} | 		} | ||||||
| @@ -95,7 +100,9 @@ func (h APIHandler) GetClient(g *gin.Context) ginresp.HTTPResponse { | |||||||
| 			return ginresp.APIError(g, 500, apierr.DATABASE_ERROR, "Failed to query client", err) | 			return ginresp.APIError(g, 500, apierr.DATABASE_ERROR, "Failed to query client", err) | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 	return ctx.FinishSuccess(ginresp.JSON(http.StatusOK, client.JSON())) | 		return finishSuccess(ginext.JSON(http.StatusOK, client)) | ||||||
|  |  | ||||||
|  | 	}) | ||||||
| } | } | ||||||
|  |  | ||||||
| // AddClient swaggerdoc | // AddClient swaggerdoc | ||||||
| @@ -108,13 +115,13 @@ func (h APIHandler) GetClient(g *gin.Context) ginresp.HTTPResponse { | |||||||
| // | // | ||||||
| //	@Param		post_body	body		handler.AddClient.body	false	" " | //	@Param		post_body	body		handler.AddClient.body	false	" " | ||||||
| // | // | ||||||
| //	@Success	200			{object}	models.ClientJSON | //	@Success	200			{object}	models.Client | ||||||
| //	@Failure	400			{object}	ginresp.apiError	"supplied values/parameters cannot be parsed / are invalid" | //	@Failure	400			{object}	ginresp.apiError	"supplied values/parameters cannot be parsed / are invalid" | ||||||
| //	@Failure	401			{object}	ginresp.apiError	"user is not authorized / has missing permissions" | //	@Failure	401			{object}	ginresp.apiError	"user is not authorized / has missing permissions" | ||||||
| //	@Failure	500			{object}	ginresp.apiError	"internal server error" | //	@Failure	500			{object}	ginresp.apiError	"internal server error" | ||||||
| // | // | ||||||
| //	@Router		/api/v2/users/{uid}/clients [POST] | //	@Router		/api/v2/users/{uid}/clients [POST] | ||||||
| func (h APIHandler) AddClient(g *gin.Context) ginresp.HTTPResponse { | func (h APIHandler) AddClient(pctx ginext.PreContext) ginext.HTTPResponse { | ||||||
| 	type uri struct { | 	type uri struct { | ||||||
| 		UserID models.UserID `uri:"uid" binding:"entityid"` | 		UserID models.UserID `uri:"uid" binding:"entityid"` | ||||||
| 	} | 	} | ||||||
| @@ -128,12 +135,14 @@ func (h APIHandler) AddClient(g *gin.Context) ginresp.HTTPResponse { | |||||||
|  |  | ||||||
| 	var u uri | 	var u uri | ||||||
| 	var b body | 	var b body | ||||||
| 	ctx, errResp := h.app.StartRequest(g, &u, nil, &b, nil) | 	ctx, g, errResp := pctx.URI(&u).Body(&b).Start() | ||||||
| 	if errResp != nil { | 	if errResp != nil { | ||||||
| 		return *errResp | 		return *errResp | ||||||
| 	} | 	} | ||||||
| 	defer ctx.Cancel() | 	defer ctx.Cancel() | ||||||
|  |  | ||||||
|  | 	return h.app.DoRequest(ctx, g, models.TLockReadWrite, func(ctx *logic.AppContext, finishSuccess func(r ginext.HTTPResponse) ginext.HTTPResponse) ginext.HTTPResponse { | ||||||
|  |  | ||||||
| 		if !b.ClientType.Valid() { | 		if !b.ClientType.Valid() { | ||||||
| 			return ginresp.APIError(g, 400, apierr.INVALID_CLIENTTYPE, "Invalid ClientType", nil) | 			return ginresp.APIError(g, 400, apierr.INVALID_CLIENTTYPE, "Invalid ClientType", nil) | ||||||
| 		} | 		} | ||||||
| @@ -153,7 +162,9 @@ func (h APIHandler) AddClient(g *gin.Context) ginresp.HTTPResponse { | |||||||
| 			return ginresp.APIError(g, 500, apierr.DATABASE_ERROR, "Failed to create client in db", err) | 			return ginresp.APIError(g, 500, apierr.DATABASE_ERROR, "Failed to create client in db", err) | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 	return ctx.FinishSuccess(ginresp.JSON(http.StatusOK, client.JSON())) | 		return finishSuccess(ginext.JSON(http.StatusOK, client)) | ||||||
|  |  | ||||||
|  | 	}) | ||||||
| } | } | ||||||
|  |  | ||||||
| // DeleteClient swaggerdoc | // DeleteClient swaggerdoc | ||||||
| @@ -165,26 +176,28 @@ func (h APIHandler) AddClient(g *gin.Context) ginresp.HTTPResponse { | |||||||
| //	@Param		uid	path		string	true	"UserID" | //	@Param		uid	path		string	true	"UserID" | ||||||
| //	@Param		cid	path		string	true	"ClientID" | //	@Param		cid	path		string	true	"ClientID" | ||||||
| // | // | ||||||
| //	@Success	200	{object}	models.ClientJSON | //	@Success	200	{object}	models.Client | ||||||
| //	@Failure	400	{object}	ginresp.apiError	"supplied values/parameters cannot be parsed / are invalid" | //	@Failure	400	{object}	ginresp.apiError	"supplied values/parameters cannot be parsed / are invalid" | ||||||
| //	@Failure	401	{object}	ginresp.apiError	"user is not authorized / has missing permissions" | //	@Failure	401	{object}	ginresp.apiError	"user is not authorized / has missing permissions" | ||||||
| //	@Failure	404	{object}	ginresp.apiError	"client not found" | //	@Failure	404	{object}	ginresp.apiError	"client not found" | ||||||
| //	@Failure	500	{object}	ginresp.apiError	"internal server error" | //	@Failure	500	{object}	ginresp.apiError	"internal server error" | ||||||
| // | // | ||||||
| //	@Router		/api/v2/users/{uid}/clients/{cid} [DELETE] | //	@Router		/api/v2/users/{uid}/clients/{cid} [DELETE] | ||||||
| func (h APIHandler) DeleteClient(g *gin.Context) ginresp.HTTPResponse { | func (h APIHandler) DeleteClient(pctx ginext.PreContext) ginext.HTTPResponse { | ||||||
| 	type uri struct { | 	type uri struct { | ||||||
| 		UserID   models.UserID   `uri:"uid" binding:"entityid"` | 		UserID   models.UserID   `uri:"uid" binding:"entityid"` | ||||||
| 		ClientID models.ClientID `uri:"cid" binding:"entityid"` | 		ClientID models.ClientID `uri:"cid" binding:"entityid"` | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	var u uri | 	var u uri | ||||||
| 	ctx, errResp := h.app.StartRequest(g, &u, nil, nil, nil) | 	ctx, g, errResp := pctx.URI(&u).Start() | ||||||
| 	if errResp != nil { | 	if errResp != nil { | ||||||
| 		return *errResp | 		return *errResp | ||||||
| 	} | 	} | ||||||
| 	defer ctx.Cancel() | 	defer ctx.Cancel() | ||||||
|  |  | ||||||
|  | 	return h.app.DoRequest(ctx, g, models.TLockReadWrite, func(ctx *logic.AppContext, finishSuccess func(r ginext.HTTPResponse) ginext.HTTPResponse) ginext.HTTPResponse { | ||||||
|  |  | ||||||
| 		if permResp := ctx.CheckPermissionUserAdmin(u.UserID); permResp != nil { | 		if permResp := ctx.CheckPermissionUserAdmin(u.UserID); permResp != nil { | ||||||
| 			return *permResp | 			return *permResp | ||||||
| 		} | 		} | ||||||
| @@ -202,7 +215,9 @@ func (h APIHandler) DeleteClient(g *gin.Context) ginresp.HTTPResponse { | |||||||
| 			return ginresp.APIError(g, 500, apierr.DATABASE_ERROR, "Failed to delete client", err) | 			return ginresp.APIError(g, 500, apierr.DATABASE_ERROR, "Failed to delete client", err) | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 	return ctx.FinishSuccess(ginresp.JSON(http.StatusOK, client.JSON())) | 		return finishSuccess(ginext.JSON(http.StatusOK, client)) | ||||||
|  |  | ||||||
|  | 	}) | ||||||
| } | } | ||||||
|  |  | ||||||
| // UpdateClient swaggerdoc | // UpdateClient swaggerdoc | ||||||
| @@ -218,14 +233,14 @@ func (h APIHandler) DeleteClient(g *gin.Context) ginresp.HTTPResponse { | |||||||
| //	@Param			clientname	body		string	false	"Change the clientname (send an empty string to clear it)" | //	@Param			clientname	body		string	false	"Change the clientname (send an empty string to clear it)" | ||||||
| //	@Param			pro_token	body		string	false	"Send a verification of premium purchase" | //	@Param			pro_token	body		string	false	"Send a verification of premium purchase" | ||||||
| // | // | ||||||
| //	@Success		200			{object}	models.ClientJSON | //	@Success		200			{object}	models.Client | ||||||
| //	@Failure		400			{object}	ginresp.apiError	"supplied values/parameters cannot be parsed / are invalid" | //	@Failure		400			{object}	ginresp.apiError	"supplied values/parameters cannot be parsed / are invalid" | ||||||
| //	@Failure		401			{object}	ginresp.apiError	"client is not authorized / has missing permissions" | //	@Failure		401			{object}	ginresp.apiError	"client is not authorized / has missing permissions" | ||||||
| //	@Failure		404			{object}	ginresp.apiError	"client not found" | //	@Failure		404			{object}	ginresp.apiError	"client not found" | ||||||
| //	@Failure		500			{object}	ginresp.apiError	"internal server error" | //	@Failure		500			{object}	ginresp.apiError	"internal server error" | ||||||
| // | // | ||||||
| //	@Router			/api/v2/users/{uid}/clients/{cid} [PATCH] | //	@Router			/api/v2/users/{uid}/clients/{cid} [PATCH] | ||||||
| func (h APIHandler) UpdateClient(g *gin.Context) ginresp.HTTPResponse { | func (h APIHandler) UpdateClient(pctx ginext.PreContext) ginext.HTTPResponse { | ||||||
| 	type uri struct { | 	type uri struct { | ||||||
| 		UserID   models.UserID   `uri:"uid" binding:"entityid"` | 		UserID   models.UserID   `uri:"uid" binding:"entityid"` | ||||||
| 		ClientID models.ClientID `uri:"cid" binding:"entityid"` | 		ClientID models.ClientID `uri:"cid" binding:"entityid"` | ||||||
| @@ -239,12 +254,14 @@ func (h APIHandler) UpdateClient(g *gin.Context) ginresp.HTTPResponse { | |||||||
|  |  | ||||||
| 	var u uri | 	var u uri | ||||||
| 	var b body | 	var b body | ||||||
| 	ctx, errResp := h.app.StartRequest(g, &u, nil, &b, nil) | 	ctx, g, errResp := pctx.URI(&u).Body(&b).Start() | ||||||
| 	if errResp != nil { | 	if errResp != nil { | ||||||
| 		return *errResp | 		return *errResp | ||||||
| 	} | 	} | ||||||
| 	defer ctx.Cancel() | 	defer ctx.Cancel() | ||||||
|  |  | ||||||
|  | 	return h.app.DoRequest(ctx, g, models.TLockReadWrite, func(ctx *logic.AppContext, finishSuccess func(r ginext.HTTPResponse) ginext.HTTPResponse) ginext.HTTPResponse { | ||||||
|  |  | ||||||
| 		if permResp := ctx.CheckPermissionUserAdmin(u.UserID); permResp != nil { | 		if permResp := ctx.CheckPermissionUserAdmin(u.UserID); permResp != nil { | ||||||
| 			return *permResp | 			return *permResp | ||||||
| 		} | 		} | ||||||
| @@ -303,5 +320,7 @@ func (h APIHandler) UpdateClient(g *gin.Context) ginresp.HTTPResponse { | |||||||
| 			return ginresp.APIError(g, 500, apierr.DATABASE_ERROR, "Failed to query (updated) client", err) | 			return ginresp.APIError(g, 500, apierr.DATABASE_ERROR, "Failed to query (updated) client", err) | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 	return ctx.FinishSuccess(ginresp.JSON(http.StatusOK, client.JSON())) | 		return finishSuccess(ginext.JSON(http.StatusOK, client)) | ||||||
|  |  | ||||||
|  | 	}) | ||||||
| } | } | ||||||
|   | |||||||
| @@ -3,10 +3,11 @@ package handler | |||||||
| import ( | import ( | ||||||
| 	"blackforestbytes.com/simplecloudnotifier/api/apierr" | 	"blackforestbytes.com/simplecloudnotifier/api/apierr" | ||||||
| 	"blackforestbytes.com/simplecloudnotifier/api/ginresp" | 	"blackforestbytes.com/simplecloudnotifier/api/ginresp" | ||||||
|  | 	"blackforestbytes.com/simplecloudnotifier/logic" | ||||||
| 	"blackforestbytes.com/simplecloudnotifier/models" | 	"blackforestbytes.com/simplecloudnotifier/models" | ||||||
| 	"database/sql" | 	"database/sql" | ||||||
| 	"errors" | 	"errors" | ||||||
| 	"github.com/gin-gonic/gin" | 	"gogs.mikescher.com/BlackForestBytes/goext/ginext" | ||||||
| 	"gogs.mikescher.com/BlackForestBytes/goext/langext" | 	"gogs.mikescher.com/BlackForestBytes/goext/langext" | ||||||
| 	"net/http" | 	"net/http" | ||||||
| ) | ) | ||||||
| @@ -27,21 +28,23 @@ import ( | |||||||
| //	@Failure		500	{object}	ginresp.apiError	"internal server error" | //	@Failure		500	{object}	ginresp.apiError	"internal server error" | ||||||
| // | // | ||||||
| //	@Router			/api/v2/users/{uid}/keys [GET] | //	@Router			/api/v2/users/{uid}/keys [GET] | ||||||
| func (h APIHandler) ListUserKeys(g *gin.Context) ginresp.HTTPResponse { | func (h APIHandler) ListUserKeys(pctx ginext.PreContext) ginext.HTTPResponse { | ||||||
| 	type uri struct { | 	type uri struct { | ||||||
| 		UserID models.UserID `uri:"uid" binding:"entityid"` | 		UserID models.UserID `uri:"uid" binding:"entityid"` | ||||||
| 	} | 	} | ||||||
| 	type response struct { | 	type response struct { | ||||||
| 		Keys []models.KeyTokenJSON `json:"keys"` | 		Keys []models.KeyToken `json:"keys"` | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	var u uri | 	var u uri | ||||||
| 	ctx, errResp := h.app.StartRequest(g, &u, nil, nil, nil) | 	ctx, g, errResp := pctx.URI(&u).Start() | ||||||
| 	if errResp != nil { | 	if errResp != nil { | ||||||
| 		return *errResp | 		return *errResp | ||||||
| 	} | 	} | ||||||
| 	defer ctx.Cancel() | 	defer ctx.Cancel() | ||||||
|  |  | ||||||
|  | 	return h.app.DoRequest(ctx, g, models.TLockRead, func(ctx *logic.AppContext, finishSuccess func(r ginext.HTTPResponse) ginext.HTTPResponse) ginext.HTTPResponse { | ||||||
|  |  | ||||||
| 		if permResp := ctx.CheckPermissionUserAdmin(u.UserID); permResp != nil { | 		if permResp := ctx.CheckPermissionUserAdmin(u.UserID); permResp != nil { | ||||||
| 			return *permResp | 			return *permResp | ||||||
| 		} | 		} | ||||||
| @@ -51,9 +54,9 @@ func (h APIHandler) ListUserKeys(g *gin.Context) ginresp.HTTPResponse { | |||||||
| 			return ginresp.APIError(g, 500, apierr.DATABASE_ERROR, "Failed to query keys", err) | 			return ginresp.APIError(g, 500, apierr.DATABASE_ERROR, "Failed to query keys", err) | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 	res := langext.ArrMap(toks, func(v models.KeyToken) models.KeyTokenJSON { return v.JSON() }) | 		return finishSuccess(ginext.JSON(http.StatusOK, response{Keys: toks})) | ||||||
|  |  | ||||||
| 	return ctx.FinishSuccess(ginresp.JSON(http.StatusOK, response{Keys: res})) | 	}) | ||||||
| } | } | ||||||
|  |  | ||||||
| // GetCurrentUserKey swaggerdoc | // GetCurrentUserKey swaggerdoc | ||||||
| @@ -66,25 +69,27 @@ func (h APIHandler) ListUserKeys(g *gin.Context) ginresp.HTTPResponse { | |||||||
| //	@Param			uid	path		string	true	"UserID" | //	@Param			uid	path		string	true	"UserID" | ||||||
| //	@Param			kid	path		string	true	"TokenKeyID" | //	@Param			kid	path		string	true	"TokenKeyID" | ||||||
| // | // | ||||||
| //	@Success		200	{object}	models.KeyTokenWithTokenJSON | //	@Success		200	{object}	models.KeyToken | ||||||
| //	@Failure		400	{object}	ginresp.apiError	"supplied values/parameters cannot be parsed / are invalid" | //	@Failure		400	{object}	ginresp.apiError	"supplied values/parameters cannot be parsed / are invalid" | ||||||
| //	@Failure		401	{object}	ginresp.apiError	"user is not authorized / has missing permissions" | //	@Failure		401	{object}	ginresp.apiError	"user is not authorized / has missing permissions" | ||||||
| //	@Failure		404	{object}	ginresp.apiError	"message not found" | //	@Failure		404	{object}	ginresp.apiError	"message not found" | ||||||
| //	@Failure		500	{object}	ginresp.apiError	"internal server error" | //	@Failure		500	{object}	ginresp.apiError	"internal server error" | ||||||
| // | // | ||||||
| //	@Router			/api/v2/users/{uid}/keys/current [GET] | //	@Router			/api/v2/users/{uid}/keys/current [GET] | ||||||
| func (h APIHandler) GetCurrentUserKey(g *gin.Context) ginresp.HTTPResponse { | func (h APIHandler) GetCurrentUserKey(pctx ginext.PreContext) ginext.HTTPResponse { | ||||||
| 	type uri struct { | 	type uri struct { | ||||||
| 		UserID models.UserID `uri:"uid" binding:"entityid"` | 		UserID models.UserID `uri:"uid" binding:"entityid"` | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	var u uri | 	var u uri | ||||||
| 	ctx, errResp := h.app.StartRequest(g, &u, nil, nil, nil) | 	ctx, g, errResp := pctx.URI(&u).Start() | ||||||
| 	if errResp != nil { | 	if errResp != nil { | ||||||
| 		return *errResp | 		return *errResp | ||||||
| 	} | 	} | ||||||
| 	defer ctx.Cancel() | 	defer ctx.Cancel() | ||||||
|  |  | ||||||
|  | 	return h.app.DoRequest(ctx, g, models.TLockRead, func(ctx *logic.AppContext, finishSuccess func(r ginext.HTTPResponse) ginext.HTTPResponse) ginext.HTTPResponse { | ||||||
|  |  | ||||||
| 		if permResp := ctx.CheckPermissionAny(); permResp != nil { | 		if permResp := ctx.CheckPermissionAny(); permResp != nil { | ||||||
| 			return *permResp | 			return *permResp | ||||||
| 		} | 		} | ||||||
| @@ -102,7 +107,9 @@ func (h APIHandler) GetCurrentUserKey(g *gin.Context) ginresp.HTTPResponse { | |||||||
| 			return ginresp.APIError(g, 500, apierr.DATABASE_ERROR, "Failed to query client", err) | 			return ginresp.APIError(g, 500, apierr.DATABASE_ERROR, "Failed to query client", err) | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 	return ctx.FinishSuccess(ginresp.JSON(http.StatusOK, keytoken.JSON().WithToken(keytoken.Token))) | 		return finishSuccess(ginext.JSONWithFilter(http.StatusOK, keytoken, "INCLUDE_TOKEN")) | ||||||
|  |  | ||||||
|  | 	}) | ||||||
| } | } | ||||||
|  |  | ||||||
| // GetUserKey swaggerdoc | // GetUserKey swaggerdoc | ||||||
| @@ -115,26 +122,28 @@ func (h APIHandler) GetCurrentUserKey(g *gin.Context) ginresp.HTTPResponse { | |||||||
| //	@Param			uid	path		string	true	"UserID" | //	@Param			uid	path		string	true	"UserID" | ||||||
| //	@Param			kid	path		string	true	"TokenKeyID" | //	@Param			kid	path		string	true	"TokenKeyID" | ||||||
| // | // | ||||||
| //	@Success		200	{object}	models.KeyTokenJSON | //	@Success		200	{object}	models.KeyToken | ||||||
| //	@Failure		400	{object}	ginresp.apiError	"supplied values/parameters cannot be parsed / are invalid" | //	@Failure		400	{object}	ginresp.apiError	"supplied values/parameters cannot be parsed / are invalid" | ||||||
| //	@Failure		401	{object}	ginresp.apiError	"user is not authorized / has missing permissions" | //	@Failure		401	{object}	ginresp.apiError	"user is not authorized / has missing permissions" | ||||||
| //	@Failure		404	{object}	ginresp.apiError	"message not found" | //	@Failure		404	{object}	ginresp.apiError	"message not found" | ||||||
| //	@Failure		500	{object}	ginresp.apiError	"internal server error" | //	@Failure		500	{object}	ginresp.apiError	"internal server error" | ||||||
| // | // | ||||||
| //	@Router			/api/v2/users/{uid}/keys/{kid} [GET] | //	@Router			/api/v2/users/{uid}/keys/{kid} [GET] | ||||||
| func (h APIHandler) GetUserKey(g *gin.Context) ginresp.HTTPResponse { | func (h APIHandler) GetUserKey(pctx ginext.PreContext) ginext.HTTPResponse { | ||||||
| 	type uri struct { | 	type uri struct { | ||||||
| 		UserID models.UserID     `uri:"uid" binding:"entityid"` | 		UserID models.UserID     `uri:"uid" binding:"entityid"` | ||||||
| 		KeyID  models.KeyTokenID `uri:"kid" binding:"entityid"` | 		KeyID  models.KeyTokenID `uri:"kid" binding:"entityid"` | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	var u uri | 	var u uri | ||||||
| 	ctx, errResp := h.app.StartRequest(g, &u, nil, nil, nil) | 	ctx, g, errResp := pctx.URI(&u).Start() | ||||||
| 	if errResp != nil { | 	if errResp != nil { | ||||||
| 		return *errResp | 		return *errResp | ||||||
| 	} | 	} | ||||||
| 	defer ctx.Cancel() | 	defer ctx.Cancel() | ||||||
|  |  | ||||||
|  | 	return h.app.DoRequest(ctx, g, models.TLockRead, func(ctx *logic.AppContext, finishSuccess func(r ginext.HTTPResponse) ginext.HTTPResponse) ginext.HTTPResponse { | ||||||
|  |  | ||||||
| 		if permResp := ctx.CheckPermissionUserAdmin(u.UserID); permResp != nil { | 		if permResp := ctx.CheckPermissionUserAdmin(u.UserID); permResp != nil { | ||||||
| 			return *permResp | 			return *permResp | ||||||
| 		} | 		} | ||||||
| @@ -147,7 +156,9 @@ func (h APIHandler) GetUserKey(g *gin.Context) ginresp.HTTPResponse { | |||||||
| 			return ginresp.APIError(g, 500, apierr.DATABASE_ERROR, "Failed to query client", err) | 			return ginresp.APIError(g, 500, apierr.DATABASE_ERROR, "Failed to query client", err) | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 	return ctx.FinishSuccess(ginresp.JSON(http.StatusOK, keytoken.JSON())) | 		return finishSuccess(ginext.JSON(http.StatusOK, keytoken)) | ||||||
|  |  | ||||||
|  | 	}) | ||||||
| } | } | ||||||
|  |  | ||||||
| // UpdateUserKey swaggerdoc | // UpdateUserKey swaggerdoc | ||||||
| @@ -161,14 +172,14 @@ func (h APIHandler) GetUserKey(g *gin.Context) ginresp.HTTPResponse { | |||||||
| // | // | ||||||
| //	@Param		post_body	body		handler.UpdateUserKey.body	false	" " | //	@Param		post_body	body		handler.UpdateUserKey.body	false	" " | ||||||
| // | // | ||||||
| //	@Success	200			{object}	models.KeyTokenJSON | //	@Success	200			{object}	models.KeyToken | ||||||
| //	@Failure	400			{object}	ginresp.apiError	"supplied values/parameters cannot be parsed / are invalid" | //	@Failure	400			{object}	ginresp.apiError	"supplied values/parameters cannot be parsed / are invalid" | ||||||
| //	@Failure	401			{object}	ginresp.apiError	"user is not authorized / has missing permissions" | //	@Failure	401			{object}	ginresp.apiError	"user is not authorized / has missing permissions" | ||||||
| //	@Failure	404			{object}	ginresp.apiError	"message not found" | //	@Failure	404			{object}	ginresp.apiError	"message not found" | ||||||
| //	@Failure	500			{object}	ginresp.apiError	"internal server error" | //	@Failure	500			{object}	ginresp.apiError	"internal server error" | ||||||
| // | // | ||||||
| //	@Router		/api/v2/users/{uid}/keys/{kid} [PATCH] | //	@Router		/api/v2/users/{uid}/keys/{kid} [PATCH] | ||||||
| func (h APIHandler) UpdateUserKey(g *gin.Context) ginresp.HTTPResponse { | func (h APIHandler) UpdateUserKey(pctx ginext.PreContext) ginext.HTTPResponse { | ||||||
| 	type uri struct { | 	type uri struct { | ||||||
| 		UserID models.UserID     `uri:"uid" binding:"entityid"` | 		UserID models.UserID     `uri:"uid" binding:"entityid"` | ||||||
| 		KeyID  models.KeyTokenID `uri:"kid" binding:"entityid"` | 		KeyID  models.KeyTokenID `uri:"kid" binding:"entityid"` | ||||||
| @@ -182,12 +193,14 @@ func (h APIHandler) UpdateUserKey(g *gin.Context) ginresp.HTTPResponse { | |||||||
|  |  | ||||||
| 	var u uri | 	var u uri | ||||||
| 	var b body | 	var b body | ||||||
| 	ctx, errResp := h.app.StartRequest(g, &u, nil, &b, nil) | 	ctx, g, errResp := pctx.URI(&u).Body(&b).Start() | ||||||
| 	if errResp != nil { | 	if errResp != nil { | ||||||
| 		return *errResp | 		return *errResp | ||||||
| 	} | 	} | ||||||
| 	defer ctx.Cancel() | 	defer ctx.Cancel() | ||||||
|  |  | ||||||
|  | 	return h.app.DoRequest(ctx, g, models.TLockReadWrite, func(ctx *logic.AppContext, finishSuccess func(r ginext.HTTPResponse) ginext.HTTPResponse) ginext.HTTPResponse { | ||||||
|  |  | ||||||
| 		if permResp := ctx.CheckPermissionUserAdmin(u.UserID); permResp != nil { | 		if permResp := ctx.CheckPermissionUserAdmin(u.UserID); permResp != nil { | ||||||
| 			return *permResp | 			return *permResp | ||||||
| 		} | 		} | ||||||
| @@ -245,7 +258,9 @@ func (h APIHandler) UpdateUserKey(g *gin.Context) ginresp.HTTPResponse { | |||||||
| 			keytoken.Channels = *b.Channels | 			keytoken.Channels = *b.Channels | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 	return ctx.FinishSuccess(ginresp.JSON(http.StatusOK, keytoken.JSON())) | 		return finishSuccess(ginext.JSON(http.StatusOK, keytoken)) | ||||||
|  |  | ||||||
|  | 	}) | ||||||
| } | } | ||||||
|  |  | ||||||
| // CreateUserKey swaggerdoc | // CreateUserKey swaggerdoc | ||||||
| @@ -258,14 +273,14 @@ func (h APIHandler) UpdateUserKey(g *gin.Context) ginresp.HTTPResponse { | |||||||
| // | // | ||||||
| //	@Param		post_body	body		handler.CreateUserKey.body	false	" " | //	@Param		post_body	body		handler.CreateUserKey.body	false	" " | ||||||
| // | // | ||||||
| //	@Success	200			{object}	models.KeyTokenJSON | //	@Success	200			{object}	models.KeyToken | ||||||
| //	@Failure	400			{object}	ginresp.apiError	"supplied values/parameters cannot be parsed / are invalid" | //	@Failure	400			{object}	ginresp.apiError	"supplied values/parameters cannot be parsed / are invalid" | ||||||
| //	@Failure	401			{object}	ginresp.apiError	"user is not authorized / has missing permissions" | //	@Failure	401			{object}	ginresp.apiError	"user is not authorized / has missing permissions" | ||||||
| //	@Failure	404			{object}	ginresp.apiError	"message not found" | //	@Failure	404			{object}	ginresp.apiError	"message not found" | ||||||
| //	@Failure	500			{object}	ginresp.apiError	"internal server error" | //	@Failure	500			{object}	ginresp.apiError	"internal server error" | ||||||
| // | // | ||||||
| //	@Router		/api/v2/users/{uid}/keys [POST] | //	@Router		/api/v2/users/{uid}/keys [POST] | ||||||
| func (h APIHandler) CreateUserKey(g *gin.Context) ginresp.HTTPResponse { | func (h APIHandler) CreateUserKey(pctx ginext.PreContext) ginext.HTTPResponse { | ||||||
| 	type uri struct { | 	type uri struct { | ||||||
| 		UserID models.UserID `uri:"uid" binding:"entityid"` | 		UserID models.UserID `uri:"uid" binding:"entityid"` | ||||||
| 	} | 	} | ||||||
| @@ -278,12 +293,14 @@ func (h APIHandler) CreateUserKey(g *gin.Context) ginresp.HTTPResponse { | |||||||
|  |  | ||||||
| 	var u uri | 	var u uri | ||||||
| 	var b body | 	var b body | ||||||
| 	ctx, errResp := h.app.StartRequest(g, &u, nil, &b, nil) | 	ctx, g, errResp := pctx.URI(&u).Body(&b).Start() | ||||||
| 	if errResp != nil { | 	if errResp != nil { | ||||||
| 		return *errResp | 		return *errResp | ||||||
| 	} | 	} | ||||||
| 	defer ctx.Cancel() | 	defer ctx.Cancel() | ||||||
|  |  | ||||||
|  | 	return h.app.DoRequest(ctx, g, models.TLockReadWrite, func(ctx *logic.AppContext, finishSuccess func(r ginext.HTTPResponse) ginext.HTTPResponse) ginext.HTTPResponse { | ||||||
|  |  | ||||||
| 		channels := langext.Coalesce(b.Channels, make([]models.ChannelID, 0)) | 		channels := langext.Coalesce(b.Channels, make([]models.ChannelID, 0)) | ||||||
|  |  | ||||||
| 		var allChan bool | 		var allChan bool | ||||||
| @@ -314,7 +331,9 @@ func (h APIHandler) CreateUserKey(g *gin.Context) ginresp.HTTPResponse { | |||||||
| 			return ginresp.APIError(g, 500, apierr.DATABASE_ERROR, "Failed to create keytoken in db", err) | 			return ginresp.APIError(g, 500, apierr.DATABASE_ERROR, "Failed to create keytoken in db", err) | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 	return ctx.FinishSuccess(ginresp.JSON(http.StatusOK, keytok.JSON().WithToken(token))) | 		return finishSuccess(ginext.JSONWithFilter(http.StatusOK, keytok, "INCLUDE_TOKEN")) | ||||||
|  |  | ||||||
|  | 	}) | ||||||
| } | } | ||||||
|  |  | ||||||
| // DeleteUserKey swaggerdoc | // DeleteUserKey swaggerdoc | ||||||
| @@ -327,26 +346,28 @@ func (h APIHandler) CreateUserKey(g *gin.Context) ginresp.HTTPResponse { | |||||||
| //	@Param			uid	path		string	true	"UserID" | //	@Param			uid	path		string	true	"UserID" | ||||||
| //	@Param			kid	path		string	true	"TokenKeyID" | //	@Param			kid	path		string	true	"TokenKeyID" | ||||||
| // | // | ||||||
| //	@Success		200	{object}	models.KeyTokenJSON | //	@Success		200	{object}	models.KeyToken | ||||||
| //	@Failure		400	{object}	ginresp.apiError	"supplied values/parameters cannot be parsed / are invalid" | //	@Failure		400	{object}	ginresp.apiError	"supplied values/parameters cannot be parsed / are invalid" | ||||||
| //	@Failure		401	{object}	ginresp.apiError	"user is not authorized / has missing permissions" | //	@Failure		401	{object}	ginresp.apiError	"user is not authorized / has missing permissions" | ||||||
| //	@Failure		404	{object}	ginresp.apiError	"message not found" | //	@Failure		404	{object}	ginresp.apiError	"message not found" | ||||||
| //	@Failure		500	{object}	ginresp.apiError	"internal server error" | //	@Failure		500	{object}	ginresp.apiError	"internal server error" | ||||||
| // | // | ||||||
| //	@Router			/api/v2/users/{uid}/keys/{kid} [DELETE] | //	@Router			/api/v2/users/{uid}/keys/{kid} [DELETE] | ||||||
| func (h APIHandler) DeleteUserKey(g *gin.Context) ginresp.HTTPResponse { | func (h APIHandler) DeleteUserKey(pctx ginext.PreContext) ginext.HTTPResponse { | ||||||
| 	type uri struct { | 	type uri struct { | ||||||
| 		UserID models.UserID     `uri:"uid" binding:"entityid"` | 		UserID models.UserID     `uri:"uid" binding:"entityid"` | ||||||
| 		KeyID  models.KeyTokenID `uri:"kid" binding:"entityid"` | 		KeyID  models.KeyTokenID `uri:"kid" binding:"entityid"` | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	var u uri | 	var u uri | ||||||
| 	ctx, errResp := h.app.StartRequest(g, &u, nil, nil, nil) | 	ctx, g, errResp := pctx.URI(&u).Start() | ||||||
| 	if errResp != nil { | 	if errResp != nil { | ||||||
| 		return *errResp | 		return *errResp | ||||||
| 	} | 	} | ||||||
| 	defer ctx.Cancel() | 	defer ctx.Cancel() | ||||||
|  |  | ||||||
|  | 	return h.app.DoRequest(ctx, g, models.TLockReadWrite, func(ctx *logic.AppContext, finishSuccess func(r ginext.HTTPResponse) ginext.HTTPResponse) ginext.HTTPResponse { | ||||||
|  |  | ||||||
| 		if permResp := ctx.CheckPermissionUserAdmin(u.UserID); permResp != nil { | 		if permResp := ctx.CheckPermissionUserAdmin(u.UserID); permResp != nil { | ||||||
| 			return *permResp | 			return *permResp | ||||||
| 		} | 		} | ||||||
| @@ -368,5 +389,7 @@ func (h APIHandler) DeleteUserKey(g *gin.Context) ginresp.HTTPResponse { | |||||||
| 			return ginresp.APIError(g, 500, apierr.DATABASE_ERROR, "Failed to delete client", err) | 			return ginresp.APIError(g, 500, apierr.DATABASE_ERROR, "Failed to delete client", err) | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 	return ctx.FinishSuccess(ginresp.JSON(http.StatusOK, client.JSON())) | 		return finishSuccess(ginext.JSON(http.StatusOK, client)) | ||||||
|  |  | ||||||
|  | 	}) | ||||||
| } | } | ||||||
|   | |||||||
| @@ -1,8 +1,10 @@ | |||||||
| package handler | package handler | ||||||
|  |  | ||||||
| import ( | import ( | ||||||
|  | 	"blackforestbytes.com/simplecloudnotifier/logic" | ||||||
| 	"database/sql" | 	"database/sql" | ||||||
| 	"errors" | 	"errors" | ||||||
|  | 	"gogs.mikescher.com/BlackForestBytes/goext/ginext" | ||||||
| 	"net/http" | 	"net/http" | ||||||
| 	"strings" | 	"strings" | ||||||
| 	"time" | 	"time" | ||||||
| @@ -11,7 +13,6 @@ import ( | |||||||
| 	"blackforestbytes.com/simplecloudnotifier/api/ginresp" | 	"blackforestbytes.com/simplecloudnotifier/api/ginresp" | ||||||
| 	ct "blackforestbytes.com/simplecloudnotifier/db/cursortoken" | 	ct "blackforestbytes.com/simplecloudnotifier/db/cursortoken" | ||||||
| 	"blackforestbytes.com/simplecloudnotifier/models" | 	"blackforestbytes.com/simplecloudnotifier/models" | ||||||
| 	"github.com/gin-gonic/gin" |  | ||||||
| 	"gogs.mikescher.com/BlackForestBytes/goext/langext" | 	"gogs.mikescher.com/BlackForestBytes/goext/langext" | ||||||
| 	"gogs.mikescher.com/BlackForestBytes/goext/mathext" | 	"gogs.mikescher.com/BlackForestBytes/goext/mathext" | ||||||
| ) | ) | ||||||
| @@ -34,7 +35,7 @@ import ( | |||||||
| //	@Failure		500			{object}	ginresp.apiError	"internal server error" | //	@Failure		500			{object}	ginresp.apiError	"internal server error" | ||||||
| // | // | ||||||
| //	@Router			/api/v2/messages [GET] | //	@Router			/api/v2/messages [GET] | ||||||
| func (h APIHandler) ListMessages(g *gin.Context) ginresp.HTTPResponse { | func (h APIHandler) ListMessages(pctx ginext.PreContext) ginext.HTTPResponse { | ||||||
| 	type query struct { | 	type query struct { | ||||||
| 		PageSize      *int     `json:"page_size"       form:"page_size"` | 		PageSize      *int     `json:"page_size"       form:"page_size"` | ||||||
| 		NextPageToken *string  `json:"next_page_token" form:"next_page_token"` | 		NextPageToken *string  `json:"next_page_token" form:"next_page_token"` | ||||||
| @@ -49,18 +50,20 @@ func (h APIHandler) ListMessages(g *gin.Context) ginresp.HTTPResponse { | |||||||
| 		KeyTokens     []string `json:"used_key"        form:"used_key"` | 		KeyTokens     []string `json:"used_key"        form:"used_key"` | ||||||
| 	} | 	} | ||||||
| 	type response struct { | 	type response struct { | ||||||
| 		Messages      []models.MessageJSON `json:"messages"` | 		Messages      []models.Message `json:"messages"` | ||||||
| 		NextPageToken string           `json:"next_page_token"` | 		NextPageToken string           `json:"next_page_token"` | ||||||
| 		PageSize      int              `json:"page_size"` | 		PageSize      int              `json:"page_size"` | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	var q query | 	var q query | ||||||
| 	ctx, errResp := h.app.StartRequest(g, nil, &q, nil, nil) | 	ctx, g, errResp := pctx.Query(&q).Start() | ||||||
| 	if errResp != nil { | 	if errResp != nil { | ||||||
| 		return *errResp | 		return *errResp | ||||||
| 	} | 	} | ||||||
| 	defer ctx.Cancel() | 	defer ctx.Cancel() | ||||||
|  |  | ||||||
|  | 	return h.app.DoRequest(ctx, g, models.TLockReadWrite, func(ctx *logic.AppContext, finishSuccess func(r ginext.HTTPResponse) ginext.HTTPResponse) ginext.HTTPResponse { | ||||||
|  |  | ||||||
| 		trimmed := langext.Coalesce(q.Trimmed, true) | 		trimmed := langext.Coalesce(q.Trimmed, true) | ||||||
|  |  | ||||||
| 		maxPageSize := langext.Conditional(trimmed, 16, 256) | 		maxPageSize := langext.Conditional(trimmed, 16, 256) | ||||||
| @@ -148,14 +151,14 @@ func (h APIHandler) ListMessages(g *gin.Context) ginresp.HTTPResponse { | |||||||
| 			return ginresp.APIError(g, 500, apierr.DATABASE_ERROR, "Failed to query messages", err) | 			return ginresp.APIError(g, 500, apierr.DATABASE_ERROR, "Failed to query messages", err) | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 	var res []models.MessageJSON |  | ||||||
| 		if trimmed { | 		if trimmed { | ||||||
| 		res = langext.ArrMap(messages, func(v models.Message) models.MessageJSON { return v.TrimmedJSON() }) | 			res := langext.ArrMap(messages, func(v models.Message) models.Message { return v.PreMarshal().Trim() }) | ||||||
|  | 			return finishSuccess(ginext.JSON(http.StatusOK, response{Messages: res, NextPageToken: npt.Token(), PageSize: pageSize})) | ||||||
| 		} else { | 		} else { | ||||||
| 		res = langext.ArrMap(messages, func(v models.Message) models.MessageJSON { return v.FullJSON() }) | 			res := langext.ArrMap(messages, func(v models.Message) models.Message { return v.PreMarshal() }) | ||||||
|  | 			return finishSuccess(ginext.JSON(http.StatusOK, response{Messages: res, NextPageToken: npt.Token(), PageSize: pageSize})) | ||||||
| 		} | 		} | ||||||
|  | 	}) | ||||||
| 	return ctx.FinishSuccess(ginresp.JSON(http.StatusOK, response{Messages: res, NextPageToken: npt.Token(), PageSize: pageSize})) |  | ||||||
| } | } | ||||||
|  |  | ||||||
| // GetMessage swaggerdoc | // GetMessage swaggerdoc | ||||||
| @@ -169,25 +172,27 @@ func (h APIHandler) ListMessages(g *gin.Context) ginresp.HTTPResponse { | |||||||
| // | // | ||||||
| //	@Param			mid	path		string	true	"MessageID" | //	@Param			mid	path		string	true	"MessageID" | ||||||
| // | // | ||||||
| //	@Success		200	{object}	models.MessageJSON | //	@Success		200	{object}	models.Message | ||||||
| //	@Failure		400	{object}	ginresp.apiError	"supplied values/parameters cannot be parsed / are invalid" | //	@Failure		400	{object}	ginresp.apiError	"supplied values/parameters cannot be parsed / are invalid" | ||||||
| //	@Failure		401	{object}	ginresp.apiError	"user is not authorized / has missing permissions" | //	@Failure		401	{object}	ginresp.apiError	"user is not authorized / has missing permissions" | ||||||
| //	@Failure		404	{object}	ginresp.apiError	"message not found" | //	@Failure		404	{object}	ginresp.apiError	"message not found" | ||||||
| //	@Failure		500	{object}	ginresp.apiError	"internal server error" | //	@Failure		500	{object}	ginresp.apiError	"internal server error" | ||||||
| // | // | ||||||
| //	@Router			/api/v2/messages/{mid} [GET] | //	@Router			/api/v2/messages/{mid} [GET] | ||||||
| func (h APIHandler) GetMessage(g *gin.Context) ginresp.HTTPResponse { | func (h APIHandler) GetMessage(pctx ginext.PreContext) ginext.HTTPResponse { | ||||||
| 	type uri struct { | 	type uri struct { | ||||||
| 		MessageID models.MessageID `uri:"mid" binding:"entityid"` | 		MessageID models.MessageID `uri:"mid" binding:"entityid"` | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	var u uri | 	var u uri | ||||||
| 	ctx, errResp := h.app.StartRequest(g, &u, nil, nil, nil) | 	ctx, g, errResp := pctx.URI(&u).Start() | ||||||
| 	if errResp != nil { | 	if errResp != nil { | ||||||
| 		return *errResp | 		return *errResp | ||||||
| 	} | 	} | ||||||
| 	defer ctx.Cancel() | 	defer ctx.Cancel() | ||||||
|  |  | ||||||
|  | 	return h.app.DoRequest(ctx, g, models.TLockRead, func(ctx *logic.AppContext, finishSuccess func(r ginext.HTTPResponse) ginext.HTTPResponse) ginext.HTTPResponse { | ||||||
|  |  | ||||||
| 		if permResp := ctx.CheckPermissionAny(); permResp != nil { | 		if permResp := ctx.CheckPermissionAny(); permResp != nil { | ||||||
| 			return *permResp | 			return *permResp | ||||||
| 		} | 		} | ||||||
| @@ -204,7 +209,7 @@ func (h APIHandler) GetMessage(g *gin.Context) ginresp.HTTPResponse { | |||||||
| 		// or we subscribe (+confirmed) to the channel and have read/admin key | 		// or we subscribe (+confirmed) to the channel and have read/admin key | ||||||
|  |  | ||||||
| 		if ctx.CheckPermissionMessageRead(msg) { | 		if ctx.CheckPermissionMessageRead(msg) { | ||||||
| 		return ctx.FinishSuccess(ginresp.JSON(http.StatusOK, msg.FullJSON())) | 			return finishSuccess(ginext.JSON(http.StatusOK, msg.PreMarshal())) | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		if uid := ctx.GetPermissionUserID(); uid != nil && ctx.CheckPermissionUserRead(*uid) == nil { | 		if uid := ctx.GetPermissionUserID(); uid != nil && ctx.CheckPermissionUserRead(*uid) == nil { | ||||||
| @@ -222,10 +227,12 @@ func (h APIHandler) GetMessage(g *gin.Context) ginresp.HTTPResponse { | |||||||
| 			} | 			} | ||||||
|  |  | ||||||
| 			// => perm okay | 			// => perm okay | ||||||
| 		return ctx.FinishSuccess(ginresp.JSON(http.StatusOK, msg.FullJSON())) | 			return finishSuccess(ginext.JSON(http.StatusOK, msg.PreMarshal())) | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		return ginresp.APIError(g, 401, apierr.USER_AUTH_FAILED, "You are not authorized for this action", nil) | 		return ginresp.APIError(g, 401, apierr.USER_AUTH_FAILED, "You are not authorized for this action", nil) | ||||||
|  |  | ||||||
|  | 	}) | ||||||
| } | } | ||||||
|  |  | ||||||
| // DeleteMessage swaggerdoc | // DeleteMessage swaggerdoc | ||||||
| @@ -237,25 +244,27 @@ func (h APIHandler) GetMessage(g *gin.Context) ginresp.HTTPResponse { | |||||||
| // | // | ||||||
| //	@Param			mid	path		string	true	"MessageID" | //	@Param			mid	path		string	true	"MessageID" | ||||||
| // | // | ||||||
| //	@Success		200	{object}	models.MessageJSON | //	@Success		200	{object}	models.Message | ||||||
| //	@Failure		400	{object}	ginresp.apiError	"supplied values/parameters cannot be parsed / are invalid" | //	@Failure		400	{object}	ginresp.apiError	"supplied values/parameters cannot be parsed / are invalid" | ||||||
| //	@Failure		401	{object}	ginresp.apiError	"user is not authorized / has missing permissions" | //	@Failure		401	{object}	ginresp.apiError	"user is not authorized / has missing permissions" | ||||||
| //	@Failure		404	{object}	ginresp.apiError	"message not found" | //	@Failure		404	{object}	ginresp.apiError	"message not found" | ||||||
| //	@Failure		500	{object}	ginresp.apiError	"internal server error" | //	@Failure		500	{object}	ginresp.apiError	"internal server error" | ||||||
| // | // | ||||||
| //	@Router			/api/v2/messages/{mid} [DELETE] | //	@Router			/api/v2/messages/{mid} [DELETE] | ||||||
| func (h APIHandler) DeleteMessage(g *gin.Context) ginresp.HTTPResponse { | func (h APIHandler) DeleteMessage(pctx ginext.PreContext) ginext.HTTPResponse { | ||||||
| 	type uri struct { | 	type uri struct { | ||||||
| 		MessageID models.MessageID `uri:"mid" binding:"entityid"` | 		MessageID models.MessageID `uri:"mid" binding:"entityid"` | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	var u uri | 	var u uri | ||||||
| 	ctx, errResp := h.app.StartRequest(g, &u, nil, nil, nil) | 	ctx, g, errResp := pctx.URI(&u).Start() | ||||||
| 	if errResp != nil { | 	if errResp != nil { | ||||||
| 		return *errResp | 		return *errResp | ||||||
| 	} | 	} | ||||||
| 	defer ctx.Cancel() | 	defer ctx.Cancel() | ||||||
|  |  | ||||||
|  | 	return h.app.DoRequest(ctx, g, models.TLockReadWrite, func(ctx *logic.AppContext, finishSuccess func(r ginext.HTTPResponse) ginext.HTTPResponse) ginext.HTTPResponse { | ||||||
|  |  | ||||||
| 		if permResp := ctx.CheckPermissionAny(); permResp != nil { | 		if permResp := ctx.CheckPermissionAny(); permResp != nil { | ||||||
| 			return *permResp | 			return *permResp | ||||||
| 		} | 		} | ||||||
| @@ -282,5 +291,7 @@ func (h APIHandler) DeleteMessage(g *gin.Context) ginresp.HTTPResponse { | |||||||
| 			return ginresp.APIError(g, 500, apierr.DATABASE_ERROR, "Failed to cancel deliveries", err) | 			return ginresp.APIError(g, 500, apierr.DATABASE_ERROR, "Failed to cancel deliveries", err) | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 	return ctx.FinishSuccess(ginresp.JSON(http.StatusOK, msg.FullJSON())) | 		return finishSuccess(ginext.JSON(http.StatusOK, msg.PreMarshal())) | ||||||
|  |  | ||||||
|  | 	}) | ||||||
| } | } | ||||||
|   | |||||||
| @@ -3,10 +3,11 @@ package handler | |||||||
| import ( | import ( | ||||||
| 	"blackforestbytes.com/simplecloudnotifier/api/apierr" | 	"blackforestbytes.com/simplecloudnotifier/api/apierr" | ||||||
| 	"blackforestbytes.com/simplecloudnotifier/api/ginresp" | 	"blackforestbytes.com/simplecloudnotifier/api/ginresp" | ||||||
|  | 	"blackforestbytes.com/simplecloudnotifier/logic" | ||||||
| 	"blackforestbytes.com/simplecloudnotifier/models" | 	"blackforestbytes.com/simplecloudnotifier/models" | ||||||
| 	"database/sql" | 	"database/sql" | ||||||
| 	"errors" | 	"errors" | ||||||
| 	"github.com/gin-gonic/gin" | 	"gogs.mikescher.com/BlackForestBytes/goext/ginext" | ||||||
| 	"net/http" | 	"net/http" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| @@ -18,25 +19,27 @@ import ( | |||||||
| // | // | ||||||
| //	@Param		uid	path		string	true	"UserID" | //	@Param		uid	path		string	true	"UserID" | ||||||
| // | // | ||||||
| //	@Success	200	{object}	models.UserPreviewJSON | //	@Success	200	{object}	models.UserPreview | ||||||
| //	@Failure	400	{object}	ginresp.apiError	"supplied values/parameters cannot be parsed / are invalid" | //	@Failure	400	{object}	ginresp.apiError	"supplied values/parameters cannot be parsed / are invalid" | ||||||
| //	@Failure	401	{object}	ginresp.apiError	"user is not authorized / has missing permissions" | //	@Failure	401	{object}	ginresp.apiError	"user is not authorized / has missing permissions" | ||||||
| //	@Failure	404	{object}	ginresp.apiError	"user not found" | //	@Failure	404	{object}	ginresp.apiError	"user not found" | ||||||
| //	@Failure	500	{object}	ginresp.apiError	"internal server error" | //	@Failure	500	{object}	ginresp.apiError	"internal server error" | ||||||
| // | // | ||||||
| //	@Router		/api/v2/preview/users/{uid} [GET] | //	@Router		/api/v2/preview/users/{uid} [GET] | ||||||
| func (h APIHandler) GetUserPreview(g *gin.Context) ginresp.HTTPResponse { | func (h APIHandler) GetUserPreview(pctx ginext.PreContext) ginext.HTTPResponse { | ||||||
| 	type uri struct { | 	type uri struct { | ||||||
| 		UserID models.UserID `uri:"uid" binding:"entityid"` | 		UserID models.UserID `uri:"uid" binding:"entityid"` | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	var u uri | 	var u uri | ||||||
| 	ctx, errResp := h.app.StartRequest(g, &u, nil, nil, nil) | 	ctx, g, errResp := pctx.URI(&u).Start() | ||||||
| 	if errResp != nil { | 	if errResp != nil { | ||||||
| 		return *errResp | 		return *errResp | ||||||
| 	} | 	} | ||||||
| 	defer ctx.Cancel() | 	defer ctx.Cancel() | ||||||
|  |  | ||||||
|  | 	return h.app.DoRequest(ctx, g, models.TLockRead, func(ctx *logic.AppContext, finishSuccess func(r ginext.HTTPResponse) ginext.HTTPResponse) ginext.HTTPResponse { | ||||||
|  |  | ||||||
| 		if permResp := ctx.CheckPermissionAny(); permResp != nil { | 		if permResp := ctx.CheckPermissionAny(); permResp != nil { | ||||||
| 			return *permResp | 			return *permResp | ||||||
| 		} | 		} | ||||||
| @@ -49,7 +52,9 @@ func (h APIHandler) GetUserPreview(g *gin.Context) ginresp.HTTPResponse { | |||||||
| 			return ginresp.APIError(g, 500, apierr.DATABASE_ERROR, "Failed to query user", err) | 			return ginresp.APIError(g, 500, apierr.DATABASE_ERROR, "Failed to query user", err) | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 	return ctx.FinishSuccess(ginresp.JSON(http.StatusOK, user.JSONPreview())) | 		return finishSuccess(ginext.JSON(http.StatusOK, user.JSONPreview())) | ||||||
|  |  | ||||||
|  | 	}) | ||||||
| } | } | ||||||
|  |  | ||||||
| // GetChannelPreview swaggerdoc | // GetChannelPreview swaggerdoc | ||||||
| @@ -60,25 +65,27 @@ func (h APIHandler) GetUserPreview(g *gin.Context) ginresp.HTTPResponse { | |||||||
| // | // | ||||||
| //	@Param		cid	path		string	true	"ChannelID" | //	@Param		cid	path		string	true	"ChannelID" | ||||||
| // | // | ||||||
| //	@Success	200	{object}	models.ChannelPreviewJSON | //	@Success	200	{object}	models.ChannelPreview | ||||||
| //	@Failure	400	{object}	ginresp.apiError	"supplied values/parameters cannot be parsed / are invalid" | //	@Failure	400	{object}	ginresp.apiError	"supplied values/parameters cannot be parsed / are invalid" | ||||||
| //	@Failure	401	{object}	ginresp.apiError	"user is not authorized / has missing permissions" | //	@Failure	401	{object}	ginresp.apiError	"user is not authorized / has missing permissions" | ||||||
| //	@Failure	404	{object}	ginresp.apiError	"channel not found" | //	@Failure	404	{object}	ginresp.apiError	"channel not found" | ||||||
| //	@Failure	500	{object}	ginresp.apiError	"internal server error" | //	@Failure	500	{object}	ginresp.apiError	"internal server error" | ||||||
| // | // | ||||||
| //	@Router		/api/v2/preview/channels/{cid} [GET] | //	@Router		/api/v2/preview/channels/{cid} [GET] | ||||||
| func (h APIHandler) GetChannelPreview(g *gin.Context) ginresp.HTTPResponse { | func (h APIHandler) GetChannelPreview(pctx ginext.PreContext) ginext.HTTPResponse { | ||||||
| 	type uri struct { | 	type uri struct { | ||||||
| 		ChannelID models.ChannelID `uri:"cid" binding:"entityid"` | 		ChannelID models.ChannelID `uri:"cid" binding:"entityid"` | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	var u uri | 	var u uri | ||||||
| 	ctx, errResp := h.app.StartRequest(g, &u, nil, nil, nil) | 	ctx, g, errResp := pctx.URI(&u).Start() | ||||||
| 	if errResp != nil { | 	if errResp != nil { | ||||||
| 		return *errResp | 		return *errResp | ||||||
| 	} | 	} | ||||||
| 	defer ctx.Cancel() | 	defer ctx.Cancel() | ||||||
|  |  | ||||||
|  | 	return h.app.DoRequest(ctx, g, models.TLockRead, func(ctx *logic.AppContext, finishSuccess func(r ginext.HTTPResponse) ginext.HTTPResponse) ginext.HTTPResponse { | ||||||
|  |  | ||||||
| 		if permResp := ctx.CheckPermissionAny(); permResp != nil { | 		if permResp := ctx.CheckPermissionAny(); permResp != nil { | ||||||
| 			return *permResp | 			return *permResp | ||||||
| 		} | 		} | ||||||
| @@ -91,7 +98,9 @@ func (h APIHandler) GetChannelPreview(g *gin.Context) ginresp.HTTPResponse { | |||||||
| 			return ginresp.APIError(g, 500, apierr.DATABASE_ERROR, "Failed to query channel", err) | 			return ginresp.APIError(g, 500, apierr.DATABASE_ERROR, "Failed to query channel", err) | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 	return ctx.FinishSuccess(ginresp.JSON(http.StatusOK, channel.JSONPreview())) | 		return finishSuccess(ginext.JSON(http.StatusOK, channel.Preview())) | ||||||
|  |  | ||||||
|  | 	}) | ||||||
| } | } | ||||||
|  |  | ||||||
| // GetUserKeyPreview swaggerdoc | // GetUserKeyPreview swaggerdoc | ||||||
| @@ -102,25 +111,27 @@ func (h APIHandler) GetChannelPreview(g *gin.Context) ginresp.HTTPResponse { | |||||||
| // | // | ||||||
| //	@Param		kid	path		string	true	"TokenKeyID" | //	@Param		kid	path		string	true	"TokenKeyID" | ||||||
| // | // | ||||||
| //	@Success	200	{object}	models.KeyTokenPreviewJSON | //	@Success	200	{object}	models.KeyTokenPreview | ||||||
| //	@Failure	400	{object}	ginresp.apiError	"supplied values/parameters cannot be parsed / are invalid" | //	@Failure	400	{object}	ginresp.apiError	"supplied values/parameters cannot be parsed / are invalid" | ||||||
| //	@Failure	401	{object}	ginresp.apiError	"user is not authorized / has missing permissions" | //	@Failure	401	{object}	ginresp.apiError	"user is not authorized / has missing permissions" | ||||||
| //	@Failure	404	{object}	ginresp.apiError	"message not found" | //	@Failure	404	{object}	ginresp.apiError	"message not found" | ||||||
| //	@Failure	500	{object}	ginresp.apiError	"internal server error" | //	@Failure	500	{object}	ginresp.apiError	"internal server error" | ||||||
| // | // | ||||||
| //	@Router		/api/v2/preview/keys/{kid} [GET] | //	@Router		/api/v2/preview/keys/{kid} [GET] | ||||||
| func (h APIHandler) GetUserKeyPreview(g *gin.Context) ginresp.HTTPResponse { | func (h APIHandler) GetUserKeyPreview(pctx ginext.PreContext) ginext.HTTPResponse { | ||||||
| 	type uri struct { | 	type uri struct { | ||||||
| 		KeyID models.KeyTokenID `uri:"kid" binding:"entityid"` | 		KeyID models.KeyTokenID `uri:"kid" binding:"entityid"` | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	var u uri | 	var u uri | ||||||
| 	ctx, errResp := h.app.StartRequest(g, &u, nil, nil, nil) | 	ctx, g, errResp := pctx.URI(&u).Start() | ||||||
| 	if errResp != nil { | 	if errResp != nil { | ||||||
| 		return *errResp | 		return *errResp | ||||||
| 	} | 	} | ||||||
| 	defer ctx.Cancel() | 	defer ctx.Cancel() | ||||||
|  |  | ||||||
|  | 	return h.app.DoRequest(ctx, g, models.TLockRead, func(ctx *logic.AppContext, finishSuccess func(r ginext.HTTPResponse) ginext.HTTPResponse) ginext.HTTPResponse { | ||||||
|  |  | ||||||
| 		if permResp := ctx.CheckPermissionAny(); permResp != nil { | 		if permResp := ctx.CheckPermissionAny(); permResp != nil { | ||||||
| 			return *permResp | 			return *permResp | ||||||
| 		} | 		} | ||||||
| @@ -133,5 +144,7 @@ func (h APIHandler) GetUserKeyPreview(g *gin.Context) ginresp.HTTPResponse { | |||||||
| 			return ginresp.APIError(g, 500, apierr.DATABASE_ERROR, "Failed to query client", err) | 			return ginresp.APIError(g, 500, apierr.DATABASE_ERROR, "Failed to query client", err) | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 	return ctx.FinishSuccess(ginresp.JSON(http.StatusOK, keytoken.JSONPreview())) | 		return finishSuccess(ginext.JSON(http.StatusOK, keytoken.Preview())) | ||||||
|  |  | ||||||
|  | 	}) | ||||||
| } | } | ||||||
|   | |||||||
| @@ -3,10 +3,11 @@ package handler | |||||||
| import ( | import ( | ||||||
| 	"blackforestbytes.com/simplecloudnotifier/api/apierr" | 	"blackforestbytes.com/simplecloudnotifier/api/apierr" | ||||||
| 	"blackforestbytes.com/simplecloudnotifier/api/ginresp" | 	"blackforestbytes.com/simplecloudnotifier/api/ginresp" | ||||||
|  | 	"blackforestbytes.com/simplecloudnotifier/logic" | ||||||
| 	"blackforestbytes.com/simplecloudnotifier/models" | 	"blackforestbytes.com/simplecloudnotifier/models" | ||||||
| 	"database/sql" | 	"database/sql" | ||||||
| 	"errors" | 	"errors" | ||||||
| 	"github.com/gin-gonic/gin" | 	"gogs.mikescher.com/BlackForestBytes/goext/ginext" | ||||||
| 	"gogs.mikescher.com/BlackForestBytes/goext/langext" | 	"gogs.mikescher.com/BlackForestBytes/goext/langext" | ||||||
| 	"net/http" | 	"net/http" | ||||||
| 	"strings" | 	"strings" | ||||||
| @@ -47,7 +48,7 @@ import ( | |||||||
| //	@Failure		500			{object}	ginresp.apiError	"internal server error" | //	@Failure		500			{object}	ginresp.apiError	"internal server error" | ||||||
| // | // | ||||||
| //	@Router			/api/v2/users/{uid}/subscriptions [GET] | //	@Router			/api/v2/users/{uid}/subscriptions [GET] | ||||||
| func (h APIHandler) ListUserSubscriptions(g *gin.Context) ginresp.HTTPResponse { | func (h APIHandler) ListUserSubscriptions(pctx ginext.PreContext) ginext.HTTPResponse { | ||||||
| 	type uri struct { | 	type uri struct { | ||||||
| 		UserID models.UserID `uri:"uid" binding:"entityid"` | 		UserID models.UserID `uri:"uid" binding:"entityid"` | ||||||
| 	} | 	} | ||||||
| @@ -59,17 +60,19 @@ func (h APIHandler) ListUserSubscriptions(g *gin.Context) ginresp.HTTPResponse { | |||||||
| 		ChannelOwnerUserID *models.UserID `json:"channel_owner_user_id" form:"channel_owner_user_id"` | 		ChannelOwnerUserID *models.UserID `json:"channel_owner_user_id" form:"channel_owner_user_id"` | ||||||
| 	} | 	} | ||||||
| 	type response struct { | 	type response struct { | ||||||
| 		Subscriptions []models.SubscriptionJSON `json:"subscriptions"` | 		Subscriptions []models.Subscription `json:"subscriptions"` | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	var u uri | 	var u uri | ||||||
| 	var q query | 	var q query | ||||||
| 	ctx, errResp := h.app.StartRequest(g, &u, &q, nil, nil) | 	ctx, g, errResp := pctx.URI(&u).Query(&q).Start() | ||||||
| 	if errResp != nil { | 	if errResp != nil { | ||||||
| 		return *errResp | 		return *errResp | ||||||
| 	} | 	} | ||||||
| 	defer ctx.Cancel() | 	defer ctx.Cancel() | ||||||
|  |  | ||||||
|  | 	return h.app.DoRequest(ctx, g, models.TLockRead, func(ctx *logic.AppContext, finishSuccess func(r ginext.HTTPResponse) ginext.HTTPResponse) ginext.HTTPResponse { | ||||||
|  |  | ||||||
| 		if permResp := ctx.CheckPermissionUserRead(u.UserID); permResp != nil { | 		if permResp := ctx.CheckPermissionUserRead(u.UserID); permResp != nil { | ||||||
| 			return *permResp | 			return *permResp | ||||||
| 		} | 		} | ||||||
| @@ -126,9 +129,9 @@ func (h APIHandler) ListUserSubscriptions(g *gin.Context) ginresp.HTTPResponse { | |||||||
| 			return ginresp.APIError(g, 500, apierr.DATABASE_ERROR, "Failed to query subscriptions", err) | 			return ginresp.APIError(g, 500, apierr.DATABASE_ERROR, "Failed to query subscriptions", err) | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 	jsonres := langext.ArrMap(res, func(v models.Subscription) models.SubscriptionJSON { return v.JSON() }) | 		return finishSuccess(ginext.JSON(http.StatusOK, response{Subscriptions: res})) | ||||||
|  |  | ||||||
| 	return ctx.FinishSuccess(ginresp.JSON(http.StatusOK, response{Subscriptions: jsonres})) | 	}) | ||||||
| } | } | ||||||
|  |  | ||||||
| // ListChannelSubscriptions swaggerdoc | // ListChannelSubscriptions swaggerdoc | ||||||
| @@ -147,22 +150,24 @@ func (h APIHandler) ListUserSubscriptions(g *gin.Context) ginresp.HTTPResponse { | |||||||
| //	@Failure	500	{object}	ginresp.apiError	"internal server error" | //	@Failure	500	{object}	ginresp.apiError	"internal server error" | ||||||
| // | // | ||||||
| //	@Router		/api/v2/users/{uid}/channels/{cid}/subscriptions [GET] | //	@Router		/api/v2/users/{uid}/channels/{cid}/subscriptions [GET] | ||||||
| func (h APIHandler) ListChannelSubscriptions(g *gin.Context) ginresp.HTTPResponse { | func (h APIHandler) ListChannelSubscriptions(pctx ginext.PreContext) ginext.HTTPResponse { | ||||||
| 	type uri struct { | 	type uri struct { | ||||||
| 		UserID    models.UserID    `uri:"uid" binding:"entityid"` | 		UserID    models.UserID    `uri:"uid" binding:"entityid"` | ||||||
| 		ChannelID models.ChannelID `uri:"cid" binding:"entityid"` | 		ChannelID models.ChannelID `uri:"cid" binding:"entityid"` | ||||||
| 	} | 	} | ||||||
| 	type response struct { | 	type response struct { | ||||||
| 		Subscriptions []models.SubscriptionJSON `json:"subscriptions"` | 		Subscriptions []models.Subscription `json:"subscriptions"` | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	var u uri | 	var u uri | ||||||
| 	ctx, errResp := h.app.StartRequest(g, &u, nil, nil, nil) | 	ctx, g, errResp := pctx.URI(&u).Start() | ||||||
| 	if errResp != nil { | 	if errResp != nil { | ||||||
| 		return *errResp | 		return *errResp | ||||||
| 	} | 	} | ||||||
| 	defer ctx.Cancel() | 	defer ctx.Cancel() | ||||||
|  |  | ||||||
|  | 	return h.app.DoRequest(ctx, g, models.TLockRead, func(ctx *logic.AppContext, finishSuccess func(r ginext.HTTPResponse) ginext.HTTPResponse) ginext.HTTPResponse { | ||||||
|  |  | ||||||
| 		if permResp := ctx.CheckPermissionUserRead(u.UserID); permResp != nil { | 		if permResp := ctx.CheckPermissionUserRead(u.UserID); permResp != nil { | ||||||
| 			return *permResp | 			return *permResp | ||||||
| 		} | 		} | ||||||
| @@ -175,14 +180,14 @@ func (h APIHandler) ListChannelSubscriptions(g *gin.Context) ginresp.HTTPRespons | |||||||
| 			return ginresp.APIError(g, 500, apierr.DATABASE_ERROR, "Failed to query channel", err) | 			return ginresp.APIError(g, 500, apierr.DATABASE_ERROR, "Failed to query channel", err) | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 	clients, err := h.database.ListSubscriptions(ctx, models.SubscriptionFilter{AnyUserID: langext.Ptr(u.UserID), ChannelID: langext.Ptr([]models.ChannelID{u.ChannelID})}) | 		subs, err := h.database.ListSubscriptions(ctx, models.SubscriptionFilter{AnyUserID: langext.Ptr(u.UserID), ChannelID: langext.Ptr([]models.ChannelID{u.ChannelID})}) | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| 			return ginresp.APIError(g, 500, apierr.DATABASE_ERROR, "Failed to query subscriptions", err) | 			return ginresp.APIError(g, 500, apierr.DATABASE_ERROR, "Failed to query subscriptions", err) | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 	res := langext.ArrMap(clients, func(v models.Subscription) models.SubscriptionJSON { return v.JSON() }) | 		return finishSuccess(ginext.JSON(http.StatusOK, response{Subscriptions: subs})) | ||||||
|  |  | ||||||
| 	return ctx.FinishSuccess(ginresp.JSON(http.StatusOK, response{Subscriptions: res})) | 	}) | ||||||
| } | } | ||||||
|  |  | ||||||
| // GetSubscription swaggerdoc | // GetSubscription swaggerdoc | ||||||
| @@ -194,26 +199,28 @@ func (h APIHandler) ListChannelSubscriptions(g *gin.Context) ginresp.HTTPRespons | |||||||
| //	@Param		uid	path		string	true	"UserID" | //	@Param		uid	path		string	true	"UserID" | ||||||
| //	@Param		sid	path		string	true	"SubscriptionID" | //	@Param		sid	path		string	true	"SubscriptionID" | ||||||
| // | // | ||||||
| //	@Success	200	{object}	models.SubscriptionJSON | //	@Success	200	{object}	models.Subscription | ||||||
| //	@Failure	400	{object}	ginresp.apiError	"supplied values/parameters cannot be parsed / are invalid" | //	@Failure	400	{object}	ginresp.apiError	"supplied values/parameters cannot be parsed / are invalid" | ||||||
| //	@Failure	401	{object}	ginresp.apiError	"user is not authorized / has missing permissions" | //	@Failure	401	{object}	ginresp.apiError	"user is not authorized / has missing permissions" | ||||||
| //	@Failure	404	{object}	ginresp.apiError	"subscription not found" | //	@Failure	404	{object}	ginresp.apiError	"subscription not found" | ||||||
| //	@Failure	500	{object}	ginresp.apiError	"internal server error" | //	@Failure	500	{object}	ginresp.apiError	"internal server error" | ||||||
| // | // | ||||||
| //	@Router		/api/v2/users/{uid}/subscriptions/{sid} [GET] | //	@Router		/api/v2/users/{uid}/subscriptions/{sid} [GET] | ||||||
| func (h APIHandler) GetSubscription(g *gin.Context) ginresp.HTTPResponse { | func (h APIHandler) GetSubscription(pctx ginext.PreContext) ginext.HTTPResponse { | ||||||
| 	type uri struct { | 	type uri struct { | ||||||
| 		UserID         models.UserID         `uri:"uid" binding:"entityid"` | 		UserID         models.UserID         `uri:"uid" binding:"entityid"` | ||||||
| 		SubscriptionID models.SubscriptionID `uri:"sid" binding:"entityid"` | 		SubscriptionID models.SubscriptionID `uri:"sid" binding:"entityid"` | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	var u uri | 	var u uri | ||||||
| 	ctx, errResp := h.app.StartRequest(g, &u, nil, nil, nil) | 	ctx, g, errResp := pctx.URI(&u).Start() | ||||||
| 	if errResp != nil { | 	if errResp != nil { | ||||||
| 		return *errResp | 		return *errResp | ||||||
| 	} | 	} | ||||||
| 	defer ctx.Cancel() | 	defer ctx.Cancel() | ||||||
|  |  | ||||||
|  | 	return h.app.DoRequest(ctx, g, models.TLockRead, func(ctx *logic.AppContext, finishSuccess func(r ginext.HTTPResponse) ginext.HTTPResponse) ginext.HTTPResponse { | ||||||
|  |  | ||||||
| 		if permResp := ctx.CheckPermissionUserRead(u.UserID); permResp != nil { | 		if permResp := ctx.CheckPermissionUserRead(u.UserID); permResp != nil { | ||||||
| 			return *permResp | 			return *permResp | ||||||
| 		} | 		} | ||||||
| @@ -229,7 +236,9 @@ func (h APIHandler) GetSubscription(g *gin.Context) ginresp.HTTPResponse { | |||||||
| 			return ginresp.APIError(g, 404, apierr.SUBSCRIPTION_USER_MISMATCH, "Subscription not found", nil) | 			return ginresp.APIError(g, 404, apierr.SUBSCRIPTION_USER_MISMATCH, "Subscription not found", nil) | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 	return ctx.FinishSuccess(ginresp.JSON(http.StatusOK, subscription.JSON())) | 		return finishSuccess(ginext.JSON(http.StatusOK, subscription)) | ||||||
|  |  | ||||||
|  | 	}) | ||||||
| } | } | ||||||
|  |  | ||||||
| // CancelSubscription swaggerdoc | // CancelSubscription swaggerdoc | ||||||
| @@ -241,26 +250,28 @@ func (h APIHandler) GetSubscription(g *gin.Context) ginresp.HTTPResponse { | |||||||
| //	@Param		uid	path		string	true	"UserID" | //	@Param		uid	path		string	true	"UserID" | ||||||
| //	@Param		sid	path		string	true	"SubscriptionID" | //	@Param		sid	path		string	true	"SubscriptionID" | ||||||
| // | // | ||||||
| //	@Success	200	{object}	models.SubscriptionJSON | //	@Success	200	{object}	models.Subscription | ||||||
| //	@Failure	400	{object}	ginresp.apiError	"supplied values/parameters cannot be parsed / are invalid" | //	@Failure	400	{object}	ginresp.apiError	"supplied values/parameters cannot be parsed / are invalid" | ||||||
| //	@Failure	401	{object}	ginresp.apiError	"user is not authorized / has missing permissions" | //	@Failure	401	{object}	ginresp.apiError	"user is not authorized / has missing permissions" | ||||||
| //	@Failure	404	{object}	ginresp.apiError	"subscription not found" | //	@Failure	404	{object}	ginresp.apiError	"subscription not found" | ||||||
| //	@Failure	500	{object}	ginresp.apiError	"internal server error" | //	@Failure	500	{object}	ginresp.apiError	"internal server error" | ||||||
| // | // | ||||||
| //	@Router		/api/v2/users/{uid}/subscriptions/{sid} [DELETE] | //	@Router		/api/v2/users/{uid}/subscriptions/{sid} [DELETE] | ||||||
| func (h APIHandler) CancelSubscription(g *gin.Context) ginresp.HTTPResponse { | func (h APIHandler) CancelSubscription(pctx ginext.PreContext) ginext.HTTPResponse { | ||||||
| 	type uri struct { | 	type uri struct { | ||||||
| 		UserID         models.UserID         `uri:"uid" binding:"entityid"` | 		UserID         models.UserID         `uri:"uid" binding:"entityid"` | ||||||
| 		SubscriptionID models.SubscriptionID `uri:"sid" binding:"entityid"` | 		SubscriptionID models.SubscriptionID `uri:"sid" binding:"entityid"` | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	var u uri | 	var u uri | ||||||
| 	ctx, errResp := h.app.StartRequest(g, &u, nil, nil, nil) | 	ctx, g, errResp := pctx.URI(&u).Start() | ||||||
| 	if errResp != nil { | 	if errResp != nil { | ||||||
| 		return *errResp | 		return *errResp | ||||||
| 	} | 	} | ||||||
| 	defer ctx.Cancel() | 	defer ctx.Cancel() | ||||||
|  |  | ||||||
|  | 	return h.app.DoRequest(ctx, g, models.TLockReadWrite, func(ctx *logic.AppContext, finishSuccess func(r ginext.HTTPResponse) ginext.HTTPResponse) ginext.HTTPResponse { | ||||||
|  |  | ||||||
| 		if permResp := ctx.CheckPermissionUserAdmin(u.UserID); permResp != nil { | 		if permResp := ctx.CheckPermissionUserAdmin(u.UserID); permResp != nil { | ||||||
| 			return *permResp | 			return *permResp | ||||||
| 		} | 		} | ||||||
| @@ -281,7 +292,9 @@ func (h APIHandler) CancelSubscription(g *gin.Context) ginresp.HTTPResponse { | |||||||
| 			return ginresp.APIError(g, 500, apierr.DATABASE_ERROR, "Failed to delete subscription", err) | 			return ginresp.APIError(g, 500, apierr.DATABASE_ERROR, "Failed to delete subscription", err) | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 	return ctx.FinishSuccess(ginresp.JSON(http.StatusOK, subscription.JSON())) | 		return finishSuccess(ginext.JSON(http.StatusOK, subscription)) | ||||||
|  |  | ||||||
|  | 	}) | ||||||
| } | } | ||||||
|  |  | ||||||
| // CreateSubscription swaggerdoc | // CreateSubscription swaggerdoc | ||||||
| @@ -295,13 +308,13 @@ func (h APIHandler) CancelSubscription(g *gin.Context) ginresp.HTTPResponse { | |||||||
| //	@Param			query_data	query		handler.CreateSubscription.query	false	" " | //	@Param			query_data	query		handler.CreateSubscription.query	false	" " | ||||||
| //	@Param			post_data	body		handler.CreateSubscription.body		false	" " | //	@Param			post_data	body		handler.CreateSubscription.body		false	" " | ||||||
| // | // | ||||||
| //	@Success		200			{object}	models.SubscriptionJSON | //	@Success		200			{object}	models.Subscription | ||||||
| //	@Failure		400			{object}	ginresp.apiError	"supplied values/parameters cannot be parsed / are invalid" | //	@Failure		400			{object}	ginresp.apiError	"supplied values/parameters cannot be parsed / are invalid" | ||||||
| //	@Failure		401			{object}	ginresp.apiError	"user is not authorized / has missing permissions" | //	@Failure		401			{object}	ginresp.apiError	"user is not authorized / has missing permissions" | ||||||
| //	@Failure		500			{object}	ginresp.apiError	"internal server error" | //	@Failure		500			{object}	ginresp.apiError	"internal server error" | ||||||
| // | // | ||||||
| //	@Router			/api/v2/users/{uid}/subscriptions [POST] | //	@Router			/api/v2/users/{uid}/subscriptions [POST] | ||||||
| func (h APIHandler) CreateSubscription(g *gin.Context) ginresp.HTTPResponse { | func (h APIHandler) CreateSubscription(pctx ginext.PreContext) ginext.HTTPResponse { | ||||||
| 	type uri struct { | 	type uri struct { | ||||||
| 		UserID models.UserID `uri:"uid" binding:"entityid"` | 		UserID models.UserID `uri:"uid" binding:"entityid"` | ||||||
| 	} | 	} | ||||||
| @@ -317,12 +330,14 @@ func (h APIHandler) CreateSubscription(g *gin.Context) ginresp.HTTPResponse { | |||||||
| 	var u uri | 	var u uri | ||||||
| 	var q query | 	var q query | ||||||
| 	var b body | 	var b body | ||||||
| 	ctx, errResp := h.app.StartRequest(g, &u, &q, &b, nil) | 	ctx, g, errResp := pctx.URI(&u).Query(&q).Body(&b).Start() | ||||||
| 	if errResp != nil { | 	if errResp != nil { | ||||||
| 		return *errResp | 		return *errResp | ||||||
| 	} | 	} | ||||||
| 	defer ctx.Cancel() | 	defer ctx.Cancel() | ||||||
|  |  | ||||||
|  | 	return h.app.DoRequest(ctx, g, models.TLockReadWrite, func(ctx *logic.AppContext, finishSuccess func(r ginext.HTTPResponse) ginext.HTTPResponse) ginext.HTTPResponse { | ||||||
|  |  | ||||||
| 		if permResp := ctx.CheckPermissionUserAdmin(u.UserID); permResp != nil { | 		if permResp := ctx.CheckPermissionUserAdmin(u.UserID); permResp != nil { | ||||||
| 			return *permResp | 			return *permResp | ||||||
| 		} | 		} | ||||||
| @@ -378,7 +393,7 @@ func (h APIHandler) CreateSubscription(g *gin.Context) ginresp.HTTPResponse { | |||||||
| 				existingSub.Confirmed = true | 				existingSub.Confirmed = true | ||||||
| 			} | 			} | ||||||
|  |  | ||||||
| 		return ctx.FinishSuccess(ginresp.JSON(http.StatusOK, existingSub.JSON())) | 			return finishSuccess(ginext.JSON(http.StatusOK, existingSub)) | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		sub, err := h.database.CreateSubscription(ctx, u.UserID, channel, channel.OwnerUserID == u.UserID) | 		sub, err := h.database.CreateSubscription(ctx, u.UserID, channel, channel.OwnerUserID == u.UserID) | ||||||
| @@ -386,7 +401,9 @@ func (h APIHandler) CreateSubscription(g *gin.Context) ginresp.HTTPResponse { | |||||||
| 			return ginresp.APIError(g, 500, apierr.DATABASE_ERROR, "Failed to create subscription", err) | 			return ginresp.APIError(g, 500, apierr.DATABASE_ERROR, "Failed to create subscription", err) | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 	return ctx.FinishSuccess(ginresp.JSON(http.StatusOK, sub.JSON())) | 		return finishSuccess(ginext.JSON(http.StatusOK, sub)) | ||||||
|  |  | ||||||
|  | 	}) | ||||||
| } | } | ||||||
|  |  | ||||||
| // UpdateSubscription swaggerdoc | // UpdateSubscription swaggerdoc | ||||||
| @@ -399,14 +416,14 @@ func (h APIHandler) CreateSubscription(g *gin.Context) ginresp.HTTPResponse { | |||||||
| //	@Param		sid			path		string							true	"SubscriptionID" | //	@Param		sid			path		string							true	"SubscriptionID" | ||||||
| //	@Param		post_data	body		handler.UpdateSubscription.body	false	" " | //	@Param		post_data	body		handler.UpdateSubscription.body	false	" " | ||||||
| // | // | ||||||
| //	@Success	200			{object}	models.SubscriptionJSON | //	@Success	200			{object}	models.Subscription | ||||||
| //	@Failure	400			{object}	ginresp.apiError	"supplied values/parameters cannot be parsed / are invalid" | //	@Failure	400			{object}	ginresp.apiError	"supplied values/parameters cannot be parsed / are invalid" | ||||||
| //	@Failure	401			{object}	ginresp.apiError	"user is not authorized / has missing permissions" | //	@Failure	401			{object}	ginresp.apiError	"user is not authorized / has missing permissions" | ||||||
| //	@Failure	404			{object}	ginresp.apiError	"subscription not found" | //	@Failure	404			{object}	ginresp.apiError	"subscription not found" | ||||||
| //	@Failure	500			{object}	ginresp.apiError	"internal server error" | //	@Failure	500			{object}	ginresp.apiError	"internal server error" | ||||||
| // | // | ||||||
| //	@Router		/api/v2/users/{uid}/subscriptions/{sid} [PATCH] | //	@Router		/api/v2/users/{uid}/subscriptions/{sid} [PATCH] | ||||||
| func (h APIHandler) UpdateSubscription(g *gin.Context) ginresp.HTTPResponse { | func (h APIHandler) UpdateSubscription(pctx ginext.PreContext) ginext.HTTPResponse { | ||||||
| 	type uri struct { | 	type uri struct { | ||||||
| 		UserID         models.UserID         `uri:"uid" binding:"entityid"` | 		UserID         models.UserID         `uri:"uid" binding:"entityid"` | ||||||
| 		SubscriptionID models.SubscriptionID `uri:"sid" binding:"entityid"` | 		SubscriptionID models.SubscriptionID `uri:"sid" binding:"entityid"` | ||||||
| @@ -417,12 +434,14 @@ func (h APIHandler) UpdateSubscription(g *gin.Context) ginresp.HTTPResponse { | |||||||
|  |  | ||||||
| 	var u uri | 	var u uri | ||||||
| 	var b body | 	var b body | ||||||
| 	ctx, errResp := h.app.StartRequest(g, &u, nil, &b, nil) | 	ctx, g, errResp := pctx.URI(&u).Body(&b).Start() | ||||||
| 	if errResp != nil { | 	if errResp != nil { | ||||||
| 		return *errResp | 		return *errResp | ||||||
| 	} | 	} | ||||||
| 	defer ctx.Cancel() | 	defer ctx.Cancel() | ||||||
|  |  | ||||||
|  | 	return h.app.DoRequest(ctx, g, models.TLockReadWrite, func(ctx *logic.AppContext, finishSuccess func(r ginext.HTTPResponse) ginext.HTTPResponse) ginext.HTTPResponse { | ||||||
|  |  | ||||||
| 		if permResp := ctx.CheckPermissionUserAdmin(u.UserID); permResp != nil { | 		if permResp := ctx.CheckPermissionUserAdmin(u.UserID); permResp != nil { | ||||||
| 			return *permResp | 			return *permResp | ||||||
| 		} | 		} | ||||||
| @@ -455,5 +474,7 @@ func (h APIHandler) UpdateSubscription(g *gin.Context) ginresp.HTTPResponse { | |||||||
| 			return ginresp.APIError(g, 500, apierr.DATABASE_ERROR, "Failed to query subscription", err) | 			return ginresp.APIError(g, 500, apierr.DATABASE_ERROR, "Failed to query subscription", err) | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 	return ctx.FinishSuccess(ginresp.JSON(http.StatusOK, subscription.JSON())) | 		return finishSuccess(ginext.JSON(http.StatusOK, subscription)) | ||||||
|  |  | ||||||
|  | 	}) | ||||||
| } | } | ||||||
|   | |||||||
| @@ -3,12 +3,13 @@ package handler | |||||||
| import ( | import ( | ||||||
| 	"blackforestbytes.com/simplecloudnotifier/api/apierr" | 	"blackforestbytes.com/simplecloudnotifier/api/apierr" | ||||||
| 	"blackforestbytes.com/simplecloudnotifier/api/ginresp" | 	"blackforestbytes.com/simplecloudnotifier/api/ginresp" | ||||||
|  | 	"blackforestbytes.com/simplecloudnotifier/logic" | ||||||
| 	"blackforestbytes.com/simplecloudnotifier/models" | 	"blackforestbytes.com/simplecloudnotifier/models" | ||||||
| 	"database/sql" | 	"database/sql" | ||||||
| 	"errors" | 	"errors" | ||||||
| 	"fmt" | 	"fmt" | ||||||
| 	"github.com/gin-gonic/gin" |  | ||||||
| 	"github.com/rs/zerolog/log" | 	"github.com/rs/zerolog/log" | ||||||
|  | 	"gogs.mikescher.com/BlackForestBytes/goext/ginext" | ||||||
| 	"gogs.mikescher.com/BlackForestBytes/goext/langext" | 	"gogs.mikescher.com/BlackForestBytes/goext/langext" | ||||||
| 	"net/http" | 	"net/http" | ||||||
| ) | ) | ||||||
| @@ -21,12 +22,12 @@ import ( | |||||||
| // | // | ||||||
| //	@Param		post_body	body		handler.CreateUser.body	false	" " | //	@Param		post_body	body		handler.CreateUser.body	false	" " | ||||||
| // | // | ||||||
| //	@Success	200			{object}	models.UserJSONWithClientsAndKeys | //	@Success	200			{object}	models.UserWithClientsAndKeys | ||||||
| //	@Failure	400			{object}	ginresp.apiError	"supplied values/parameters cannot be parsed / are invalid" | //	@Failure	400			{object}	ginresp.apiError	"supplied values/parameters cannot be parsed / are invalid" | ||||||
| //	@Failure	500			{object}	ginresp.apiError	"internal server error" | //	@Failure	500			{object}	ginresp.apiError	"internal server error" | ||||||
| // | // | ||||||
| //	@Router		/api/v2/users [POST] | //	@Router		/api/v2/users [POST] | ||||||
| func (h APIHandler) CreateUser(g *gin.Context) ginresp.HTTPResponse { | func (h APIHandler) CreateUser(pctx ginext.PreContext) ginext.HTTPResponse { | ||||||
| 	type body struct { | 	type body struct { | ||||||
| 		FCMToken     string            `json:"fcm_token"` | 		FCMToken     string            `json:"fcm_token"` | ||||||
| 		ProToken     *string           `json:"pro_token"` | 		ProToken     *string           `json:"pro_token"` | ||||||
| @@ -39,12 +40,14 @@ func (h APIHandler) CreateUser(g *gin.Context) ginresp.HTTPResponse { | |||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	var b body | 	var b body | ||||||
| 	ctx, errResp := h.app.StartRequest(g, nil, nil, &b, nil) | 	ctx, g, errResp := pctx.Body(&b).Start() | ||||||
| 	if errResp != nil { | 	if errResp != nil { | ||||||
| 		return *errResp | 		return *errResp | ||||||
| 	} | 	} | ||||||
| 	defer ctx.Cancel() | 	defer ctx.Cancel() | ||||||
|  |  | ||||||
|  | 	return h.app.DoRequest(ctx, g, models.TLockReadWrite, func(ctx *logic.AppContext, finishSuccess func(r ginext.HTTPResponse) ginext.HTTPResponse) ginext.HTTPResponse { | ||||||
|  |  | ||||||
| 		var clientType models.ClientType | 		var clientType models.ClientType | ||||||
| 		if !b.NoClient { | 		if !b.NoClient { | ||||||
| 			if b.FCMToken == "" { | 			if b.FCMToken == "" { | ||||||
| @@ -117,7 +120,7 @@ func (h APIHandler) CreateUser(g *gin.Context) ginresp.HTTPResponse { | |||||||
| 		log.Info().Msg(fmt.Sprintf("Sucessfully created new user %s (client: %v)", userobj.UserID, b.NoClient)) | 		log.Info().Msg(fmt.Sprintf("Sucessfully created new user %s (client: %v)", userobj.UserID, b.NoClient)) | ||||||
|  |  | ||||||
| 		if b.NoClient { | 		if b.NoClient { | ||||||
| 		return ctx.FinishSuccess(ginresp.JSON(http.StatusOK, userobj.JSONWithClients(make([]models.Client, 0), adminKey, sendKey, readKey))) | 			return finishSuccess(ginext.JSON(http.StatusOK, userobj.PreMarshal().WithClients(make([]models.Client, 0), adminKey, sendKey, readKey))) | ||||||
| 		} else { | 		} else { | ||||||
| 			err := h.database.DeleteClientsByFCM(ctx, b.FCMToken) | 			err := h.database.DeleteClientsByFCM(ctx, b.FCMToken) | ||||||
| 			if err != nil { | 			if err != nil { | ||||||
| @@ -129,9 +132,9 @@ func (h APIHandler) CreateUser(g *gin.Context) ginresp.HTTPResponse { | |||||||
| 				return ginresp.APIError(g, 500, apierr.DATABASE_ERROR, "Failed to create client in db", err) | 				return ginresp.APIError(g, 500, apierr.DATABASE_ERROR, "Failed to create client in db", err) | ||||||
| 			} | 			} | ||||||
|  |  | ||||||
| 		return ctx.FinishSuccess(ginresp.JSON(http.StatusOK, userobj.JSONWithClients([]models.Client{client}, adminKey, sendKey, readKey))) | 			return finishSuccess(ginext.JSON(http.StatusOK, userobj.PreMarshal().WithClients([]models.Client{client}, adminKey, sendKey, readKey))) | ||||||
| 		} | 		} | ||||||
|  | 	}) | ||||||
| } | } | ||||||
|  |  | ||||||
| // GetUser swaggerdoc | // GetUser swaggerdoc | ||||||
| @@ -142,25 +145,27 @@ func (h APIHandler) CreateUser(g *gin.Context) ginresp.HTTPResponse { | |||||||
| // | // | ||||||
| //	@Param		uid	path		string	true	"UserID" | //	@Param		uid	path		string	true	"UserID" | ||||||
| // | // | ||||||
| //	@Success	200	{object}	models.UserJSON | //	@Success	200	{object}	models.User | ||||||
| //	@Failure	400	{object}	ginresp.apiError	"supplied values/parameters cannot be parsed / are invalid" | //	@Failure	400	{object}	ginresp.apiError	"supplied values/parameters cannot be parsed / are invalid" | ||||||
| //	@Failure	401	{object}	ginresp.apiError	"user is not authorized / has missing permissions" | //	@Failure	401	{object}	ginresp.apiError	"user is not authorized / has missing permissions" | ||||||
| //	@Failure	404	{object}	ginresp.apiError	"user not found" | //	@Failure	404	{object}	ginresp.apiError	"user not found" | ||||||
| //	@Failure	500	{object}	ginresp.apiError	"internal server error" | //	@Failure	500	{object}	ginresp.apiError	"internal server error" | ||||||
| // | // | ||||||
| //	@Router		/api/v2/users/{uid} [GET] | //	@Router		/api/v2/users/{uid} [GET] | ||||||
| func (h APIHandler) GetUser(g *gin.Context) ginresp.HTTPResponse { | func (h APIHandler) GetUser(pctx ginext.PreContext) ginext.HTTPResponse { | ||||||
| 	type uri struct { | 	type uri struct { | ||||||
| 		UserID models.UserID `uri:"uid" binding:"entityid"` | 		UserID models.UserID `uri:"uid" binding:"entityid"` | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	var u uri | 	var u uri | ||||||
| 	ctx, errResp := h.app.StartRequest(g, &u, nil, nil, nil) | 	ctx, g, errResp := pctx.URI(&u).Start() | ||||||
| 	if errResp != nil { | 	if errResp != nil { | ||||||
| 		return *errResp | 		return *errResp | ||||||
| 	} | 	} | ||||||
| 	defer ctx.Cancel() | 	defer ctx.Cancel() | ||||||
|  |  | ||||||
|  | 	return h.app.DoRequest(ctx, g, models.TLockRead, func(ctx *logic.AppContext, finishSuccess func(r ginext.HTTPResponse) ginext.HTTPResponse) ginext.HTTPResponse { | ||||||
|  |  | ||||||
| 		if permResp := ctx.CheckPermissionUserRead(u.UserID); permResp != nil { | 		if permResp := ctx.CheckPermissionUserRead(u.UserID); permResp != nil { | ||||||
| 			return *permResp | 			return *permResp | ||||||
| 		} | 		} | ||||||
| @@ -173,7 +178,10 @@ func (h APIHandler) GetUser(g *gin.Context) ginresp.HTTPResponse { | |||||||
| 			return ginresp.APIError(g, 500, apierr.DATABASE_ERROR, "Failed to query user", err) | 			return ginresp.APIError(g, 500, apierr.DATABASE_ERROR, "Failed to query user", err) | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 	return ctx.FinishSuccess(ginresp.JSON(http.StatusOK, user.JSON())) | 		return finishSuccess(ginext.JSON(http.StatusOK, user.PreMarshal())) | ||||||
|  |  | ||||||
|  | 	}) | ||||||
|  |  | ||||||
| } | } | ||||||
|  |  | ||||||
| // UpdateUser swaggerdoc | // UpdateUser swaggerdoc | ||||||
| @@ -188,14 +196,14 @@ func (h APIHandler) GetUser(g *gin.Context) ginresp.HTTPResponse { | |||||||
| //	@Param			username	body		string	false	"Change the username (send an empty string to clear it)" | //	@Param			username	body		string	false	"Change the username (send an empty string to clear it)" | ||||||
| //	@Param			pro_token	body		string	false	"Send a verification of premium purchase" | //	@Param			pro_token	body		string	false	"Send a verification of premium purchase" | ||||||
| // | // | ||||||
| //	@Success		200			{object}	models.UserJSON | //	@Success		200			{object}	models.User | ||||||
| //	@Failure		400			{object}	ginresp.apiError	"supplied values/parameters cannot be parsed / are invalid" | //	@Failure		400			{object}	ginresp.apiError	"supplied values/parameters cannot be parsed / are invalid" | ||||||
| //	@Failure		401			{object}	ginresp.apiError	"user is not authorized / has missing permissions" | //	@Failure		401			{object}	ginresp.apiError	"user is not authorized / has missing permissions" | ||||||
| //	@Failure		404			{object}	ginresp.apiError	"user not found" | //	@Failure		404			{object}	ginresp.apiError	"user not found" | ||||||
| //	@Failure		500			{object}	ginresp.apiError	"internal server error" | //	@Failure		500			{object}	ginresp.apiError	"internal server error" | ||||||
| // | // | ||||||
| //	@Router			/api/v2/users/{uid} [PATCH] | //	@Router			/api/v2/users/{uid} [PATCH] | ||||||
| func (h APIHandler) UpdateUser(g *gin.Context) ginresp.HTTPResponse { | func (h APIHandler) UpdateUser(pctx ginext.PreContext) ginext.HTTPResponse { | ||||||
| 	type uri struct { | 	type uri struct { | ||||||
| 		UserID models.UserID `uri:"uid" binding:"entityid"` | 		UserID models.UserID `uri:"uid" binding:"entityid"` | ||||||
| 	} | 	} | ||||||
| @@ -206,12 +214,14 @@ func (h APIHandler) UpdateUser(g *gin.Context) ginresp.HTTPResponse { | |||||||
|  |  | ||||||
| 	var u uri | 	var u uri | ||||||
| 	var b body | 	var b body | ||||||
| 	ctx, errResp := h.app.StartRequest(g, &u, nil, &b, nil) | 	ctx, g, errResp := pctx.URI(&u).Body(&b).Start() | ||||||
| 	if errResp != nil { | 	if errResp != nil { | ||||||
| 		return *errResp | 		return *errResp | ||||||
| 	} | 	} | ||||||
| 	defer ctx.Cancel() | 	defer ctx.Cancel() | ||||||
|  |  | ||||||
|  | 	return h.app.DoRequest(ctx, g, models.TLockReadWrite, func(ctx *logic.AppContext, finishSuccess func(r ginext.HTTPResponse) ginext.HTTPResponse) ginext.HTTPResponse { | ||||||
|  |  | ||||||
| 		if permResp := ctx.CheckPermissionUserAdmin(u.UserID); permResp != nil { | 		if permResp := ctx.CheckPermissionUserAdmin(u.UserID); permResp != nil { | ||||||
| 			return *permResp | 			return *permResp | ||||||
| 		} | 		} | ||||||
| @@ -261,5 +271,6 @@ func (h APIHandler) UpdateUser(g *gin.Context) ginresp.HTTPResponse { | |||||||
| 			return ginresp.APIError(g, 500, apierr.DATABASE_ERROR, "Failed to query (updated) user", err) | 			return ginresp.APIError(g, 500, apierr.DATABASE_ERROR, "Failed to query (updated) user", err) | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 	return ctx.FinishSuccess(ginresp.JSON(http.StatusOK, user.JSON())) | 		return finishSuccess(ginext.JSON(http.StatusOK, user.PreMarshal())) | ||||||
|  | 	}) | ||||||
| } | } | ||||||
|   | |||||||
| @@ -5,11 +5,12 @@ import ( | |||||||
| 	"blackforestbytes.com/simplecloudnotifier/api/ginresp" | 	"blackforestbytes.com/simplecloudnotifier/api/ginresp" | ||||||
| 	"blackforestbytes.com/simplecloudnotifier/db/simplectx" | 	"blackforestbytes.com/simplecloudnotifier/db/simplectx" | ||||||
| 	"blackforestbytes.com/simplecloudnotifier/logic" | 	"blackforestbytes.com/simplecloudnotifier/logic" | ||||||
|  | 	"blackforestbytes.com/simplecloudnotifier/models" | ||||||
| 	"bytes" | 	"bytes" | ||||||
| 	"context" |  | ||||||
| 	"errors" | 	"errors" | ||||||
| 	"github.com/gin-gonic/gin" | 	"github.com/gin-gonic/gin" | ||||||
| 	sqlite3 "github.com/mattn/go-sqlite3" | 	"github.com/mattn/go-sqlite3" | ||||||
|  | 	"gogs.mikescher.com/BlackForestBytes/goext/ginext" | ||||||
| 	"gogs.mikescher.com/BlackForestBytes/goext/langext" | 	"gogs.mikescher.com/BlackForestBytes/goext/langext" | ||||||
| 	"gogs.mikescher.com/BlackForestBytes/goext/timeext" | 	"gogs.mikescher.com/BlackForestBytes/goext/timeext" | ||||||
| 	"net/http" | 	"net/http" | ||||||
| @@ -51,12 +52,20 @@ type pingResponseInfo struct { | |||||||
| //	@Router		/api/ping [put] | //	@Router		/api/ping [put] | ||||||
| //	@Router		/api/ping [delete] | //	@Router		/api/ping [delete] | ||||||
| //	@Router		/api/ping [patch] | //	@Router		/api/ping [patch] | ||||||
| func (h CommonHandler) Ping(g *gin.Context) ginresp.HTTPResponse { | func (h CommonHandler) Ping(pctx ginext.PreContext) ginext.HTTPResponse { | ||||||
|  | 	ctx, g, errResp := pctx.Start() | ||||||
|  | 	if errResp != nil { | ||||||
|  | 		return *errResp | ||||||
|  | 	} | ||||||
|  | 	defer ctx.Cancel() | ||||||
|  |  | ||||||
|  | 	return h.app.DoRequest(ctx, g, models.TLockRead, func(ctx *logic.AppContext, finishSuccess func(r ginext.HTTPResponse) ginext.HTTPResponse) ginext.HTTPResponse { | ||||||
|  |  | ||||||
| 		buf := new(bytes.Buffer) | 		buf := new(bytes.Buffer) | ||||||
| 		_, _ = buf.ReadFrom(g.Request.Body) | 		_, _ = buf.ReadFrom(g.Request.Body) | ||||||
| 		resuestBody := buf.String() | 		resuestBody := buf.String() | ||||||
|  |  | ||||||
| 	return ginresp.JSON(http.StatusOK, pingResponse{ | 		return ginext.JSON(http.StatusOK, pingResponse{ | ||||||
| 			Message: "Pong", | 			Message: "Pong", | ||||||
| 			Info: pingResponseInfo{ | 			Info: pingResponseInfo{ | ||||||
| 				Method:  g.Request.Method, | 				Method:  g.Request.Method, | ||||||
| @@ -66,6 +75,8 @@ func (h CommonHandler) Ping(g *gin.Context) ginresp.HTTPResponse { | |||||||
| 				Address: g.Request.RemoteAddr, | 				Address: g.Request.RemoteAddr, | ||||||
| 			}, | 			}, | ||||||
| 		}) | 		}) | ||||||
|  |  | ||||||
|  | 	}) | ||||||
| } | } | ||||||
|  |  | ||||||
| // DatabaseTest swaggerdoc | // DatabaseTest swaggerdoc | ||||||
| @@ -78,7 +89,7 @@ func (h CommonHandler) Ping(g *gin.Context) ginresp.HTTPResponse { | |||||||
| //	@Failure	500	{object}	ginresp.apiError | //	@Failure	500	{object}	ginresp.apiError | ||||||
| // | // | ||||||
| //	@Router		/api/db-test [post] | //	@Router		/api/db-test [post] | ||||||
| func (h CommonHandler) DatabaseTest(g *gin.Context) ginresp.HTTPResponse { | func (h CommonHandler) DatabaseTest(pctx ginext.PreContext) ginext.HTTPResponse { | ||||||
| 	type response struct { | 	type response struct { | ||||||
| 		Success          bool   `json:"success"` | 		Success          bool   `json:"success"` | ||||||
| 		LibVersion       string `json:"libVersion"` | 		LibVersion       string `json:"libVersion"` | ||||||
| @@ -86,8 +97,13 @@ func (h CommonHandler) DatabaseTest(g *gin.Context) ginresp.HTTPResponse { | |||||||
| 		SourceID         string `json:"sourceID"` | 		SourceID         string `json:"sourceID"` | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	ctx, cancel := context.WithTimeout(context.Background(), 2*time.Second) | 	ctx, g, errResp := pctx.Start() | ||||||
| 	defer cancel() | 	if errResp != nil { | ||||||
|  | 		return *errResp | ||||||
|  | 	} | ||||||
|  | 	defer ctx.Cancel() | ||||||
|  |  | ||||||
|  | 	return h.app.DoRequest(ctx, g, models.TLockRead, func(ctx *logic.AppContext, finishSuccess func(r ginext.HTTPResponse) ginext.HTTPResponse) ginext.HTTPResponse { | ||||||
|  |  | ||||||
| 		libVersion, libVersionNumber, sourceID := sqlite3.Version() | 		libVersion, libVersionNumber, sourceID := sqlite3.Version() | ||||||
|  |  | ||||||
| @@ -96,12 +112,14 @@ func (h CommonHandler) DatabaseTest(g *gin.Context) ginresp.HTTPResponse { | |||||||
| 			return ginresp.InternalError(err) | 			return ginresp.InternalError(err) | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 	return ginresp.JSON(http.StatusOK, response{ | 		return ginext.JSON(http.StatusOK, response{ | ||||||
| 			Success:          true, | 			Success:          true, | ||||||
| 			LibVersion:       libVersion, | 			LibVersion:       libVersion, | ||||||
| 			LibVersionNumber: libVersionNumber, | 			LibVersionNumber: libVersionNumber, | ||||||
| 			SourceID:         sourceID, | 			SourceID:         sourceID, | ||||||
| 		}) | 		}) | ||||||
|  |  | ||||||
|  | 	}) | ||||||
| } | } | ||||||
|  |  | ||||||
| // Health swaggerdoc | // Health swaggerdoc | ||||||
| @@ -114,13 +132,18 @@ func (h CommonHandler) DatabaseTest(g *gin.Context) ginresp.HTTPResponse { | |||||||
| //	@Failure	500	{object}	ginresp.apiError | //	@Failure	500	{object}	ginresp.apiError | ||||||
| // | // | ||||||
| //	@Router		/api/health [get] | //	@Router		/api/health [get] | ||||||
| func (h CommonHandler) Health(g *gin.Context) ginresp.HTTPResponse { | func (h CommonHandler) Health(pctx ginext.PreContext) ginext.HTTPResponse { | ||||||
| 	type response struct { | 	type response struct { | ||||||
| 		Status string `json:"status"` | 		Status string `json:"status"` | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	ctx, cancel := context.WithTimeout(context.Background(), 2*time.Second) | 	ctx, g, errResp := pctx.Start() | ||||||
| 	defer cancel() | 	if errResp != nil { | ||||||
|  | 		return *errResp | ||||||
|  | 	} | ||||||
|  | 	defer ctx.Cancel() | ||||||
|  |  | ||||||
|  | 	return h.app.DoRequest(ctx, g, models.TLockReadWrite, func(ctx *logic.AppContext, finishSuccess func(r ginext.HTTPResponse) ginext.HTTPResponse) ginext.HTTPResponse { | ||||||
|  |  | ||||||
| 		_, libVersionNumber, _ := sqlite3.Version() | 		_, libVersionNumber, _ := sqlite3.Version() | ||||||
|  |  | ||||||
| @@ -161,7 +184,9 @@ func (h CommonHandler) Health(g *gin.Context) ginresp.HTTPResponse { | |||||||
|  |  | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 	return ginresp.JSON(http.StatusOK, response{Status: "ok"}) | 		return ginext.JSON(http.StatusOK, response{Status: "ok"}) | ||||||
|  |  | ||||||
|  | 	}) | ||||||
| } | } | ||||||
|  |  | ||||||
| // Sleep swaggerdoc | // Sleep swaggerdoc | ||||||
| @@ -177,7 +202,7 @@ func (h CommonHandler) Health(g *gin.Context) ginresp.HTTPResponse { | |||||||
| //	@Failure	500		{object}	ginresp.apiError | //	@Failure	500		{object}	ginresp.apiError | ||||||
| // | // | ||||||
| //	@Router		/api/sleep/{secs} [post] | //	@Router		/api/sleep/{secs} [post] | ||||||
| func (h CommonHandler) Sleep(g *gin.Context) ginresp.HTTPResponse { | func (h CommonHandler) Sleep(pctx ginext.PreContext) ginext.HTTPResponse { | ||||||
| 	type uri struct { | 	type uri struct { | ||||||
| 		Seconds float64 `uri:"secs"` | 		Seconds float64 `uri:"secs"` | ||||||
| 	} | 	} | ||||||
| @@ -187,6 +212,14 @@ func (h CommonHandler) Sleep(g *gin.Context) ginresp.HTTPResponse { | |||||||
| 		Duration float64 `json:"duration"` | 		Duration float64 `json:"duration"` | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
|  | 	ctx, g, errResp := pctx.Start() | ||||||
|  | 	if errResp != nil { | ||||||
|  | 		return *errResp | ||||||
|  | 	} | ||||||
|  | 	defer ctx.Cancel() | ||||||
|  |  | ||||||
|  | 	return h.app.DoRequest(ctx, g, models.TLockRead, func(ctx *logic.AppContext, finishSuccess func(r ginext.HTTPResponse) ginext.HTTPResponse) ginext.HTTPResponse { | ||||||
|  |  | ||||||
| 		t0 := time.Now().Format(time.RFC3339Nano) | 		t0 := time.Now().Format(time.RFC3339Nano) | ||||||
|  |  | ||||||
| 		var u uri | 		var u uri | ||||||
| @@ -198,15 +231,25 @@ func (h CommonHandler) Sleep(g *gin.Context) ginresp.HTTPResponse { | |||||||
|  |  | ||||||
| 		t1 := time.Now().Format(time.RFC3339Nano) | 		t1 := time.Now().Format(time.RFC3339Nano) | ||||||
|  |  | ||||||
| 	return ginresp.JSON(http.StatusOK, response{ | 		return ginext.JSON(http.StatusOK, response{ | ||||||
| 			Start:    t0, | 			Start:    t0, | ||||||
| 			End:      t1, | 			End:      t1, | ||||||
| 			Duration: u.Seconds, | 			Duration: u.Seconds, | ||||||
| 		}) | 		}) | ||||||
|  |  | ||||||
|  | 	}) | ||||||
| } | } | ||||||
|  |  | ||||||
| func (h CommonHandler) NoRoute(g *gin.Context) ginresp.HTTPResponse { | func (h CommonHandler) NoRoute(pctx ginext.PreContext) ginext.HTTPResponse { | ||||||
| 	return ginresp.JSON(http.StatusNotFound, gin.H{ | 	ctx, g, errResp := pctx.Start() | ||||||
|  | 	if errResp != nil { | ||||||
|  | 		return *errResp | ||||||
|  | 	} | ||||||
|  | 	defer ctx.Cancel() | ||||||
|  |  | ||||||
|  | 	return h.app.DoRequest(ctx, g, models.TLockRead, func(ctx *logic.AppContext, finishSuccess func(r ginext.HTTPResponse) ginext.HTTPResponse) ginext.HTTPResponse { | ||||||
|  |  | ||||||
|  | 		return ginext.JSON(http.StatusNotFound, gin.H{ | ||||||
| 			"":           "================ ROUTE NOT FOUND ================", | 			"":           "================ ROUTE NOT FOUND ================", | ||||||
| 			"FullPath":   g.FullPath(), | 			"FullPath":   g.FullPath(), | ||||||
| 			"Method":     g.Request.Method, | 			"Method":     g.Request.Method, | ||||||
| @@ -216,4 +259,6 @@ func (h CommonHandler) NoRoute(g *gin.Context) ginresp.HTTPResponse { | |||||||
| 			"Header":     g.Request.Header, | 			"Header":     g.Request.Header, | ||||||
| 			"~":          "================ ROUTE NOT FOUND ================", | 			"~":          "================ ROUTE NOT FOUND ================", | ||||||
| 		}) | 		}) | ||||||
|  |  | ||||||
|  | 	}) | ||||||
| } | } | ||||||
|   | |||||||
| @@ -11,8 +11,8 @@ import ( | |||||||
| 	"database/sql" | 	"database/sql" | ||||||
| 	"errors" | 	"errors" | ||||||
| 	"fmt" | 	"fmt" | ||||||
| 	"github.com/gin-gonic/gin" |  | ||||||
| 	"gogs.mikescher.com/BlackForestBytes/goext/dataext" | 	"gogs.mikescher.com/BlackForestBytes/goext/dataext" | ||||||
|  | 	"gogs.mikescher.com/BlackForestBytes/goext/ginext" | ||||||
| 	"gogs.mikescher.com/BlackForestBytes/goext/langext" | 	"gogs.mikescher.com/BlackForestBytes/goext/langext" | ||||||
| 	"net/http" | 	"net/http" | ||||||
| ) | ) | ||||||
| @@ -47,7 +47,7 @@ func NewCompatHandler(app *logic.Application) CompatHandler { | |||||||
| //	@Failure		500			{object}	ginresp.apiError | //	@Failure		500			{object}	ginresp.apiError | ||||||
| // | // | ||||||
| //	@Router			/send.php [POST] | //	@Router			/send.php [POST] | ||||||
| func (h CompatHandler) SendMessage(g *gin.Context) ginresp.HTTPResponse { | func (h CompatHandler) SendMessage(pctx ginext.PreContext) ginext.HTTPResponse { | ||||||
| 	type combined struct { | 	type combined struct { | ||||||
| 		UserID        *int64   `json:"user_id"   form:"user_id"` | 		UserID        *int64   `json:"user_id"   form:"user_id"` | ||||||
| 		UserKey       *string  `json:"user_key"  form:"user_key"` | 		UserKey       *string  `json:"user_key"  form:"user_key"` | ||||||
| @@ -72,12 +72,14 @@ func (h CompatHandler) SendMessage(g *gin.Context) ginresp.HTTPResponse { | |||||||
|  |  | ||||||
| 	var f combined | 	var f combined | ||||||
| 	var q combined | 	var q combined | ||||||
| 	ctx, errResp := h.app.StartRequest(g, nil, &q, nil, &f, logic.RequestOptions{IgnoreWrongContentType: true}) | 	ctx, g, errResp := pctx.Query(&q).Form(&f).IgnoreWrongContentType().Start() | ||||||
| 	if errResp != nil { | 	if errResp != nil { | ||||||
| 		return *errResp | 		return *errResp | ||||||
| 	} | 	} | ||||||
| 	defer ctx.Cancel() | 	defer ctx.Cancel() | ||||||
|  |  | ||||||
|  | 	return h.app.DoRequest(ctx, g, models.TLockReadWrite, func(ctx *logic.AppContext, finishSuccess func(r ginext.HTTPResponse) ginext.HTTPResponse) ginext.HTTPResponse { | ||||||
|  |  | ||||||
| 		data := dataext.ObjectMerge(f, q) | 		data := dataext.ObjectMerge(f, q) | ||||||
|  |  | ||||||
| 		newid, err := h.database.ConvertCompatID(ctx, langext.Coalesce(data.UserID, -1), "userid") | 		newid, err := h.database.ConvertCompatID(ctx, langext.Coalesce(data.UserID, -1), "userid") | ||||||
| @@ -92,7 +94,7 @@ func (h CompatHandler) SendMessage(g *gin.Context) ginresp.HTTPResponse { | |||||||
| 		if errResp != nil { | 		if errResp != nil { | ||||||
| 			return *errResp | 			return *errResp | ||||||
| 		} else { | 		} else { | ||||||
| 		return ctx.FinishSuccess(ginresp.JSON(http.StatusOK, response{ | 			return finishSuccess(ginext.JSON(http.StatusOK, response{ | ||||||
| 				Success:        true, | 				Success:        true, | ||||||
| 				ErrorID:        apierr.NO_ERROR, | 				ErrorID:        apierr.NO_ERROR, | ||||||
| 				ErrorHighlight: -1, | 				ErrorHighlight: -1, | ||||||
| @@ -105,6 +107,7 @@ func (h CompatHandler) SendMessage(g *gin.Context) ginresp.HTTPResponse { | |||||||
| 				SCNMessageID:   okResp.CompatMessageID, | 				SCNMessageID:   okResp.CompatMessageID, | ||||||
| 			})) | 			})) | ||||||
| 		} | 		} | ||||||
|  | 	}) | ||||||
| } | } | ||||||
|  |  | ||||||
| // Register swaggerdoc | // Register swaggerdoc | ||||||
| @@ -127,7 +130,7 @@ func (h CompatHandler) SendMessage(g *gin.Context) ginresp.HTTPResponse { | |||||||
| //	@Failure	default		{object}	ginresp.compatAPIError | //	@Failure	default		{object}	ginresp.compatAPIError | ||||||
| // | // | ||||||
| //	@Router		/api/register.php [get] | //	@Router		/api/register.php [get] | ||||||
| func (h CompatHandler) Register(g *gin.Context) ginresp.HTTPResponse { | func (h CompatHandler) Register(pctx ginext.PreContext) ginext.HTTPResponse { | ||||||
| 	type query struct { | 	type query struct { | ||||||
| 		FCMToken *string `json:"fcm_token" form:"fcm_token"` | 		FCMToken *string `json:"fcm_token" form:"fcm_token"` | ||||||
| 		Pro      *string `json:"pro"       form:"pro"` | 		Pro      *string `json:"pro"       form:"pro"` | ||||||
| @@ -145,12 +148,14 @@ func (h CompatHandler) Register(g *gin.Context) ginresp.HTTPResponse { | |||||||
|  |  | ||||||
| 	var datq query | 	var datq query | ||||||
| 	var datb query | 	var datb query | ||||||
| 	ctx, errResp := h.app.StartRequest(g, nil, &datq, nil, &datb, logic.RequestOptions{IgnoreWrongContentType: true}) | 	ctx, g, errResp := pctx.Query(&datq).Body(&datb).IgnoreWrongContentType().Start() | ||||||
| 	if errResp != nil { | 	if errResp != nil { | ||||||
| 		return *errResp | 		return *errResp | ||||||
| 	} | 	} | ||||||
| 	defer ctx.Cancel() | 	defer ctx.Cancel() | ||||||
|  |  | ||||||
|  | 	return h.app.DoRequest(ctx, g, models.TLockReadWrite, func(ctx *logic.AppContext, finishSuccess func(r ginext.HTTPResponse) ginext.HTTPResponse) ginext.HTTPResponse { | ||||||
|  |  | ||||||
| 		data := dataext.ObjectMerge(datb, datq) | 		data := dataext.ObjectMerge(datb, datq) | ||||||
|  |  | ||||||
| 		if data.FCMToken == nil { | 		if data.FCMToken == nil { | ||||||
| @@ -216,7 +221,7 @@ func (h CompatHandler) Register(g *gin.Context) ginresp.HTTPResponse { | |||||||
| 			return ginresp.SendAPIError(g, 500, apierr.DATABASE_ERROR, hl.NONE, "Failed to create userid<old>", err) | 			return ginresp.SendAPIError(g, 500, apierr.DATABASE_ERROR, hl.NONE, "Failed to create userid<old>", err) | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 	return ctx.FinishSuccess(ginresp.JSON(http.StatusOK, response{ | 		return finishSuccess(ginext.JSON(http.StatusOK, response{ | ||||||
| 			Success:   true, | 			Success:   true, | ||||||
| 			Message:   "New user registered", | 			Message:   "New user registered", | ||||||
| 			UserID:    oldid, | 			UserID:    oldid, | ||||||
| @@ -225,6 +230,8 @@ func (h CompatHandler) Register(g *gin.Context) ginresp.HTTPResponse { | |||||||
| 			QuotaMax:  user.QuotaPerDay(), | 			QuotaMax:  user.QuotaPerDay(), | ||||||
| 			IsPro:     user.IsPro, | 			IsPro:     user.IsPro, | ||||||
| 		})) | 		})) | ||||||
|  |  | ||||||
|  | 	}) | ||||||
| } | } | ||||||
|  |  | ||||||
| // Info swaggerdoc | // Info swaggerdoc | ||||||
| @@ -245,7 +252,7 @@ func (h CompatHandler) Register(g *gin.Context) ginresp.HTTPResponse { | |||||||
| //	@Failure	default		{object}	ginresp.compatAPIError | //	@Failure	default		{object}	ginresp.compatAPIError | ||||||
| // | // | ||||||
| //	@Router		/api/info.php [get] | //	@Router		/api/info.php [get] | ||||||
| func (h CompatHandler) Info(g *gin.Context) ginresp.HTTPResponse { | func (h CompatHandler) Info(pctx ginext.PreContext) ginext.HTTPResponse { | ||||||
| 	type query struct { | 	type query struct { | ||||||
| 		UserID  *int64  `json:"user_id"  form:"user_id"` | 		UserID  *int64  `json:"user_id"  form:"user_id"` | ||||||
| 		UserKey *string `json:"user_key" form:"user_key"` | 		UserKey *string `json:"user_key" form:"user_key"` | ||||||
| @@ -264,12 +271,14 @@ func (h CompatHandler) Info(g *gin.Context) ginresp.HTTPResponse { | |||||||
|  |  | ||||||
| 	var datq query | 	var datq query | ||||||
| 	var datb query | 	var datb query | ||||||
| 	ctx, errResp := h.app.StartRequest(g, nil, &datq, nil, &datb, logic.RequestOptions{IgnoreWrongContentType: true}) | 	ctx, g, errResp := pctx.Query(&datq).Body(&datb).IgnoreWrongContentType().Start() | ||||||
| 	if errResp != nil { | 	if errResp != nil { | ||||||
| 		return *errResp | 		return *errResp | ||||||
| 	} | 	} | ||||||
| 	defer ctx.Cancel() | 	defer ctx.Cancel() | ||||||
|  |  | ||||||
|  | 	return h.app.DoRequest(ctx, g, models.TLockRead, func(ctx *logic.AppContext, finishSuccess func(r ginext.HTTPResponse) ginext.HTTPResponse) ginext.HTTPResponse { | ||||||
|  |  | ||||||
| 		data := dataext.ObjectMerge(datb, datq) | 		data := dataext.ObjectMerge(datb, datq) | ||||||
|  |  | ||||||
| 		if data.UserID == nil { | 		if data.UserID == nil { | ||||||
| @@ -321,7 +330,7 @@ func (h CompatHandler) Info(g *gin.Context) ginresp.HTTPResponse { | |||||||
| 			return ginresp.CompatAPIError(0, "Failed to query user") | 			return ginresp.CompatAPIError(0, "Failed to query user") | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 	return ctx.FinishSuccess(ginresp.JSON(http.StatusOK, response{ | 		return finishSuccess(ginext.JSON(http.StatusOK, response{ | ||||||
| 			Success:    true, | 			Success:    true, | ||||||
| 			Message:    "ok", | 			Message:    "ok", | ||||||
| 			UserID:     *data.UserID, | 			UserID:     *data.UserID, | ||||||
| @@ -332,6 +341,8 @@ func (h CompatHandler) Info(g *gin.Context) ginresp.HTTPResponse { | |||||||
| 			FCMSet:     len(clients) > 0, | 			FCMSet:     len(clients) > 0, | ||||||
| 			UnackCount: unackCount, | 			UnackCount: unackCount, | ||||||
| 		})) | 		})) | ||||||
|  |  | ||||||
|  | 	}) | ||||||
| } | } | ||||||
|  |  | ||||||
| // Ack swaggerdoc | // Ack swaggerdoc | ||||||
| @@ -354,7 +365,7 @@ func (h CompatHandler) Info(g *gin.Context) ginresp.HTTPResponse { | |||||||
| //	@Failure	default		{object}	ginresp.compatAPIError | //	@Failure	default		{object}	ginresp.compatAPIError | ||||||
| // | // | ||||||
| //	@Router		/api/ack.php [get] | //	@Router		/api/ack.php [get] | ||||||
| func (h CompatHandler) Ack(g *gin.Context) ginresp.HTTPResponse { | func (h CompatHandler) Ack(pctx ginext.PreContext) ginext.HTTPResponse { | ||||||
| 	type query struct { | 	type query struct { | ||||||
| 		UserID    *int64  `json:"user_id"    form:"user_id"` | 		UserID    *int64  `json:"user_id"    form:"user_id"` | ||||||
| 		UserKey   *string `json:"user_key"   form:"user_key"` | 		UserKey   *string `json:"user_key"   form:"user_key"` | ||||||
| @@ -369,12 +380,14 @@ func (h CompatHandler) Ack(g *gin.Context) ginresp.HTTPResponse { | |||||||
|  |  | ||||||
| 	var datq query | 	var datq query | ||||||
| 	var datb query | 	var datb query | ||||||
| 	ctx, errResp := h.app.StartRequest(g, nil, &datq, nil, &datb, logic.RequestOptions{IgnoreWrongContentType: true}) | 	ctx, g, errResp := pctx.Query(&datq).Body(&datb).IgnoreWrongContentType().Start() | ||||||
| 	if errResp != nil { | 	if errResp != nil { | ||||||
| 		return *errResp | 		return *errResp | ||||||
| 	} | 	} | ||||||
| 	defer ctx.Cancel() | 	defer ctx.Cancel() | ||||||
|  |  | ||||||
|  | 	return h.app.DoRequest(ctx, g, models.TLockReadWrite, func(ctx *logic.AppContext, finishSuccess func(r ginext.HTTPResponse) ginext.HTTPResponse) ginext.HTTPResponse { | ||||||
|  |  | ||||||
| 		data := dataext.ObjectMerge(datb, datq) | 		data := dataext.ObjectMerge(datb, datq) | ||||||
|  |  | ||||||
| 		if data.UserID == nil { | 		if data.UserID == nil { | ||||||
| @@ -434,12 +447,14 @@ func (h CompatHandler) Ack(g *gin.Context) ginresp.HTTPResponse { | |||||||
| 			} | 			} | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 	return ctx.FinishSuccess(ginresp.JSON(http.StatusOK, response{ | 		return finishSuccess(ginext.JSON(http.StatusOK, response{ | ||||||
| 			Success:      true, | 			Success:      true, | ||||||
| 			Message:      "ok", | 			Message:      "ok", | ||||||
| 			PrevAckValue: langext.Conditional(ackBefore, 1, 0), | 			PrevAckValue: langext.Conditional(ackBefore, 1, 0), | ||||||
| 			NewAckValue:  1, | 			NewAckValue:  1, | ||||||
| 		})) | 		})) | ||||||
|  |  | ||||||
|  | 	}) | ||||||
| } | } | ||||||
|  |  | ||||||
| // Requery swaggerdoc | // Requery swaggerdoc | ||||||
| @@ -460,7 +475,7 @@ func (h CompatHandler) Ack(g *gin.Context) ginresp.HTTPResponse { | |||||||
| //	@Failure	default		{object}	ginresp.compatAPIError | //	@Failure	default		{object}	ginresp.compatAPIError | ||||||
| // | // | ||||||
| //	@Router		/api/requery.php [get] | //	@Router		/api/requery.php [get] | ||||||
| func (h CompatHandler) Requery(g *gin.Context) ginresp.HTTPResponse { | func (h CompatHandler) Requery(pctx ginext.PreContext) ginext.HTTPResponse { | ||||||
| 	type query struct { | 	type query struct { | ||||||
| 		UserID  *int64  `json:"user_id"  form:"user_id"` | 		UserID  *int64  `json:"user_id"  form:"user_id"` | ||||||
| 		UserKey *string `json:"user_key" form:"user_key"` | 		UserKey *string `json:"user_key" form:"user_key"` | ||||||
| @@ -474,12 +489,14 @@ func (h CompatHandler) Requery(g *gin.Context) ginresp.HTTPResponse { | |||||||
|  |  | ||||||
| 	var datq query | 	var datq query | ||||||
| 	var datb query | 	var datb query | ||||||
| 	ctx, errResp := h.app.StartRequest(g, nil, &datq, nil, &datb, logic.RequestOptions{IgnoreWrongContentType: true}) | 	ctx, g, errResp := pctx.Query(&datq).Body(&datb).IgnoreWrongContentType().Start() | ||||||
| 	if errResp != nil { | 	if errResp != nil { | ||||||
| 		return *errResp | 		return *errResp | ||||||
| 	} | 	} | ||||||
| 	defer ctx.Cancel() | 	defer ctx.Cancel() | ||||||
|  |  | ||||||
|  | 	return h.app.DoRequest(ctx, g, models.TLockReadWrite, func(ctx *logic.AppContext, finishSuccess func(r ginext.HTTPResponse) ginext.HTTPResponse) ginext.HTTPResponse { | ||||||
|  |  | ||||||
| 		data := dataext.ObjectMerge(datb, datq) | 		data := dataext.ObjectMerge(datb, datq) | ||||||
|  |  | ||||||
| 		if data.UserID == nil { | 		if data.UserID == nil { | ||||||
| @@ -545,12 +562,14 @@ func (h CompatHandler) Requery(g *gin.Context) ginresp.HTTPResponse { | |||||||
| 			}) | 			}) | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 	return ctx.FinishSuccess(ginresp.JSON(http.StatusOK, response{ | 		return finishSuccess(ginext.JSON(http.StatusOK, response{ | ||||||
| 			Success: true, | 			Success: true, | ||||||
| 			Message: "ok", | 			Message: "ok", | ||||||
| 			Count:   len(compMsgs), | 			Count:   len(compMsgs), | ||||||
| 			Data:    compMsgs, | 			Data:    compMsgs, | ||||||
| 		})) | 		})) | ||||||
|  |  | ||||||
|  | 	}) | ||||||
| } | } | ||||||
|  |  | ||||||
| // Update swaggerdoc | // Update swaggerdoc | ||||||
| @@ -573,7 +592,7 @@ func (h CompatHandler) Requery(g *gin.Context) ginresp.HTTPResponse { | |||||||
| //	@Failure	default		{object}	ginresp.compatAPIError | //	@Failure	default		{object}	ginresp.compatAPIError | ||||||
| // | // | ||||||
| //	@Router		/api/update.php [get] | //	@Router		/api/update.php [get] | ||||||
| func (h CompatHandler) Update(g *gin.Context) ginresp.HTTPResponse { | func (h CompatHandler) Update(pctx ginext.PreContext) ginext.HTTPResponse { | ||||||
| 	type query struct { | 	type query struct { | ||||||
| 		UserID   *int64  `json:"user_id"   form:"user_id"` | 		UserID   *int64  `json:"user_id"   form:"user_id"` | ||||||
| 		UserKey  *string `json:"user_key"  form:"user_key"` | 		UserKey  *string `json:"user_key"  form:"user_key"` | ||||||
| @@ -591,12 +610,14 @@ func (h CompatHandler) Update(g *gin.Context) ginresp.HTTPResponse { | |||||||
|  |  | ||||||
| 	var datq query | 	var datq query | ||||||
| 	var datb query | 	var datb query | ||||||
| 	ctx, errResp := h.app.StartRequest(g, nil, &datq, nil, &datb, logic.RequestOptions{IgnoreWrongContentType: true}) | 	ctx, g, errResp := pctx.Query(&datq).Body(&datb).IgnoreWrongContentType().Start() | ||||||
| 	if errResp != nil { | 	if errResp != nil { | ||||||
| 		return *errResp | 		return *errResp | ||||||
| 	} | 	} | ||||||
| 	defer ctx.Cancel() | 	defer ctx.Cancel() | ||||||
|  |  | ||||||
|  | 	return h.app.DoRequest(ctx, g, models.TLockReadWrite, func(ctx *logic.AppContext, finishSuccess func(r ginext.HTTPResponse) ginext.HTTPResponse) ginext.HTTPResponse { | ||||||
|  |  | ||||||
| 		data := dataext.ObjectMerge(datb, datq) | 		data := dataext.ObjectMerge(datb, datq) | ||||||
|  |  | ||||||
| 		if data.UserID == nil { | 		if data.UserID == nil { | ||||||
| @@ -673,7 +694,7 @@ func (h CompatHandler) Update(g *gin.Context) ginresp.HTTPResponse { | |||||||
| 			return ginresp.CompatAPIError(0, "Failed to query user") | 			return ginresp.CompatAPIError(0, "Failed to query user") | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 	return ctx.FinishSuccess(ginresp.JSON(http.StatusOK, response{ | 		return finishSuccess(ginext.JSON(http.StatusOK, response{ | ||||||
| 			Success:   true, | 			Success:   true, | ||||||
| 			Message:   "user updated", | 			Message:   "user updated", | ||||||
| 			UserID:    *data.UserID, | 			UserID:    *data.UserID, | ||||||
| @@ -682,6 +703,8 @@ func (h CompatHandler) Update(g *gin.Context) ginresp.HTTPResponse { | |||||||
| 			QuotaMax:  user.QuotaPerDay(), | 			QuotaMax:  user.QuotaPerDay(), | ||||||
| 			IsPro:     langext.Conditional(user.IsPro, 1, 0), | 			IsPro:     langext.Conditional(user.IsPro, 1, 0), | ||||||
| 		})) | 		})) | ||||||
|  |  | ||||||
|  | 	}) | ||||||
| } | } | ||||||
|  |  | ||||||
| // Expand swaggerdoc | // Expand swaggerdoc | ||||||
| @@ -704,7 +727,7 @@ func (h CompatHandler) Update(g *gin.Context) ginresp.HTTPResponse { | |||||||
| //	@Failure	default		{object}	ginresp.compatAPIError | //	@Failure	default		{object}	ginresp.compatAPIError | ||||||
| // | // | ||||||
| //	@Router		/api/expand.php [get] | //	@Router		/api/expand.php [get] | ||||||
| func (h CompatHandler) Expand(g *gin.Context) ginresp.HTTPResponse { | func (h CompatHandler) Expand(pctx ginext.PreContext) ginext.HTTPResponse { | ||||||
| 	type query struct { | 	type query struct { | ||||||
| 		UserID    *int64  `json:"user_id"    form:"user_id"` | 		UserID    *int64  `json:"user_id"    form:"user_id"` | ||||||
| 		UserKey   *string `json:"user_key"   form:"user_key"` | 		UserKey   *string `json:"user_key"   form:"user_key"` | ||||||
| @@ -718,12 +741,14 @@ func (h CompatHandler) Expand(g *gin.Context) ginresp.HTTPResponse { | |||||||
|  |  | ||||||
| 	var datq query | 	var datq query | ||||||
| 	var datb query | 	var datb query | ||||||
| 	ctx, errResp := h.app.StartRequest(g, nil, &datq, nil, &datb, logic.RequestOptions{IgnoreWrongContentType: true}) | 	ctx, g, errResp := pctx.Query(&datq).Body(&datb).IgnoreWrongContentType().Start() | ||||||
| 	if errResp != nil { | 	if errResp != nil { | ||||||
| 		return *errResp | 		return *errResp | ||||||
| 	} | 	} | ||||||
| 	defer ctx.Cancel() | 	defer ctx.Cancel() | ||||||
|  |  | ||||||
|  | 	return h.app.DoRequest(ctx, g, models.TLockRead, func(ctx *logic.AppContext, finishSuccess func(r ginext.HTTPResponse) ginext.HTTPResponse) ginext.HTTPResponse { | ||||||
|  |  | ||||||
| 		data := dataext.ObjectMerge(datb, datq) | 		data := dataext.ObjectMerge(datb, datq) | ||||||
|  |  | ||||||
| 		if data.UserID == nil { | 		if data.UserID == nil { | ||||||
| @@ -779,7 +804,7 @@ func (h CompatHandler) Expand(g *gin.Context) ginresp.HTTPResponse { | |||||||
| 			return ginresp.CompatAPIError(0, "Failed to query message") | 			return ginresp.CompatAPIError(0, "Failed to query message") | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 	return ctx.FinishSuccess(ginresp.JSON(http.StatusOK, response{ | 		return finishSuccess(ginext.JSON(http.StatusOK, response{ | ||||||
| 			Success: true, | 			Success: true, | ||||||
| 			Message: "ok", | 			Message: "ok", | ||||||
| 			Data: models.CompatMessage{ | 			Data: models.CompatMessage{ | ||||||
| @@ -792,6 +817,8 @@ func (h CompatHandler) Expand(g *gin.Context) ginresp.HTTPResponse { | |||||||
| 				SCNMessageID:  *data.MessageID, | 				SCNMessageID:  *data.MessageID, | ||||||
| 			}, | 			}, | ||||||
| 		})) | 		})) | ||||||
|  |  | ||||||
|  | 	}) | ||||||
| } | } | ||||||
|  |  | ||||||
| // Upgrade swaggerdoc | // Upgrade swaggerdoc | ||||||
| @@ -816,7 +843,7 @@ func (h CompatHandler) Expand(g *gin.Context) ginresp.HTTPResponse { | |||||||
| //	@Failure	default		{object}	ginresp.compatAPIError | //	@Failure	default		{object}	ginresp.compatAPIError | ||||||
| // | // | ||||||
| //	@Router		/api/upgrade.php [get] | //	@Router		/api/upgrade.php [get] | ||||||
| func (h CompatHandler) Upgrade(g *gin.Context) ginresp.HTTPResponse { | func (h CompatHandler) Upgrade(pctx ginext.PreContext) ginext.HTTPResponse { | ||||||
| 	type query struct { | 	type query struct { | ||||||
| 		UserID   *int64  `json:"user_id"   form:"user_id"` | 		UserID   *int64  `json:"user_id"   form:"user_id"` | ||||||
| 		UserKey  *string `json:"user_key"  form:"user_key"` | 		UserKey  *string `json:"user_key"  form:"user_key"` | ||||||
| @@ -834,12 +861,14 @@ func (h CompatHandler) Upgrade(g *gin.Context) ginresp.HTTPResponse { | |||||||
|  |  | ||||||
| 	var datq query | 	var datq query | ||||||
| 	var datb query | 	var datb query | ||||||
| 	ctx, errResp := h.app.StartRequest(g, nil, &datq, nil, &datb, logic.RequestOptions{IgnoreWrongContentType: true}) | 	ctx, g, errResp := pctx.Query(&datq).Body(&datb).IgnoreWrongContentType().Start() | ||||||
| 	if errResp != nil { | 	if errResp != nil { | ||||||
| 		return *errResp | 		return *errResp | ||||||
| 	} | 	} | ||||||
| 	defer ctx.Cancel() | 	defer ctx.Cancel() | ||||||
|  |  | ||||||
|  | 	return h.app.DoRequest(ctx, g, models.TLockReadWrite, func(ctx *logic.AppContext, finishSuccess func(r ginext.HTTPResponse) ginext.HTTPResponse) ginext.HTTPResponse { | ||||||
|  |  | ||||||
| 		data := dataext.ObjectMerge(datb, datq) | 		data := dataext.ObjectMerge(datb, datq) | ||||||
|  |  | ||||||
| 		if data.UserID == nil { | 		if data.UserID == nil { | ||||||
| @@ -921,7 +950,7 @@ func (h CompatHandler) Upgrade(g *gin.Context) ginresp.HTTPResponse { | |||||||
| 			return ginresp.CompatAPIError(0, "Failed to query user") | 			return ginresp.CompatAPIError(0, "Failed to query user") | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 	return ctx.FinishSuccess(ginresp.JSON(http.StatusOK, response{ | 		return finishSuccess(ginext.JSON(http.StatusOK, response{ | ||||||
| 			Success:   true, | 			Success:   true, | ||||||
| 			Message:   "user updated", | 			Message:   "user updated", | ||||||
| 			UserID:    *data.UserID, | 			UserID:    *data.UserID, | ||||||
| @@ -929,4 +958,6 @@ func (h CompatHandler) Upgrade(g *gin.Context) ginresp.HTTPResponse { | |||||||
| 			QuotaMax:  user.QuotaPerDay(), | 			QuotaMax:  user.QuotaPerDay(), | ||||||
| 			IsPro:     user.IsPro, | 			IsPro:     user.IsPro, | ||||||
| 		})) | 		})) | ||||||
|  |  | ||||||
|  | 	}) | ||||||
| } | } | ||||||
|   | |||||||
| @@ -7,7 +7,7 @@ import ( | |||||||
| 	"blackforestbytes.com/simplecloudnotifier/logic" | 	"blackforestbytes.com/simplecloudnotifier/logic" | ||||||
| 	"blackforestbytes.com/simplecloudnotifier/models" | 	"blackforestbytes.com/simplecloudnotifier/models" | ||||||
| 	"fmt" | 	"fmt" | ||||||
| 	"github.com/gin-gonic/gin" | 	"gogs.mikescher.com/BlackForestBytes/goext/ginext" | ||||||
| 	"gogs.mikescher.com/BlackForestBytes/goext/langext" | 	"gogs.mikescher.com/BlackForestBytes/goext/langext" | ||||||
| 	"net/http" | 	"net/http" | ||||||
| 	"time" | 	"time" | ||||||
| @@ -41,7 +41,7 @@ func NewExternalHandler(app *logic.Application) ExternalHandler { | |||||||
| //	@Failure		500			{object}	ginresp.apiError	"An internal server error occurred - try again later" | //	@Failure		500			{object}	ginresp.apiError	"An internal server error occurred - try again later" | ||||||
| // | // | ||||||
| //	@Router			/external/v1/uptime-kuma [POST] | //	@Router			/external/v1/uptime-kuma [POST] | ||||||
| func (h ExternalHandler) UptimeKuma(g *gin.Context) ginresp.HTTPResponse { | func (h ExternalHandler) UptimeKuma(pctx ginext.PreContext) ginext.HTTPResponse { | ||||||
| 	type query struct { | 	type query struct { | ||||||
| 		UserID       *models.UserID `form:"user_id"     example:"7725"` | 		UserID       *models.UserID `form:"user_id"     example:"7725"` | ||||||
| 		KeyToken     *string        `form:"key"         example:"P3TNH8mvv14fm"` | 		KeyToken     *string        `form:"key"         example:"P3TNH8mvv14fm"` | ||||||
| @@ -74,12 +74,14 @@ func (h ExternalHandler) UptimeKuma(g *gin.Context) ginresp.HTTPResponse { | |||||||
|  |  | ||||||
| 	var b body | 	var b body | ||||||
| 	var q query | 	var q query | ||||||
| 	ctx, httpErr := h.app.StartRequest(g, nil, &q, &b, nil) | 	ctx, g, errResp := pctx.Query(&q).Body(&b).Start() | ||||||
| 	if httpErr != nil { | 	if errResp != nil { | ||||||
| 		return *httpErr | 		return *errResp | ||||||
| 	} | 	} | ||||||
| 	defer ctx.Cancel() | 	defer ctx.Cancel() | ||||||
|  |  | ||||||
|  | 	return h.app.DoRequest(ctx, g, models.TLockReadWrite, func(ctx *logic.AppContext, finishSuccess func(r ginext.HTTPResponse) ginext.HTTPResponse) ginext.HTTPResponse { | ||||||
|  |  | ||||||
| 		if b.Heartbeat == nil { | 		if b.Heartbeat == nil { | ||||||
| 			return ginresp.APIError(g, 400, apierr.BINDFAIL_BODY_PARAM, "missing field 'heartbeat' in request body", nil) | 			return ginresp.APIError(g, 400, apierr.BINDFAIL_BODY_PARAM, "missing field 'heartbeat' in request body", nil) | ||||||
| 		} | 		} | ||||||
| @@ -128,7 +130,9 @@ func (h ExternalHandler) UptimeKuma(g *gin.Context) ginresp.HTTPResponse { | |||||||
| 			return *errResp | 			return *errResp | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 	return ctx.FinishSuccess(ginresp.JSON(http.StatusOK, response{ | 		return finishSuccess(ginext.JSON(http.StatusOK, response{ | ||||||
| 			MessageID: okResp.Message.MessageID, | 			MessageID: okResp.Message.MessageID, | ||||||
| 		})) | 		})) | ||||||
|  |  | ||||||
|  | 	}) | ||||||
| } | } | ||||||
|   | |||||||
| @@ -2,12 +2,11 @@ package handler | |||||||
|  |  | ||||||
| import ( | import ( | ||||||
| 	"blackforestbytes.com/simplecloudnotifier/api/apierr" | 	"blackforestbytes.com/simplecloudnotifier/api/apierr" | ||||||
| 	"blackforestbytes.com/simplecloudnotifier/api/ginresp" |  | ||||||
| 	primarydb "blackforestbytes.com/simplecloudnotifier/db/impl/primary" | 	primarydb "blackforestbytes.com/simplecloudnotifier/db/impl/primary" | ||||||
| 	"blackforestbytes.com/simplecloudnotifier/logic" | 	"blackforestbytes.com/simplecloudnotifier/logic" | ||||||
| 	"blackforestbytes.com/simplecloudnotifier/models" | 	"blackforestbytes.com/simplecloudnotifier/models" | ||||||
| 	"github.com/gin-gonic/gin" |  | ||||||
| 	"gogs.mikescher.com/BlackForestBytes/goext/dataext" | 	"gogs.mikescher.com/BlackForestBytes/goext/dataext" | ||||||
|  | 	"gogs.mikescher.com/BlackForestBytes/goext/ginext" | ||||||
| 	"gogs.mikescher.com/BlackForestBytes/goext/langext" | 	"gogs.mikescher.com/BlackForestBytes/goext/langext" | ||||||
| 	"net/http" | 	"net/http" | ||||||
| ) | ) | ||||||
| @@ -49,7 +48,7 @@ func NewMessageHandler(app *logic.Application) MessageHandler { | |||||||
| // | // | ||||||
| //	@Router			/     [POST] | //	@Router			/     [POST] | ||||||
| //	@Router			/send [POST] | //	@Router			/send [POST] | ||||||
| func (h MessageHandler) SendMessage(g *gin.Context) ginresp.HTTPResponse { | func (h MessageHandler) SendMessage(pctx ginext.PreContext) ginext.HTTPResponse { | ||||||
| 	type combined struct { | 	type combined struct { | ||||||
| 		UserID        *models.UserID `json:"user_id"     form:"user_id"     example:"7725"                               ` | 		UserID        *models.UserID `json:"user_id"     form:"user_id"     example:"7725"                               ` | ||||||
| 		KeyToken      *string        `json:"key"         form:"key"         example:"P3TNH8mvv14fm"                      ` | 		KeyToken      *string        `json:"key"         form:"key"         example:"P3TNH8mvv14fm"                      ` | ||||||
| @@ -78,12 +77,14 @@ func (h MessageHandler) SendMessage(g *gin.Context) ginresp.HTTPResponse { | |||||||
| 	var b combined | 	var b combined | ||||||
| 	var q combined | 	var q combined | ||||||
| 	var f combined | 	var f combined | ||||||
| 	ctx, errResp := h.app.StartRequest(g, nil, &q, &b, &f, logic.RequestOptions{IgnoreWrongContentType: true}) | 	ctx, g, errResp := pctx.Form(&f).Query(&q).Body(&b).IgnoreWrongContentType().Start() | ||||||
| 	if errResp != nil { | 	if errResp != nil { | ||||||
| 		return *errResp | 		return *errResp | ||||||
| 	} | 	} | ||||||
| 	defer ctx.Cancel() | 	defer ctx.Cancel() | ||||||
|  |  | ||||||
|  | 	return h.app.DoRequest(ctx, g, models.TLockReadWrite, func(ctx *logic.AppContext, finishSuccess func(r ginext.HTTPResponse) ginext.HTTPResponse) ginext.HTTPResponse { | ||||||
|  |  | ||||||
| 		// query has highest prio, then form, then json | 		// query has highest prio, then form, then json | ||||||
| 		data := dataext.ObjectMerge(dataext.ObjectMerge(b, f), q) | 		data := dataext.ObjectMerge(dataext.ObjectMerge(b, f), q) | ||||||
|  |  | ||||||
| @@ -91,7 +92,7 @@ func (h MessageHandler) SendMessage(g *gin.Context) ginresp.HTTPResponse { | |||||||
| 		if errResp != nil { | 		if errResp != nil { | ||||||
| 			return *errResp | 			return *errResp | ||||||
| 		} else { | 		} else { | ||||||
| 		return ctx.FinishSuccess(ginresp.JSON(http.StatusOK, response{ | 			return finishSuccess(ginext.JSON(http.StatusOK, response{ | ||||||
| 				Success:        true, | 				Success:        true, | ||||||
| 				ErrorID:        apierr.NO_ERROR, | 				ErrorID:        apierr.NO_ERROR, | ||||||
| 				ErrorHighlight: -1, | 				ErrorHighlight: -1, | ||||||
| @@ -104,4 +105,6 @@ func (h MessageHandler) SendMessage(g *gin.Context) ginresp.HTTPResponse { | |||||||
| 				SCNMessageID:   okResp.Message.MessageID, | 				SCNMessageID:   okResp.Message.MessageID, | ||||||
| 			})) | 			})) | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
|  | 	}) | ||||||
| } | } | ||||||
|   | |||||||
| @@ -3,10 +3,12 @@ package handler | |||||||
| import ( | import ( | ||||||
| 	"blackforestbytes.com/simplecloudnotifier/api/ginresp" | 	"blackforestbytes.com/simplecloudnotifier/api/ginresp" | ||||||
| 	"blackforestbytes.com/simplecloudnotifier/logic" | 	"blackforestbytes.com/simplecloudnotifier/logic" | ||||||
|  | 	"blackforestbytes.com/simplecloudnotifier/models" | ||||||
| 	"blackforestbytes.com/simplecloudnotifier/website" | 	"blackforestbytes.com/simplecloudnotifier/website" | ||||||
| 	"errors" | 	"errors" | ||||||
| 	"github.com/gin-gonic/gin" | 	"github.com/gin-gonic/gin" | ||||||
| 	"github.com/rs/zerolog/log" | 	"github.com/rs/zerolog/log" | ||||||
|  | 	"gogs.mikescher.com/BlackForestBytes/goext/ginext" | ||||||
| 	"gogs.mikescher.com/BlackForestBytes/goext/rext" | 	"gogs.mikescher.com/BlackForestBytes/goext/rext" | ||||||
| 	"net/http" | 	"net/http" | ||||||
| 	"regexp" | 	"regexp" | ||||||
| @@ -27,60 +29,121 @@ func NewWebsiteHandler(app *logic.Application) WebsiteHandler { | |||||||
| 	} | 	} | ||||||
| } | } | ||||||
|  |  | ||||||
| func (h WebsiteHandler) Index(g *gin.Context) ginresp.HTTPResponse { | func (h WebsiteHandler) Index(pctx ginext.PreContext) ginext.HTTPResponse { | ||||||
|  | 	ctx, g, errResp := pctx.Start() | ||||||
|  | 	if errResp != nil { | ||||||
|  | 		return *errResp | ||||||
|  | 	} | ||||||
|  | 	defer ctx.Cancel() | ||||||
|  |  | ||||||
|  | 	return h.app.DoRequest(ctx, g, models.TLockNone, func(ctx *logic.AppContext, finishSuccess func(r ginext.HTTPResponse) ginext.HTTPResponse) ginext.HTTPResponse { | ||||||
| 		return h.serveAsset(g, "index.html", true) | 		return h.serveAsset(g, "index.html", true) | ||||||
|  | 	}) | ||||||
| } | } | ||||||
|  |  | ||||||
| func (h WebsiteHandler) APIDocs(g *gin.Context) ginresp.HTTPResponse { | func (h WebsiteHandler) APIDocs(pctx ginext.PreContext) ginext.HTTPResponse { | ||||||
|  | 	ctx, g, errResp := pctx.Start() | ||||||
|  | 	if errResp != nil { | ||||||
|  | 		return *errResp | ||||||
|  | 	} | ||||||
|  | 	defer ctx.Cancel() | ||||||
|  |  | ||||||
|  | 	return h.app.DoRequest(ctx, g, models.TLockNone, func(ctx *logic.AppContext, finishSuccess func(r ginext.HTTPResponse) ginext.HTTPResponse) ginext.HTTPResponse { | ||||||
| 		return h.serveAsset(g, "api.html", true) | 		return h.serveAsset(g, "api.html", true) | ||||||
|  | 	}) | ||||||
| } | } | ||||||
|  |  | ||||||
| func (h WebsiteHandler) APIDocsMore(g *gin.Context) ginresp.HTTPResponse { | func (h WebsiteHandler) APIDocsMore(pctx ginext.PreContext) ginext.HTTPResponse { | ||||||
|  | 	ctx, g, errResp := pctx.Start() | ||||||
|  | 	if errResp != nil { | ||||||
|  | 		return *errResp | ||||||
|  | 	} | ||||||
|  | 	defer ctx.Cancel() | ||||||
|  |  | ||||||
|  | 	return h.app.DoRequest(ctx, g, models.TLockNone, func(ctx *logic.AppContext, finishSuccess func(r ginext.HTTPResponse) ginext.HTTPResponse) ginext.HTTPResponse { | ||||||
| 		return h.serveAsset(g, "api_more.html", true) | 		return h.serveAsset(g, "api_more.html", true) | ||||||
|  | 	}) | ||||||
| } | } | ||||||
|  |  | ||||||
| func (h WebsiteHandler) MessageSent(g *gin.Context) ginresp.HTTPResponse { | func (h WebsiteHandler) MessageSent(pctx ginext.PreContext) ginext.HTTPResponse { | ||||||
|  | 	ctx, g, errResp := pctx.Start() | ||||||
|  | 	if errResp != nil { | ||||||
|  | 		return *errResp | ||||||
|  | 	} | ||||||
|  | 	defer ctx.Cancel() | ||||||
|  |  | ||||||
|  | 	return h.app.DoRequest(ctx, g, models.TLockNone, func(ctx *logic.AppContext, finishSuccess func(r ginext.HTTPResponse) ginext.HTTPResponse) ginext.HTTPResponse { | ||||||
| 		return h.serveAsset(g, "message_sent.html", true) | 		return h.serveAsset(g, "message_sent.html", true) | ||||||
|  | 	}) | ||||||
| } | } | ||||||
|  |  | ||||||
| func (h WebsiteHandler) FaviconIco(g *gin.Context) ginresp.HTTPResponse { | func (h WebsiteHandler) FaviconIco(pctx ginext.PreContext) ginext.HTTPResponse { | ||||||
|  | 	ctx, g, errResp := pctx.Start() | ||||||
|  | 	if errResp != nil { | ||||||
|  | 		return *errResp | ||||||
|  | 	} | ||||||
|  | 	defer ctx.Cancel() | ||||||
|  |  | ||||||
|  | 	return h.app.DoRequest(ctx, g, models.TLockNone, func(ctx *logic.AppContext, finishSuccess func(r ginext.HTTPResponse) ginext.HTTPResponse) ginext.HTTPResponse { | ||||||
| 		return h.serveAsset(g, "favicon.ico", false) | 		return h.serveAsset(g, "favicon.ico", false) | ||||||
|  | 	}) | ||||||
| } | } | ||||||
|  |  | ||||||
| func (h WebsiteHandler) FaviconPNG(g *gin.Context) ginresp.HTTPResponse { | func (h WebsiteHandler) FaviconPNG(pctx ginext.PreContext) ginext.HTTPResponse { | ||||||
|  | 	ctx, g, errResp := pctx.Start() | ||||||
|  | 	if errResp != nil { | ||||||
|  | 		return *errResp | ||||||
|  | 	} | ||||||
|  | 	defer ctx.Cancel() | ||||||
|  |  | ||||||
|  | 	return h.app.DoRequest(ctx, g, models.TLockNone, func(ctx *logic.AppContext, finishSuccess func(r ginext.HTTPResponse) ginext.HTTPResponse) ginext.HTTPResponse { | ||||||
| 		return h.serveAsset(g, "favicon.png", false) | 		return h.serveAsset(g, "favicon.png", false) | ||||||
|  | 	}) | ||||||
| } | } | ||||||
|  |  | ||||||
| func (h WebsiteHandler) Javascript(g *gin.Context) ginresp.HTTPResponse { | func (h WebsiteHandler) Javascript(pctx ginext.PreContext) ginext.HTTPResponse { | ||||||
|  | 	ctx, g, errResp := pctx.Start() | ||||||
|  | 	if errResp != nil { | ||||||
|  | 		return *errResp | ||||||
|  | 	} | ||||||
|  | 	defer ctx.Cancel() | ||||||
|  |  | ||||||
|  | 	return h.app.DoRequest(ctx, g, models.TLockNone, func(ctx *logic.AppContext, finishSuccess func(r ginext.HTTPResponse) ginext.HTTPResponse) ginext.HTTPResponse { | ||||||
|  |  | ||||||
| 		type uri struct { | 		type uri struct { | ||||||
| 			Filename string `uri:"fn"` | 			Filename string `uri:"fn"` | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		var u uri | 		var u uri | ||||||
| 		if err := g.ShouldBindUri(&u); err != nil { | 		if err := g.ShouldBindUri(&u); err != nil { | ||||||
| 		return ginresp.JSON(http.StatusBadRequest, gin.H{"error": err.Error()}) | 			return ginext.JSON(http.StatusBadRequest, gin.H{"error": err.Error()}) | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		return h.serveAsset(g, "js/"+u.Filename, false) | 		return h.serveAsset(g, "js/"+u.Filename, false) | ||||||
|  | 	}) | ||||||
| } | } | ||||||
|  |  | ||||||
| func (h WebsiteHandler) CSS(g *gin.Context) ginresp.HTTPResponse { | func (h WebsiteHandler) CSS(pctx ginext.PreContext) ginext.HTTPResponse { | ||||||
| 	type uri struct { | 	type uri struct { | ||||||
| 		Filename string `uri:"fn"` | 		Filename string `uri:"fn"` | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	var u uri | 	var u uri | ||||||
| 	if err := g.ShouldBindUri(&u); err != nil { | 	ctx, g, errResp := pctx.URI(&u).Start() | ||||||
| 		return ginresp.JSON(http.StatusBadRequest, gin.H{"error": err.Error()}) | 	if errResp != nil { | ||||||
|  | 		return *errResp | ||||||
| 	} | 	} | ||||||
|  | 	defer ctx.Cancel() | ||||||
|  |  | ||||||
|  | 	return h.app.DoRequest(ctx, g, models.TLockNone, func(ctx *logic.AppContext, finishSuccess func(r ginext.HTTPResponse) ginext.HTTPResponse) ginext.HTTPResponse { | ||||||
| 		return h.serveAsset(g, "css/"+u.Filename, false) | 		return h.serveAsset(g, "css/"+u.Filename, false) | ||||||
|  | 	}) | ||||||
| } | } | ||||||
|  |  | ||||||
| func (h WebsiteHandler) serveAsset(g *gin.Context, fn string, repl bool) ginresp.HTTPResponse { | func (h WebsiteHandler) serveAsset(g *gin.Context, fn string, repl bool) ginext.HTTPResponse { | ||||||
| 	_data, err := website.Assets.ReadFile(fn) | 	_data, err := website.Assets.ReadFile(fn) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return ginresp.Status(http.StatusNotFound) | 		return ginext.Status(http.StatusNotFound) | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	data := string(_data) | 	data := string(_data) | ||||||
| @@ -141,7 +204,7 @@ func (h WebsiteHandler) serveAsset(g *gin.Context, fn string, repl bool) ginresp | |||||||
| 		mime = "image/svg+xml" | 		mime = "image/svg+xml" | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	return ginresp.Data(http.StatusOK, mime, []byte(data)) | 	return ginext.Data(http.StatusOK, mime, []byte(data)) | ||||||
| } | } | ||||||
|  |  | ||||||
| func (h WebsiteHandler) getReplConfig(key string) (string, bool) { | func (h WebsiteHandler) getReplConfig(key string) (string, bool) { | ||||||
|   | |||||||
| @@ -1,16 +1,14 @@ | |||||||
| package api | package api | ||||||
|  |  | ||||||
| import ( | import ( | ||||||
| 	"blackforestbytes.com/simplecloudnotifier/api/ginext" |  | ||||||
| 	"blackforestbytes.com/simplecloudnotifier/api/ginresp" |  | ||||||
| 	"blackforestbytes.com/simplecloudnotifier/api/handler" | 	"blackforestbytes.com/simplecloudnotifier/api/handler" | ||||||
| 	"blackforestbytes.com/simplecloudnotifier/logic" | 	"blackforestbytes.com/simplecloudnotifier/logic" | ||||||
| 	"blackforestbytes.com/simplecloudnotifier/models" | 	"blackforestbytes.com/simplecloudnotifier/models" | ||||||
| 	"blackforestbytes.com/simplecloudnotifier/swagger" | 	"blackforestbytes.com/simplecloudnotifier/swagger" | ||||||
| 	"errors" | 	"errors" | ||||||
| 	"github.com/gin-gonic/gin" |  | ||||||
| 	"github.com/gin-gonic/gin/binding" | 	"github.com/gin-gonic/gin/binding" | ||||||
| 	"github.com/go-playground/validator/v10" | 	"github.com/go-playground/validator/v10" | ||||||
|  | 	"gogs.mikescher.com/BlackForestBytes/goext/ginext" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| type Router struct { | type Router struct { | ||||||
| @@ -50,7 +48,7 @@ func NewRouter(app *logic.Application) *Router { | |||||||
| //	@tag.name		Common | //	@tag.name		Common | ||||||
| // | // | ||||||
| //	@BasePath		/ | //	@BasePath		/ | ||||||
| func (r *Router) Init(e *gin.Engine) error { | func (r *Router) Init(e *ginext.GinWrapper) error { | ||||||
|  |  | ||||||
| 	if v, ok := binding.Validator.Engine().(*validator.Validate); ok { | 	if v, ok := binding.Validator.Engine().(*validator.Validate); ok { | ||||||
| 		err := v.RegisterValidation("entityid", models.ValidateEntityID, true) | 		err := v.RegisterValidation("entityid", models.ValidateEntityID, true) | ||||||
| @@ -63,129 +61,125 @@ func (r *Router) Init(e *gin.Engine) error { | |||||||
|  |  | ||||||
| 	// ================ General (unversioned) ================ | 	// ================ General (unversioned) ================ | ||||||
|  |  | ||||||
| 	commonAPI := e.Group("/api") | 	commonAPI := e.Routes().Group("/api") | ||||||
| 	{ | 	{ | ||||||
| 		commonAPI.Any("/ping", r.Wrap(r.commonHandler.Ping)) | 		commonAPI.Any("/ping").Handle(r.commonHandler.Ping) | ||||||
| 		commonAPI.POST("/db-test", r.Wrap(r.commonHandler.DatabaseTest)) | 		commonAPI.POST("/db-test").Handle(r.commonHandler.DatabaseTest) | ||||||
| 		commonAPI.GET("/health", r.Wrap(r.commonHandler.Health)) | 		commonAPI.GET("/health").Handle(r.commonHandler.Health) | ||||||
| 		commonAPI.POST("/sleep/:secs", r.Wrap(r.commonHandler.Sleep)) | 		commonAPI.POST("/sleep/:secs").Handle(r.commonHandler.Sleep) | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	// ================ Swagger ================ | 	// ================ Swagger ================ | ||||||
|  |  | ||||||
| 	docs := e.Group("/documentation") | 	docs := e.Routes().Group("/documentation") | ||||||
| 	{ | 	{ | ||||||
| 		docs.GET("/swagger", ginext.RedirectTemporary("/documentation/swagger/")) | 		docs.GET("/swagger").Handle(ginext.RedirectTemporary("/documentation/swagger/")) | ||||||
| 		docs.GET("/swagger/*sub", r.Wrap(swagger.Handle)) | 		docs.GET("/swagger/*sub").Handle(swagger.Handle) | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	// ================ Website ================ | 	// ================ Website ================ | ||||||
|  |  | ||||||
| 	frontend := e.Group("") | 	frontend := e.Routes().Group("") | ||||||
| 	{ | 	{ | ||||||
| 		frontend.GET("/", r.Wrap(r.websiteHandler.Index)) | 		frontend.GET("/").Handle(r.websiteHandler.Index) | ||||||
| 		frontend.GET("/index.php", r.Wrap(r.websiteHandler.Index)) | 		frontend.GET("/index.php").Handle(r.websiteHandler.Index) | ||||||
| 		frontend.GET("/index.html", r.Wrap(r.websiteHandler.Index)) | 		frontend.GET("/index.html").Handle(r.websiteHandler.Index) | ||||||
| 		frontend.GET("/index", r.Wrap(r.websiteHandler.Index)) | 		frontend.GET("/index").Handle(r.websiteHandler.Index) | ||||||
|  |  | ||||||
| 		frontend.GET("/api", r.Wrap(r.websiteHandler.APIDocs)) | 		frontend.GET("/api").Handle(r.websiteHandler.APIDocs) | ||||||
| 		frontend.GET("/api.php", r.Wrap(r.websiteHandler.APIDocs)) | 		frontend.GET("/api.php").Handle(r.websiteHandler.APIDocs) | ||||||
| 		frontend.GET("/api.html", r.Wrap(r.websiteHandler.APIDocs)) | 		frontend.GET("/api.html").Handle(r.websiteHandler.APIDocs) | ||||||
|  |  | ||||||
| 		frontend.GET("/api_more", r.Wrap(r.websiteHandler.APIDocsMore)) | 		frontend.GET("/api_more").Handle(r.websiteHandler.APIDocsMore) | ||||||
| 		frontend.GET("/api_more.php", r.Wrap(r.websiteHandler.APIDocsMore)) | 		frontend.GET("/api_more.php").Handle(r.websiteHandler.APIDocsMore) | ||||||
| 		frontend.GET("/api_more.html", r.Wrap(r.websiteHandler.APIDocsMore)) | 		frontend.GET("/api_more.html").Handle(r.websiteHandler.APIDocsMore) | ||||||
|  |  | ||||||
| 		frontend.GET("/message_sent", r.Wrap(r.websiteHandler.MessageSent)) | 		frontend.GET("/message_sent").Handle(r.websiteHandler.MessageSent) | ||||||
| 		frontend.GET("/message_sent.php", r.Wrap(r.websiteHandler.MessageSent)) | 		frontend.GET("/message_sent.php").Handle(r.websiteHandler.MessageSent) | ||||||
| 		frontend.GET("/message_sent.html", r.Wrap(r.websiteHandler.MessageSent)) | 		frontend.GET("/message_sent.html").Handle(r.websiteHandler.MessageSent) | ||||||
|  |  | ||||||
| 		frontend.GET("/favicon.ico", r.Wrap(r.websiteHandler.FaviconIco)) | 		frontend.GET("/favicon.ico").Handle(r.websiteHandler.FaviconIco) | ||||||
| 		frontend.GET("/favicon.png", r.Wrap(r.websiteHandler.FaviconPNG)) | 		frontend.GET("/favicon.png").Handle(r.websiteHandler.FaviconPNG) | ||||||
|  |  | ||||||
| 		frontend.GET("/js/:fn", r.Wrap(r.websiteHandler.Javascript)) | 		frontend.GET("/js/:fn").Handle(r.websiteHandler.Javascript) | ||||||
| 		frontend.GET("/css/:fn", r.Wrap(r.websiteHandler.CSS)) | 		frontend.GET("/css/:fn").Handle(r.websiteHandler.CSS) | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	// ================ Compat (v1) ================ | 	// ================ Compat (v1) ================ | ||||||
|  |  | ||||||
| 	compat := e.Group("/api") | 	compat := e.Routes().Group("/api") | ||||||
| 	{ | 	{ | ||||||
| 		compat.GET("/register.php", r.Wrap(r.compatHandler.Register)) | 		compat.GET("/register.php").Handle(r.compatHandler.Register) | ||||||
| 		compat.GET("/info.php", r.Wrap(r.compatHandler.Info)) | 		compat.GET("/info.php").Handle(r.compatHandler.Info) | ||||||
| 		compat.GET("/ack.php", r.Wrap(r.compatHandler.Ack)) | 		compat.GET("/ack.php").Handle(r.compatHandler.Ack) | ||||||
| 		compat.GET("/requery.php", r.Wrap(r.compatHandler.Requery)) | 		compat.GET("/requery.php").Handle(r.compatHandler.Requery) | ||||||
| 		compat.GET("/update.php", r.Wrap(r.compatHandler.Update)) | 		compat.GET("/update.php").Handle(r.compatHandler.Update) | ||||||
| 		compat.GET("/expand.php", r.Wrap(r.compatHandler.Expand)) | 		compat.GET("/expand.php").Handle(r.compatHandler.Expand) | ||||||
| 		compat.GET("/upgrade.php", r.Wrap(r.compatHandler.Upgrade)) | 		compat.GET("/upgrade.php").Handle(r.compatHandler.Upgrade) | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	// ================ Manage API (v2) ================ | 	// ================ Manage API (v2) ================ | ||||||
|  |  | ||||||
| 	apiv2 := e.Group("/api/v2/") | 	apiv2 := e.Routes().Group("/api/v2/") | ||||||
| 	{ | 	{ | ||||||
| 		apiv2.POST("/users", r.Wrap(r.apiHandler.CreateUser)) | 		apiv2.POST("/users").Handle(r.apiHandler.CreateUser) | ||||||
| 		apiv2.GET("/users/:uid", r.Wrap(r.apiHandler.GetUser)) | 		apiv2.GET("/users/:uid").Handle(r.apiHandler.GetUser) | ||||||
| 		apiv2.PATCH("/users/:uid", r.Wrap(r.apiHandler.UpdateUser)) | 		apiv2.PATCH("/users/:uid").Handle(r.apiHandler.UpdateUser) | ||||||
|  |  | ||||||
| 		apiv2.GET("/users/:uid/keys", r.Wrap(r.apiHandler.ListUserKeys)) | 		apiv2.GET("/users/:uid/keys").Handle(r.apiHandler.ListUserKeys) | ||||||
| 		apiv2.POST("/users/:uid/keys", r.Wrap(r.apiHandler.CreateUserKey)) | 		apiv2.POST("/users/:uid/keys").Handle(r.apiHandler.CreateUserKey) | ||||||
| 		apiv2.GET("/users/:uid/keys/current", r.Wrap(r.apiHandler.GetCurrentUserKey)) | 		apiv2.GET("/users/:uid/keys/current").Handle(r.apiHandler.GetCurrentUserKey) | ||||||
| 		apiv2.GET("/users/:uid/keys/:kid", r.Wrap(r.apiHandler.GetUserKey)) | 		apiv2.GET("/users/:uid/keys/:kid").Handle(r.apiHandler.GetUserKey) | ||||||
| 		apiv2.PATCH("/users/:uid/keys/:kid", r.Wrap(r.apiHandler.UpdateUserKey)) | 		apiv2.PATCH("/users/:uid/keys/:kid").Handle(r.apiHandler.UpdateUserKey) | ||||||
| 		apiv2.DELETE("/users/:uid/keys/:kid", r.Wrap(r.apiHandler.DeleteUserKey)) | 		apiv2.DELETE("/users/:uid/keys/:kid").Handle(r.apiHandler.DeleteUserKey) | ||||||
|  |  | ||||||
| 		apiv2.GET("/users/:uid/clients", r.Wrap(r.apiHandler.ListClients)) | 		apiv2.GET("/users/:uid/clients").Handle(r.apiHandler.ListClients) | ||||||
| 		apiv2.GET("/users/:uid/clients/:cid", r.Wrap(r.apiHandler.GetClient)) | 		apiv2.GET("/users/:uid/clients/:cid").Handle(r.apiHandler.GetClient) | ||||||
| 		apiv2.PATCH("/users/:uid/clients/:cid", r.Wrap(r.apiHandler.UpdateClient)) | 		apiv2.PATCH("/users/:uid/clients/:cid").Handle(r.apiHandler.UpdateClient) | ||||||
| 		apiv2.POST("/users/:uid/clients", r.Wrap(r.apiHandler.AddClient)) | 		apiv2.POST("/users/:uid/clients").Handle(r.apiHandler.AddClient) | ||||||
| 		apiv2.DELETE("/users/:uid/clients/:cid", r.Wrap(r.apiHandler.DeleteClient)) | 		apiv2.DELETE("/users/:uid/clients/:cid").Handle(r.apiHandler.DeleteClient) | ||||||
|  |  | ||||||
| 		apiv2.GET("/users/:uid/channels", r.Wrap(r.apiHandler.ListChannels)) | 		apiv2.GET("/users/:uid/channels").Handle(r.apiHandler.ListChannels) | ||||||
| 		apiv2.POST("/users/:uid/channels", r.Wrap(r.apiHandler.CreateChannel)) | 		apiv2.POST("/users/:uid/channels").Handle(r.apiHandler.CreateChannel) | ||||||
| 		apiv2.GET("/users/:uid/channels/:cid", r.Wrap(r.apiHandler.GetChannel)) | 		apiv2.GET("/users/:uid/channels/:cid").Handle(r.apiHandler.GetChannel) | ||||||
| 		apiv2.PATCH("/users/:uid/channels/:cid", r.Wrap(r.apiHandler.UpdateChannel)) | 		apiv2.PATCH("/users/:uid/channels/:cid").Handle(r.apiHandler.UpdateChannel) | ||||||
| 		apiv2.GET("/users/:uid/channels/:cid/messages", r.Wrap(r.apiHandler.ListChannelMessages)) | 		apiv2.GET("/users/:uid/channels/:cid/messages").Handle(r.apiHandler.ListChannelMessages) | ||||||
| 		apiv2.GET("/users/:uid/channels/:cid/subscriptions", r.Wrap(r.apiHandler.ListChannelSubscriptions)) | 		apiv2.GET("/users/:uid/channels/:cid/subscriptions").Handle(r.apiHandler.ListChannelSubscriptions) | ||||||
|  |  | ||||||
| 		apiv2.GET("/users/:uid/subscriptions", r.Wrap(r.apiHandler.ListUserSubscriptions)) | 		apiv2.GET("/users/:uid/subscriptions").Handle(r.apiHandler.ListUserSubscriptions) | ||||||
| 		apiv2.POST("/users/:uid/subscriptions", r.Wrap(r.apiHandler.CreateSubscription)) | 		apiv2.POST("/users/:uid/subscriptions").Handle(r.apiHandler.CreateSubscription) | ||||||
| 		apiv2.GET("/users/:uid/subscriptions/:sid", r.Wrap(r.apiHandler.GetSubscription)) | 		apiv2.GET("/users/:uid/subscriptions/:sid").Handle(r.apiHandler.GetSubscription) | ||||||
| 		apiv2.DELETE("/users/:uid/subscriptions/:sid", r.Wrap(r.apiHandler.CancelSubscription)) | 		apiv2.DELETE("/users/:uid/subscriptions/:sid").Handle(r.apiHandler.CancelSubscription) | ||||||
| 		apiv2.PATCH("/users/:uid/subscriptions/:sid", r.Wrap(r.apiHandler.UpdateSubscription)) | 		apiv2.PATCH("/users/:uid/subscriptions/:sid").Handle(r.apiHandler.UpdateSubscription) | ||||||
|  |  | ||||||
| 		apiv2.GET("/messages", r.Wrap(r.apiHandler.ListMessages)) | 		apiv2.GET("/messages").Handle(r.apiHandler.ListMessages) | ||||||
| 		apiv2.GET("/messages/:mid", r.Wrap(r.apiHandler.GetMessage)) | 		apiv2.GET("/messages/:mid").Handle(r.apiHandler.GetMessage) | ||||||
| 		apiv2.DELETE("/messages/:mid", r.Wrap(r.apiHandler.DeleteMessage)) | 		apiv2.DELETE("/messages/:mid").Handle(r.apiHandler.DeleteMessage) | ||||||
|  |  | ||||||
| 		apiv2.GET("/preview/users/:uid", r.Wrap(r.apiHandler.GetUserPreview)) | 		apiv2.GET("/preview/users/:uid").Handle(r.apiHandler.GetUserPreview) | ||||||
| 		apiv2.GET("/preview/keys/:kid", r.Wrap(r.apiHandler.GetUserKeyPreview)) | 		apiv2.GET("/preview/keys/:kid").Handle(r.apiHandler.GetUserKeyPreview) | ||||||
| 		apiv2.GET("/preview/channels/:cid", r.Wrap(r.apiHandler.GetChannelPreview)) | 		apiv2.GET("/preview/channels/:cid").Handle(r.apiHandler.GetChannelPreview) | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	// ================ Send API (unversioned) ================ | 	// ================ Send API (unversioned) ================ | ||||||
|  |  | ||||||
| 	sendAPI := e.Group("") | 	sendAPI := e.Routes().Group("") | ||||||
| 	{ | 	{ | ||||||
| 		sendAPI.POST("/", r.Wrap(r.messageHandler.SendMessage)) | 		sendAPI.POST("/").Handle(r.messageHandler.SendMessage) | ||||||
| 		sendAPI.POST("/send", r.Wrap(r.messageHandler.SendMessage)) | 		sendAPI.POST("/send").Handle(r.messageHandler.SendMessage) | ||||||
| 		sendAPI.POST("/send.php", r.Wrap(r.compatHandler.SendMessage)) | 		sendAPI.POST("/send.php").Handle(r.compatHandler.SendMessage) | ||||||
|  |  | ||||||
| 		sendAPI.POST("/external/v1/uptime-kuma", r.Wrap(r.externalHandler.UptimeKuma)) | 		sendAPI.POST("/external/v1/uptime-kuma").Handle(r.externalHandler.UptimeKuma) | ||||||
|  |  | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	// ================ | 	// ================ | ||||||
|  |  | ||||||
| 	if r.app.Config.ReturnRawErrors { | 	if r.app.Config.ReturnRawErrors { | ||||||
| 		e.NoRoute(r.Wrap(r.commonHandler.NoRoute)) | 		e.NoRoute(r.commonHandler.NoRoute) | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	// ================ | 	// ================ | ||||||
|  |  | ||||||
| 	return nil | 	return nil | ||||||
| } | } | ||||||
|  |  | ||||||
| func (r *Router) Wrap(fn ginresp.WHandlerFunc) gin.HandlerFunc { |  | ||||||
| 	return ginresp.Wrap(r.app, fn) |  | ||||||
| } |  | ||||||
|   | |||||||
| @@ -3,9 +3,11 @@ package main | |||||||
| import ( | import ( | ||||||
| 	"blackforestbytes.com/simplecloudnotifier/db/schema" | 	"blackforestbytes.com/simplecloudnotifier/db/schema" | ||||||
| 	"context" | 	"context" | ||||||
|  | 	"database/sql" | ||||||
| 	"fmt" | 	"fmt" | ||||||
| 	"github.com/mattn/go-sqlite3" | 	"github.com/glebarez/go-sqlite" | ||||||
| 	"gogs.mikescher.com/BlackForestBytes/goext/exerr" | 	"gogs.mikescher.com/BlackForestBytes/goext/exerr" | ||||||
|  | 	"gogs.mikescher.com/BlackForestBytes/goext/langext" | ||||||
| 	"gogs.mikescher.com/BlackForestBytes/goext/sq" | 	"gogs.mikescher.com/BlackForestBytes/goext/sq" | ||||||
| 	"time" | 	"time" | ||||||
| ) | ) | ||||||
| @@ -16,12 +18,14 @@ func main() { | |||||||
| 	ctx, cancel := context.WithTimeout(context.Background(), 10*time.Second) | 	ctx, cancel := context.WithTimeout(context.Background(), 10*time.Second) | ||||||
| 	defer cancel() | 	defer cancel() | ||||||
|  |  | ||||||
| 	sqlite3.Version() // ensure slite3 loaded | 	if !langext.InArray("sqlite3", sql.Drivers()) { | ||||||
|  | 		sqlite.RegisterAsSQLITE3() | ||||||
|  | 	} | ||||||
|  |  | ||||||
| 	fmt.Println() | 	fmt.Println() | ||||||
|  |  | ||||||
| 	for i := 2; i <= schema.PrimarySchemaVersion; i++ { | 	for i := 2; i <= schema.PrimarySchemaVersion; i++ { | ||||||
| 		h0, err := sq.HashMattnSqliteSchema(ctx, schema.PrimarySchema[i].SQL) | 		h0, err := sq.HashGoSqliteSchema(ctx, schema.PrimarySchema[i].SQL) | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| 			h0 = "ERR" | 			h0 = "ERR" | ||||||
| 		} | 		} | ||||||
| @@ -29,7 +33,7 @@ func main() { | |||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	for i := 1; i <= schema.RequestsSchemaVersion; i++ { | 	for i := 1; i <= schema.RequestsSchemaVersion; i++ { | ||||||
| 		h0, err := sq.HashMattnSqliteSchema(ctx, schema.RequestsSchema[i].SQL) | 		h0, err := sq.HashGoSqliteSchema(ctx, schema.RequestsSchema[i].SQL) | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| 			h0 = "ERR" | 			h0 = "ERR" | ||||||
| 		} | 		} | ||||||
| @@ -37,7 +41,7 @@ func main() { | |||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	for i := 1; i <= schema.LogsSchemaVersion; i++ { | 	for i := 1; i <= schema.LogsSchemaVersion; i++ { | ||||||
| 		h0, err := sq.HashMattnSqliteSchema(ctx, schema.LogsSchema[i].SQL) | 		h0, err := sq.HashGoSqliteSchema(ctx, schema.LogsSchema[i].SQL) | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| 			h0 = "ERR" | 			h0 = "ERR" | ||||||
| 		} | 		} | ||||||
|   | |||||||
| @@ -135,7 +135,7 @@ func main() { | |||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		panic(err) | 		panic(err) | ||||||
| 	} | 	} | ||||||
| 	dbold := sq.NewDB(_dbold) | 	dbold := sq.NewDB(_dbold, sq.DBOptions{}) | ||||||
|  |  | ||||||
| 	rowsUser, err := dbold.Query(ctx, "SELECT * FROM users", sq.PP{}) | 	rowsUser, err := dbold.Query(ctx, "SELECT * FROM users", sq.PP{}) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
|   | |||||||
| @@ -3,13 +3,15 @@ package main | |||||||
| import ( | import ( | ||||||
| 	scn "blackforestbytes.com/simplecloudnotifier" | 	scn "blackforestbytes.com/simplecloudnotifier" | ||||||
| 	"blackforestbytes.com/simplecloudnotifier/api" | 	"blackforestbytes.com/simplecloudnotifier/api" | ||||||
| 	"blackforestbytes.com/simplecloudnotifier/api/ginext" |  | ||||||
| 	"blackforestbytes.com/simplecloudnotifier/google" | 	"blackforestbytes.com/simplecloudnotifier/google" | ||||||
| 	"blackforestbytes.com/simplecloudnotifier/jobs" | 	"blackforestbytes.com/simplecloudnotifier/jobs" | ||||||
| 	"blackforestbytes.com/simplecloudnotifier/logic" | 	"blackforestbytes.com/simplecloudnotifier/logic" | ||||||
| 	"blackforestbytes.com/simplecloudnotifier/push" | 	"blackforestbytes.com/simplecloudnotifier/push" | ||||||
| 	"fmt" | 	"fmt" | ||||||
| 	"github.com/rs/zerolog/log" | 	"github.com/rs/zerolog/log" | ||||||
|  | 	"gogs.mikescher.com/BlackForestBytes/goext/ginext" | ||||||
|  | 	"gogs.mikescher.com/BlackForestBytes/goext/langext" | ||||||
|  | 	"time" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| func main() { | func main() { | ||||||
| @@ -31,7 +33,13 @@ func main() { | |||||||
| 		return | 		return | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	ginengine := ginext.NewEngine(conf) | 	ginengine := ginext.NewEngine(ginext.Options{ | ||||||
|  | 		AllowCors:             &conf.Cors, | ||||||
|  | 		GinDebug:              &conf.GinDebug, | ||||||
|  | 		BufferBody:            langext.PTrue, | ||||||
|  | 		Timeout:               langext.Ptr(time.Duration(int64(conf.RequestTimeout) * int64(conf.RequestMaxRetry))), | ||||||
|  | 		BuildRequestBindError: logic.BuildGinRequestError, | ||||||
|  | 	}) | ||||||
|  |  | ||||||
| 	router := api.NewRouter(app) | 	router := api.NewRouter(app) | ||||||
|  |  | ||||||
|   | |||||||
| @@ -5,12 +5,13 @@ import ( | |||||||
| 	"blackforestbytes.com/simplecloudnotifier/db/dbtools" | 	"blackforestbytes.com/simplecloudnotifier/db/dbtools" | ||||||
| 	"blackforestbytes.com/simplecloudnotifier/db/schema" | 	"blackforestbytes.com/simplecloudnotifier/db/schema" | ||||||
| 	"blackforestbytes.com/simplecloudnotifier/db/simplectx" | 	"blackforestbytes.com/simplecloudnotifier/db/simplectx" | ||||||
|  | 	"blackforestbytes.com/simplecloudnotifier/models" | ||||||
| 	"context" | 	"context" | ||||||
| 	"database/sql" | 	"database/sql" | ||||||
| 	"errors" | 	"errors" | ||||||
| 	"fmt" | 	"fmt" | ||||||
|  | 	"github.com/glebarez/go-sqlite" | ||||||
| 	"github.com/jmoiron/sqlx" | 	"github.com/jmoiron/sqlx" | ||||||
| 	_ "github.com/mattn/go-sqlite3" |  | ||||||
| 	"github.com/rs/zerolog/log" | 	"github.com/rs/zerolog/log" | ||||||
| 	"gogs.mikescher.com/BlackForestBytes/goext/langext" | 	"gogs.mikescher.com/BlackForestBytes/goext/langext" | ||||||
| 	"gogs.mikescher.com/BlackForestBytes/goext/sq" | 	"gogs.mikescher.com/BlackForestBytes/goext/sq" | ||||||
| @@ -26,7 +27,16 @@ type Database struct { | |||||||
| func NewLogsDatabase(cfg server.Config) (*Database, error) { | func NewLogsDatabase(cfg server.Config) (*Database, error) { | ||||||
| 	conf := cfg.DBLogs | 	conf := cfg.DBLogs | ||||||
|  |  | ||||||
| 	url := fmt.Sprintf("file:%s?_journal=%s&_timeout=%d&_fk=%s&_busy_timeout=%d", conf.File, conf.Journal, conf.Timeout.Milliseconds(), langext.FormatBool(conf.CheckForeignKeys, "true", "false"), conf.BusyTimeout.Milliseconds()) | 	url := fmt.Sprintf("file:%s?_pragma=journal_mode(%s)&_pragma=timeout(%d)&_pragma=foreign_keys(%s)&_pragma=busy_timeout(%d)", | ||||||
|  | 		conf.File, | ||||||
|  | 		conf.Journal, | ||||||
|  | 		conf.Timeout.Milliseconds(), | ||||||
|  | 		langext.FormatBool(conf.CheckForeignKeys, "true", "false"), | ||||||
|  | 		conf.BusyTimeout.Milliseconds()) | ||||||
|  |  | ||||||
|  | 	if !langext.InArray("sqlite3", sql.Drivers()) { | ||||||
|  | 		sqlite.RegisterAsSQLITE3() | ||||||
|  | 	} | ||||||
|  |  | ||||||
| 	xdb, err := sqlx.Open("sqlite3", url) | 	xdb, err := sqlx.Open("sqlite3", url) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| @@ -42,7 +52,8 @@ func NewLogsDatabase(cfg server.Config) (*Database, error) { | |||||||
| 		xdb.SetConnMaxIdleTime(60 * time.Minute) | 		xdb.SetConnMaxIdleTime(60 * time.Minute) | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	qqdb := sq.NewDB(xdb, sq.DBOptions{}) | 	qqdb := sq.NewDB(xdb, sq.DBOptions{RegisterDefaultConverter: langext.PTrue, RegisterCommentTrimmer: langext.PTrue}) | ||||||
|  | 	models.RegisterConverter(qqdb) | ||||||
|  |  | ||||||
| 	if conf.EnableLogger { | 	if conf.EnableLogger { | ||||||
| 		qqdb.AddListener(dbtools.DBLogger{}) | 		qqdb.AddListener(dbtools.DBLogger{}) | ||||||
|   | |||||||
| @@ -3,8 +3,6 @@ package primary | |||||||
| import ( | import ( | ||||||
| 	"blackforestbytes.com/simplecloudnotifier/db" | 	"blackforestbytes.com/simplecloudnotifier/db" | ||||||
| 	"blackforestbytes.com/simplecloudnotifier/models" | 	"blackforestbytes.com/simplecloudnotifier/models" | ||||||
| 	"database/sql" |  | ||||||
| 	"errors" |  | ||||||
| 	"gogs.mikescher.com/BlackForestBytes/goext/sq" | 	"gogs.mikescher.com/BlackForestBytes/goext/sq" | ||||||
| 	"time" | 	"time" | ||||||
| ) | ) | ||||||
| @@ -15,23 +13,7 @@ func (db *Database) GetChannelByName(ctx db.TxContext, userid models.UserID, cha | |||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	rows, err := tx.Query(ctx, "SELECT * FROM channels WHERE owner_user_id = :uid AND internal_name = :nam LIMIT 1", sq.PP{ | 	return sq.QuerySingleOpt[models.Channel](ctx, tx, "SELECT * FROM channels WHERE owner_user_id = :uid AND internal_name = :nam LIMIT 1", sq.PP{"uid": userid, "nam": chanName}, sq.SModeExtended, sq.Safe) | ||||||
| 		"uid": userid, |  | ||||||
| 		"nam": chanName, |  | ||||||
| 	}) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return nil, err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	channel, err := models.DecodeChannel(ctx, tx, rows) |  | ||||||
| 	if errors.Is(err, sql.ErrNoRows) { |  | ||||||
| 		return nil, nil |  | ||||||
| 	} |  | ||||||
| 	if err != nil { |  | ||||||
| 		return nil, err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	return &channel, nil |  | ||||||
| } | } | ||||||
|  |  | ||||||
| func (db *Database) GetChannelByID(ctx db.TxContext, chanid models.ChannelID) (*models.Channel, error) { | func (db *Database) GetChannelByID(ctx db.TxContext, chanid models.ChannelID) (*models.Channel, error) { | ||||||
| @@ -40,22 +22,7 @@ func (db *Database) GetChannelByID(ctx db.TxContext, chanid models.ChannelID) (* | |||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	rows, err := tx.Query(ctx, "SELECT * FROM channels WHERE channel_id = :cid LIMIT 1", sq.PP{ | 	return sq.QuerySingleOpt[models.Channel](ctx, tx, "SELECT * FROM channels WHERE channel_id = :cid LIMIT 1", sq.PP{"cid": chanid}, sq.SModeExtended, sq.Safe) | ||||||
| 		"cid": chanid, |  | ||||||
| 	}) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return nil, err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	channel, err := models.DecodeChannel(ctx, tx, rows) |  | ||||||
| 	if errors.Is(err, sql.ErrNoRows) { |  | ||||||
| 		return nil, nil |  | ||||||
| 	} |  | ||||||
| 	if err != nil { |  | ||||||
| 		return nil, err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	return &channel, nil |  | ||||||
| } | } | ||||||
|  |  | ||||||
| type CreateChanel struct { | type CreateChanel struct { | ||||||
| @@ -72,14 +39,14 @@ func (db *Database) CreateChannel(ctx db.TxContext, userid models.UserID, dispNa | |||||||
| 		return models.Channel{}, err | 		return models.Channel{}, err | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	entity := models.ChannelDB{ | 	entity := models.Channel{ | ||||||
| 		ChannelID:         models.NewChannelID(), | 		ChannelID:         models.NewChannelID(), | ||||||
| 		OwnerUserID:       userid, | 		OwnerUserID:       userid, | ||||||
| 		DisplayName:       dispName, | 		DisplayName:       dispName, | ||||||
| 		InternalName:      intName, | 		InternalName:      intName, | ||||||
| 		SubscribeKey:      subscribeKey, | 		SubscribeKey:      subscribeKey, | ||||||
| 		DescriptionName:   description, | 		DescriptionName:   description, | ||||||
| 		TimestampCreated:  time2DB(time.Now()), | 		TimestampCreated:  models.NowSCNTime(), | ||||||
| 		TimestampLastSent: nil, | 		TimestampLastSent: nil, | ||||||
| 		MessagesSent:      0, | 		MessagesSent:      0, | ||||||
| 	} | 	} | ||||||
| @@ -89,7 +56,7 @@ func (db *Database) CreateChannel(ctx db.TxContext, userid models.UserID, dispNa | |||||||
| 		return models.Channel{}, err | 		return models.Channel{}, err | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	return entity.Model(), nil | 	return entity, nil | ||||||
| } | } | ||||||
|  |  | ||||||
| func (db *Database) ListChannelsByOwner(ctx db.TxContext, userid models.UserID, subUserID models.UserID) ([]models.ChannelWithSubscription, error) { | func (db *Database) ListChannelsByOwner(ctx db.TxContext, userid models.UserID, subUserID models.UserID) ([]models.ChannelWithSubscription, error) { | ||||||
| @@ -100,20 +67,14 @@ func (db *Database) ListChannelsByOwner(ctx db.TxContext, userid models.UserID, | |||||||
|  |  | ||||||
| 	order := " ORDER BY channels.timestamp_created ASC, channels.channel_id ASC " | 	order := " ORDER BY channels.timestamp_created ASC, channels.channel_id ASC " | ||||||
|  |  | ||||||
| 	rows, err := tx.Query(ctx, "SELECT channels.*, sub.* FROM channels LEFT JOIN subscriptions AS sub ON channels.channel_id = sub.channel_id AND sub.subscriber_user_id = :subuid WHERE owner_user_id = :ouid"+order, sq.PP{ | 	sql := "SELECT channels.*, sub.* FROM channels LEFT JOIN subscriptions AS sub ON channels.channel_id = sub.channel_id AND sub.subscriber_user_id = :subuid WHERE owner_user_id = :ouid" + order | ||||||
|  |  | ||||||
|  | 	pp := sq.PP{ | ||||||
| 		"ouid":   userid, | 		"ouid":   userid, | ||||||
| 		"subuid": subUserID, | 		"subuid": subUserID, | ||||||
| 	}) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return nil, err |  | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	data, err := models.DecodeChannelsWithSubscription(ctx, tx, rows) | 	return sq.QueryAll[models.ChannelWithSubscription](ctx, tx, sql, pp, sq.SModeExtended, sq.Safe) | ||||||
| 	if err != nil { |  | ||||||
| 		return nil, err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	return data, nil |  | ||||||
| } | } | ||||||
|  |  | ||||||
| func (db *Database) ListChannelsBySubscriber(ctx db.TxContext, userid models.UserID, confirmed *bool) ([]models.ChannelWithSubscription, error) { | func (db *Database) ListChannelsBySubscriber(ctx db.TxContext, userid models.UserID, confirmed *bool) ([]models.ChannelWithSubscription, error) { | ||||||
| @@ -131,19 +92,13 @@ func (db *Database) ListChannelsBySubscriber(ctx db.TxContext, userid models.Use | |||||||
|  |  | ||||||
| 	order := " ORDER BY channels.timestamp_created ASC, channels.channel_id ASC " | 	order := " ORDER BY channels.timestamp_created ASC, channels.channel_id ASC " | ||||||
|  |  | ||||||
| 	rows, err := tx.Query(ctx, "SELECT channels.*, sub.* FROM channels LEFT JOIN subscriptions AS sub on channels.channel_id = sub.channel_id AND sub.subscriber_user_id = :subuid WHERE sub.subscription_id IS NOT NULL "+confCond+order, sq.PP{ | 	sql := "SELECT channels.*, sub.* FROM channels LEFT JOIN subscriptions AS sub on channels.channel_id = sub.channel_id AND sub.subscriber_user_id = :subuid WHERE sub.subscription_id IS NOT NULL " + confCond + order | ||||||
|  |  | ||||||
|  | 	pp := sq.PP{ | ||||||
| 		"subuid": userid, | 		"subuid": userid, | ||||||
| 	}) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return nil, err |  | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	data, err := models.DecodeChannelsWithSubscription(ctx, tx, rows) | 	return sq.QueryAll[models.ChannelWithSubscription](ctx, tx, sql, pp, sq.SModeExtended, sq.Safe) | ||||||
| 	if err != nil { |  | ||||||
| 		return nil, err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	return data, nil |  | ||||||
| } | } | ||||||
|  |  | ||||||
| func (db *Database) ListChannelsByAccess(ctx db.TxContext, userid models.UserID, confirmed *bool) ([]models.ChannelWithSubscription, error) { | func (db *Database) ListChannelsByAccess(ctx db.TxContext, userid models.UserID, confirmed *bool) ([]models.ChannelWithSubscription, error) { | ||||||
| @@ -161,20 +116,14 @@ func (db *Database) ListChannelsByAccess(ctx db.TxContext, userid models.UserID, | |||||||
|  |  | ||||||
| 	order := " ORDER BY channels.timestamp_created ASC, channels.channel_id ASC " | 	order := " ORDER BY channels.timestamp_created ASC, channels.channel_id ASC " | ||||||
|  |  | ||||||
| 	rows, err := tx.Query(ctx, "SELECT channels.*, sub.* FROM channels LEFT JOIN subscriptions AS sub on channels.channel_id = sub.channel_id AND sub.subscriber_user_id = :subuid WHERE owner_user_id = :ouid "+confCond+order, sq.PP{ | 	sql := "SELECT channels.*, sub.* FROM channels LEFT JOIN subscriptions AS sub on channels.channel_id = sub.channel_id AND sub.subscriber_user_id = :subuid WHERE owner_user_id = :ouid " + confCond + order | ||||||
|  |  | ||||||
|  | 	pp := sq.PP{ | ||||||
| 		"ouid":   userid, | 		"ouid":   userid, | ||||||
| 		"subuid": userid, | 		"subuid": userid, | ||||||
| 	}) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return nil, err |  | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	data, err := models.DecodeChannelsWithSubscription(ctx, tx, rows) | 	return sq.QueryAll[models.ChannelWithSubscription](ctx, tx, sql, pp, sq.SModeExtended, sq.Safe) | ||||||
| 	if err != nil { |  | ||||||
| 		return nil, err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	return data, nil |  | ||||||
| } | } | ||||||
|  |  | ||||||
| func (db *Database) GetChannel(ctx db.TxContext, userid models.UserID, channelid models.ChannelID, enforceOwner bool) (models.ChannelWithSubscription, error) { | func (db *Database) GetChannel(ctx db.TxContext, userid models.UserID, channelid models.ChannelID, enforceOwner bool) (models.ChannelWithSubscription, error) { | ||||||
| @@ -198,17 +147,9 @@ func (db *Database) GetChannel(ctx db.TxContext, userid models.UserID, channelid | |||||||
| 		params["ouid"] = userid | 		params["ouid"] = userid | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	rows, err := tx.Query(ctx, "SELECT "+selectors+" FROM channels "+join+" WHERE "+cond+" LIMIT 1", params) | 	sql := "SELECT " + selectors + " FROM channels " + join + " WHERE " + cond + " LIMIT 1" | ||||||
| 	if err != nil { |  | ||||||
| 		return models.ChannelWithSubscription{}, err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	channel, err := models.DecodeChannelWithSubscription(ctx, tx, rows) | 	return sq.QuerySingle[models.ChannelWithSubscription](ctx, tx, sql, params, sq.SModeExtended, sq.Safe) | ||||||
| 	if err != nil { |  | ||||||
| 		return models.ChannelWithSubscription{}, err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	return channel, nil |  | ||||||
| } | } | ||||||
|  |  | ||||||
| func (db *Database) IncChannelMessageCounter(ctx db.TxContext, channel *models.Channel) error { | func (db *Database) IncChannelMessageCounter(ctx db.TxContext, channel *models.Channel) error { | ||||||
| @@ -228,7 +169,7 @@ func (db *Database) IncChannelMessageCounter(ctx db.TxContext, channel *models.C | |||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	channel.MessagesSent += 1 | 	channel.MessagesSent += 1 | ||||||
| 	channel.TimestampLastSent = &now | 	channel.TimestampLastSent = models.NewSCNTimePtr(&now) | ||||||
|  |  | ||||||
| 	return nil | 	return nil | ||||||
| } | } | ||||||
|   | |||||||
| @@ -4,7 +4,6 @@ import ( | |||||||
| 	"blackforestbytes.com/simplecloudnotifier/db" | 	"blackforestbytes.com/simplecloudnotifier/db" | ||||||
| 	"blackforestbytes.com/simplecloudnotifier/models" | 	"blackforestbytes.com/simplecloudnotifier/models" | ||||||
| 	"gogs.mikescher.com/BlackForestBytes/goext/sq" | 	"gogs.mikescher.com/BlackForestBytes/goext/sq" | ||||||
| 	"time" |  | ||||||
| ) | ) | ||||||
|  |  | ||||||
| func (db *Database) CreateClient(ctx db.TxContext, userid models.UserID, ctype models.ClientType, fcmToken string, agentModel string, agentVersion string, name *string) (models.Client, error) { | func (db *Database) CreateClient(ctx db.TxContext, userid models.UserID, ctype models.ClientType, fcmToken string, agentModel string, agentVersion string, name *string) (models.Client, error) { | ||||||
| @@ -13,12 +12,12 @@ func (db *Database) CreateClient(ctx db.TxContext, userid models.UserID, ctype m | |||||||
| 		return models.Client{}, err | 		return models.Client{}, err | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	entity := models.ClientDB{ | 	entity := models.Client{ | ||||||
| 		ClientID:         models.NewClientID(), | 		ClientID:         models.NewClientID(), | ||||||
| 		UserID:           userid, | 		UserID:           userid, | ||||||
| 		Type:             ctype, | 		Type:             ctype, | ||||||
| 		FCMToken:         fcmToken, | 		FCMToken:         fcmToken, | ||||||
| 		TimestampCreated: time2DB(time.Now()), | 		TimestampCreated: models.NowSCNTime(), | ||||||
| 		AgentModel:       agentModel, | 		AgentModel:       agentModel, | ||||||
| 		AgentVersion:     agentVersion, | 		AgentVersion:     agentVersion, | ||||||
| 		Name:             name, | 		Name:             name, | ||||||
| @@ -29,7 +28,7 @@ func (db *Database) CreateClient(ctx db.TxContext, userid models.UserID, ctype m | |||||||
| 		return models.Client{}, err | 		return models.Client{}, err | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	return entity.Model(), nil | 	return entity, nil | ||||||
| } | } | ||||||
|  |  | ||||||
| func (db *Database) ClearFCMTokens(ctx db.TxContext, fcmtoken string) error { | func (db *Database) ClearFCMTokens(ctx db.TxContext, fcmtoken string) error { | ||||||
| @@ -52,17 +51,7 @@ func (db *Database) ListClients(ctx db.TxContext, userid models.UserID) ([]model | |||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	rows, err := tx.Query(ctx, "SELECT * FROM clients WHERE user_id = :uid ORDER BY clients.timestamp_created DESC, clients.client_id ASC", sq.PP{"uid": userid}) | 	return sq.QueryAll[models.Client](ctx, tx, "SELECT * FROM clients WHERE user_id = :uid ORDER BY clients.timestamp_created DESC, clients.client_id ASC", sq.PP{"uid": userid}, sq.SModeExtended, sq.Safe) | ||||||
| 	if err != nil { |  | ||||||
| 		return nil, err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	data, err := models.DecodeClients(ctx, tx, rows) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return nil, err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	return data, nil |  | ||||||
| } | } | ||||||
|  |  | ||||||
| func (db *Database) GetClient(ctx db.TxContext, userid models.UserID, clientid models.ClientID) (models.Client, error) { | func (db *Database) GetClient(ctx db.TxContext, userid models.UserID, clientid models.ClientID) (models.Client, error) { | ||||||
| @@ -71,20 +60,10 @@ func (db *Database) GetClient(ctx db.TxContext, userid models.UserID, clientid m | |||||||
| 		return models.Client{}, err | 		return models.Client{}, err | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	rows, err := tx.Query(ctx, "SELECT * FROM clients WHERE user_id = :uid AND client_id = :cid LIMIT 1", sq.PP{ | 	return sq.QuerySingle[models.Client](ctx, tx, "SELECT * FROM clients WHERE user_id = :uid AND client_id = :cid LIMIT 1", sq.PP{ | ||||||
| 		"uid": userid, | 		"uid": userid, | ||||||
| 		"cid": clientid, | 		"cid": clientid, | ||||||
| 	}) | 	}, sq.SModeExtended, sq.Safe) | ||||||
| 	if err != nil { |  | ||||||
| 		return models.Client{}, err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	client, err := models.DecodeClient(ctx, tx, rows) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return models.Client{}, err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	return client, nil |  | ||||||
| } | } | ||||||
|  |  | ||||||
| func (db *Database) DeleteClient(ctx db.TxContext, clientid models.ClientID) error { | func (db *Database) DeleteClient(ctx db.TxContext, clientid models.ClientID) error { | ||||||
|   | |||||||
| @@ -5,12 +5,13 @@ import ( | |||||||
| 	"blackforestbytes.com/simplecloudnotifier/db/dbtools" | 	"blackforestbytes.com/simplecloudnotifier/db/dbtools" | ||||||
| 	"blackforestbytes.com/simplecloudnotifier/db/schema" | 	"blackforestbytes.com/simplecloudnotifier/db/schema" | ||||||
| 	"blackforestbytes.com/simplecloudnotifier/db/simplectx" | 	"blackforestbytes.com/simplecloudnotifier/db/simplectx" | ||||||
|  | 	"blackforestbytes.com/simplecloudnotifier/models" | ||||||
| 	"context" | 	"context" | ||||||
| 	"database/sql" | 	"database/sql" | ||||||
| 	"errors" | 	"errors" | ||||||
| 	"fmt" | 	"fmt" | ||||||
|  | 	"github.com/glebarez/go-sqlite" | ||||||
| 	"github.com/jmoiron/sqlx" | 	"github.com/jmoiron/sqlx" | ||||||
| 	_ "github.com/mattn/go-sqlite3" |  | ||||||
| 	"github.com/rs/zerolog/log" | 	"github.com/rs/zerolog/log" | ||||||
| 	"gogs.mikescher.com/BlackForestBytes/goext/langext" | 	"gogs.mikescher.com/BlackForestBytes/goext/langext" | ||||||
| 	"gogs.mikescher.com/BlackForestBytes/goext/sq" | 	"gogs.mikescher.com/BlackForestBytes/goext/sq" | ||||||
| @@ -26,7 +27,16 @@ type Database struct { | |||||||
| func NewPrimaryDatabase(cfg server.Config) (*Database, error) { | func NewPrimaryDatabase(cfg server.Config) (*Database, error) { | ||||||
| 	conf := cfg.DBMain | 	conf := cfg.DBMain | ||||||
|  |  | ||||||
| 	url := fmt.Sprintf("file:%s?_journal=%s&_timeout=%d&_fk=%s&_busy_timeout=%d", conf.File, conf.Journal, conf.Timeout.Milliseconds(), langext.FormatBool(conf.CheckForeignKeys, "true", "false"), conf.BusyTimeout.Milliseconds()) | 	url := fmt.Sprintf("file:%s?_pragma=journal_mode(%s)&_pragma=timeout(%d)&_pragma=foreign_keys(%s)&_pragma=busy_timeout(%d)", | ||||||
|  | 		conf.File, | ||||||
|  | 		conf.Journal, | ||||||
|  | 		conf.Timeout.Milliseconds(), | ||||||
|  | 		langext.FormatBool(conf.CheckForeignKeys, "true", "false"), | ||||||
|  | 		conf.BusyTimeout.Milliseconds()) | ||||||
|  |  | ||||||
|  | 	if !langext.InArray("sqlite3", sql.Drivers()) { | ||||||
|  | 		sqlite.RegisterAsSQLITE3() | ||||||
|  | 	} | ||||||
|  |  | ||||||
| 	xdb, err := sqlx.Open("sqlite3", url) | 	xdb, err := sqlx.Open("sqlite3", url) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| @@ -42,7 +52,8 @@ func NewPrimaryDatabase(cfg server.Config) (*Database, error) { | |||||||
| 		xdb.SetConnMaxIdleTime(60 * time.Minute) | 		xdb.SetConnMaxIdleTime(60 * time.Minute) | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	qqdb := sq.NewDB(xdb, sq.DBOptions{}) | 	qqdb := sq.NewDB(xdb, sq.DBOptions{RegisterDefaultConverter: langext.PTrue, RegisterCommentTrimmer: langext.PTrue}) | ||||||
|  | 	models.RegisterConverter(qqdb) | ||||||
|  |  | ||||||
| 	if conf.EnableLogger { | 	if conf.EnableLogger { | ||||||
| 		qqdb.AddListener(dbtools.DBLogger{}) | 		qqdb.AddListener(dbtools.DBLogger{}) | ||||||
|   | |||||||
| @@ -18,16 +18,16 @@ func (db *Database) CreateRetryDelivery(ctx db.TxContext, client models.Client, | |||||||
| 	now := time.Now() | 	now := time.Now() | ||||||
| 	next := scn.NextDeliveryTimestamp(now) | 	next := scn.NextDeliveryTimestamp(now) | ||||||
|  |  | ||||||
| 	entity := models.DeliveryDB{ | 	entity := models.Delivery{ | ||||||
| 		DeliveryID:         models.NewDeliveryID(), | 		DeliveryID:         models.NewDeliveryID(), | ||||||
| 		MessageID:          msg.MessageID, | 		MessageID:          msg.MessageID, | ||||||
| 		ReceiverUserID:     client.UserID, | 		ReceiverUserID:     client.UserID, | ||||||
| 		ReceiverClientID:   client.ClientID, | 		ReceiverClientID:   client.ClientID, | ||||||
| 		TimestampCreated:   time2DB(now), | 		TimestampCreated:   models.NewSCNTime(now), | ||||||
| 		TimestampFinalized: nil, | 		TimestampFinalized: nil, | ||||||
| 		Status:             models.DeliveryStatusRetry, | 		Status:             models.DeliveryStatusRetry, | ||||||
| 		RetryCount:         0, | 		RetryCount:         0, | ||||||
| 		NextDelivery:       langext.Ptr(time2DB(next)), | 		NextDelivery:       models.NewSCNTimePtr(&next), | ||||||
| 		FCMMessageID:       nil, | 		FCMMessageID:       nil, | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| @@ -36,7 +36,7 @@ func (db *Database) CreateRetryDelivery(ctx db.TxContext, client models.Client, | |||||||
| 		return models.Delivery{}, err | 		return models.Delivery{}, err | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	return entity.Model(), nil | 	return entity, nil | ||||||
| } | } | ||||||
|  |  | ||||||
| func (db *Database) CreateSuccessDelivery(ctx db.TxContext, client models.Client, msg models.Message, fcmDelivID string) (models.Delivery, error) { | func (db *Database) CreateSuccessDelivery(ctx db.TxContext, client models.Client, msg models.Message, fcmDelivID string) (models.Delivery, error) { | ||||||
| @@ -47,13 +47,13 @@ func (db *Database) CreateSuccessDelivery(ctx db.TxContext, client models.Client | |||||||
|  |  | ||||||
| 	now := time.Now() | 	now := time.Now() | ||||||
|  |  | ||||||
| 	entity := models.DeliveryDB{ | 	entity := models.Delivery{ | ||||||
| 		DeliveryID:         models.NewDeliveryID(), | 		DeliveryID:         models.NewDeliveryID(), | ||||||
| 		MessageID:          msg.MessageID, | 		MessageID:          msg.MessageID, | ||||||
| 		ReceiverUserID:     client.UserID, | 		ReceiverUserID:     client.UserID, | ||||||
| 		ReceiverClientID:   client.ClientID, | 		ReceiverClientID:   client.ClientID, | ||||||
| 		TimestampCreated:   time2DB(now), | 		TimestampCreated:   models.NewSCNTime(now), | ||||||
| 		TimestampFinalized: langext.Ptr(time2DB(now)), | 		TimestampFinalized: models.NewSCNTimePtr(&now), | ||||||
| 		Status:             models.DeliveryStatusSuccess, | 		Status:             models.DeliveryStatusSuccess, | ||||||
| 		RetryCount:         0, | 		RetryCount:         0, | ||||||
| 		NextDelivery:       nil, | 		NextDelivery:       nil, | ||||||
| @@ -65,7 +65,7 @@ func (db *Database) CreateSuccessDelivery(ctx db.TxContext, client models.Client | |||||||
| 		return models.Delivery{}, err | 		return models.Delivery{}, err | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	return entity.Model(), nil | 	return entity, nil | ||||||
| } | } | ||||||
|  |  | ||||||
| func (db *Database) ListRetrieableDeliveries(ctx db.TxContext, pageSize int) ([]models.Delivery, error) { | func (db *Database) ListRetrieableDeliveries(ctx db.TxContext, pageSize int) ([]models.Delivery, error) { | ||||||
| @@ -74,20 +74,10 @@ func (db *Database) ListRetrieableDeliveries(ctx db.TxContext, pageSize int) ([] | |||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	rows, err := tx.Query(ctx, "SELECT * FROM deliveries WHERE status = 'RETRY' AND next_delivery < :next ORDER BY next_delivery ASC LIMIT :lim", sq.PP{ | 	return sq.QueryAll[models.Delivery](ctx, tx, "SELECT * FROM deliveries WHERE status = 'RETRY' AND next_delivery < :next ORDER BY next_delivery ASC LIMIT :lim", sq.PP{ | ||||||
| 		"next": time2DB(time.Now()), | 		"next": time2DB(time.Now()), | ||||||
| 		"lim":  pageSize, | 		"lim":  pageSize, | ||||||
| 	}) | 	}, sq.SModeExtended, sq.Safe) | ||||||
| 	if err != nil { |  | ||||||
| 		return nil, err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	data, err := models.DecodeDeliveries(ctx, tx, rows) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return nil, err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	return data, nil |  | ||||||
| } | } | ||||||
|  |  | ||||||
| func (db *Database) SetDeliverySuccess(ctx db.TxContext, delivery models.Delivery, fcmDelivID string) error { | func (db *Database) SetDeliverySuccess(ctx db.TxContext, delivery models.Delivery, fcmDelivID string) error { | ||||||
|   | |||||||
| @@ -3,8 +3,6 @@ package primary | |||||||
| import ( | import ( | ||||||
| 	"blackforestbytes.com/simplecloudnotifier/db" | 	"blackforestbytes.com/simplecloudnotifier/db" | ||||||
| 	"blackforestbytes.com/simplecloudnotifier/models" | 	"blackforestbytes.com/simplecloudnotifier/models" | ||||||
| 	"database/sql" |  | ||||||
| 	"errors" |  | ||||||
| 	"gogs.mikescher.com/BlackForestBytes/goext/langext" | 	"gogs.mikescher.com/BlackForestBytes/goext/langext" | ||||||
| 	"gogs.mikescher.com/BlackForestBytes/goext/sq" | 	"gogs.mikescher.com/BlackForestBytes/goext/sq" | ||||||
| 	"strings" | 	"strings" | ||||||
| @@ -17,16 +15,16 @@ func (db *Database) CreateKeyToken(ctx db.TxContext, name string, owner models.U | |||||||
| 		return models.KeyToken{}, err | 		return models.KeyToken{}, err | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	entity := models.KeyTokenDB{ | 	entity := models.KeyToken{ | ||||||
| 		KeyTokenID:        models.NewKeyTokenID(), | 		KeyTokenID:        models.NewKeyTokenID(), | ||||||
| 		Name:              name, | 		Name:              name, | ||||||
| 		TimestampCreated:  time2DB(time.Now()), | 		TimestampCreated:  models.NowSCNTime(), | ||||||
| 		TimestampLastUsed: nil, | 		TimestampLastUsed: nil, | ||||||
| 		OwnerUserID:       owner, | 		OwnerUserID:       owner, | ||||||
| 		AllChannels:       allChannels, | 		AllChannels:       allChannels, | ||||||
| 		Channels:          strings.Join(langext.ArrMap(channels, func(v models.ChannelID) string { return v.String() }), ";"), | 		Channels:          channels, | ||||||
| 		Token:             token, | 		Token:             token, | ||||||
| 		Permissions:       permissions.String(), | 		Permissions:       permissions, | ||||||
| 		MessagesSent:      0, | 		MessagesSent:      0, | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| @@ -35,7 +33,7 @@ func (db *Database) CreateKeyToken(ctx db.TxContext, name string, owner models.U | |||||||
| 		return models.KeyToken{}, err | 		return models.KeyToken{}, err | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	return entity.Model(), nil | 	return entity, nil | ||||||
| } | } | ||||||
|  |  | ||||||
| func (db *Database) ListKeyTokens(ctx db.TxContext, ownerID models.UserID) ([]models.KeyToken, error) { | func (db *Database) ListKeyTokens(ctx db.TxContext, ownerID models.UserID) ([]models.KeyToken, error) { | ||||||
| @@ -44,17 +42,7 @@ func (db *Database) ListKeyTokens(ctx db.TxContext, ownerID models.UserID) ([]mo | |||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	rows, err := tx.Query(ctx, "SELECT * FROM keytokens WHERE owner_user_id = :uid ORDER BY keytokens.timestamp_created DESC, keytokens.keytoken_id ASC", sq.PP{"uid": ownerID}) | 	return sq.QueryAll[models.KeyToken](ctx, tx, "SELECT * FROM keytokens WHERE owner_user_id = :uid ORDER BY keytokens.timestamp_created DESC, keytokens.keytoken_id ASC", sq.PP{"uid": ownerID}, sq.SModeExtended, sq.Safe) | ||||||
| 	if err != nil { |  | ||||||
| 		return nil, err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	data, err := models.DecodeKeyTokens(ctx, tx, rows) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return nil, err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	return data, nil |  | ||||||
| } | } | ||||||
|  |  | ||||||
| func (db *Database) GetKeyToken(ctx db.TxContext, userid models.UserID, keyTokenid models.KeyTokenID) (models.KeyToken, error) { | func (db *Database) GetKeyToken(ctx db.TxContext, userid models.UserID, keyTokenid models.KeyTokenID) (models.KeyToken, error) { | ||||||
| @@ -63,20 +51,10 @@ func (db *Database) GetKeyToken(ctx db.TxContext, userid models.UserID, keyToken | |||||||
| 		return models.KeyToken{}, err | 		return models.KeyToken{}, err | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	rows, err := tx.Query(ctx, "SELECT * FROM keytokens WHERE owner_user_id = :uid AND keytoken_id = :cid LIMIT 1", sq.PP{ | 	return sq.QuerySingle[models.KeyToken](ctx, tx, "SELECT * FROM keytokens WHERE owner_user_id = :uid AND keytoken_id = :cid LIMIT 1", sq.PP{ | ||||||
| 		"uid": userid, | 		"uid": userid, | ||||||
| 		"cid": keyTokenid, | 		"cid": keyTokenid, | ||||||
| 	}) | 	}, sq.SModeExtended, sq.Safe) | ||||||
| 	if err != nil { |  | ||||||
| 		return models.KeyToken{}, err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	keyToken, err := models.DecodeKeyToken(ctx, tx, rows) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return models.KeyToken{}, err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	return keyToken, nil |  | ||||||
| } | } | ||||||
|  |  | ||||||
| func (db *Database) GetKeyTokenByID(ctx db.TxContext, keyTokenid models.KeyTokenID) (models.KeyToken, error) { | func (db *Database) GetKeyTokenByID(ctx db.TxContext, keyTokenid models.KeyTokenID) (models.KeyToken, error) { | ||||||
| @@ -85,19 +63,7 @@ func (db *Database) GetKeyTokenByID(ctx db.TxContext, keyTokenid models.KeyToken | |||||||
| 		return models.KeyToken{}, err | 		return models.KeyToken{}, err | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	rows, err := tx.Query(ctx, "SELECT * FROM keytokens WHERE keytoken_id = :cid LIMIT 1", sq.PP{ | 	return sq.QuerySingle[models.KeyToken](ctx, tx, "SELECT * FROM keytokens WHERE keytoken_id = :cid LIMIT 1", sq.PP{"cid": keyTokenid}, sq.SModeExtended, sq.Safe) | ||||||
| 		"cid": keyTokenid, |  | ||||||
| 	}) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return models.KeyToken{}, err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	keyToken, err := models.DecodeKeyToken(ctx, tx, rows) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return models.KeyToken{}, err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	return keyToken, nil |  | ||||||
| } | } | ||||||
|  |  | ||||||
| func (db *Database) GetKeyTokenByToken(ctx db.TxContext, key string) (*models.KeyToken, error) { | func (db *Database) GetKeyTokenByToken(ctx db.TxContext, key string) (*models.KeyToken, error) { | ||||||
| @@ -106,20 +72,7 @@ func (db *Database) GetKeyTokenByToken(ctx db.TxContext, key string) (*models.Ke | |||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	rows, err := tx.Query(ctx, "SELECT * FROM keytokens WHERE token = :key LIMIT 1", sq.PP{"key": key}) | 	return sq.QuerySingleOpt[models.KeyToken](ctx, tx, "SELECT * FROM keytokens WHERE token = :key LIMIT 1", sq.PP{"key": key}, sq.SModeExtended, sq.Safe) | ||||||
| 	if err != nil { |  | ||||||
| 		return nil, err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	user, err := models.DecodeKeyToken(ctx, tx, rows) |  | ||||||
| 	if errors.Is(err, sql.ErrNoRows) { |  | ||||||
| 		return nil, nil |  | ||||||
| 	} |  | ||||||
| 	if err != nil { |  | ||||||
| 		return nil, err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	return &user, nil |  | ||||||
| } | } | ||||||
|  |  | ||||||
| func (db *Database) DeleteKeyToken(ctx db.TxContext, keyTokenid models.KeyTokenID) error { | func (db *Database) DeleteKeyToken(ctx db.TxContext, keyTokenid models.KeyTokenID) error { | ||||||
| @@ -220,7 +173,7 @@ func (db *Database) IncKeyTokenMessageCounter(ctx db.TxContext, keyToken *models | |||||||
| 		return err | 		return err | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	keyToken.TimestampLastUsed = &now | 	keyToken.TimestampLastUsed = models.NewSCNTimePtr(&now) | ||||||
| 	keyToken.MessagesSent += 1 | 	keyToken.MessagesSent += 1 | ||||||
|  |  | ||||||
| 	return nil | 	return nil | ||||||
|   | |||||||
| @@ -4,7 +4,6 @@ import ( | |||||||
| 	"blackforestbytes.com/simplecloudnotifier/db" | 	"blackforestbytes.com/simplecloudnotifier/db" | ||||||
| 	ct "blackforestbytes.com/simplecloudnotifier/db/cursortoken" | 	ct "blackforestbytes.com/simplecloudnotifier/db/cursortoken" | ||||||
| 	"blackforestbytes.com/simplecloudnotifier/models" | 	"blackforestbytes.com/simplecloudnotifier/models" | ||||||
| 	"database/sql" |  | ||||||
| 	"errors" | 	"errors" | ||||||
| 	"gogs.mikescher.com/BlackForestBytes/goext/sq" | 	"gogs.mikescher.com/BlackForestBytes/goext/sq" | ||||||
| 	"time" | 	"time" | ||||||
| @@ -16,20 +15,7 @@ func (db *Database) GetMessageByUserMessageID(ctx db.TxContext, usrMsgId string) | |||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	rows, err := tx.Query(ctx, "SELECT * FROM messages WHERE usr_message_id = :umid LIMIT 1", sq.PP{"umid": usrMsgId}) | 	return sq.QuerySingleOpt[models.Message](ctx, tx, "SELECT * FROM messages WHERE usr_message_id = :umid LIMIT 1", sq.PP{"umid": usrMsgId}, sq.SModeExtended, sq.Safe) | ||||||
| 	if err != nil { |  | ||||||
| 		return nil, err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	msg, err := models.DecodeMessage(ctx, tx, rows) |  | ||||||
| 	if errors.Is(err, sql.ErrNoRows) { |  | ||||||
| 		return nil, nil |  | ||||||
| 	} |  | ||||||
| 	if err != nil { |  | ||||||
| 		return nil, err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	return &msg, nil |  | ||||||
| } | } | ||||||
|  |  | ||||||
| func (db *Database) GetMessage(ctx db.TxContext, scnMessageID models.MessageID, allowDeleted bool) (models.Message, error) { | func (db *Database) GetMessage(ctx db.TxContext, scnMessageID models.MessageID, allowDeleted bool) (models.Message, error) { | ||||||
| @@ -45,17 +31,7 @@ func (db *Database) GetMessage(ctx db.TxContext, scnMessageID models.MessageID, | |||||||
| 		sqlcmd = "SELECT * FROM messages WHERE message_id = :mid AND deleted=0 LIMIT 1" | 		sqlcmd = "SELECT * FROM messages WHERE message_id = :mid AND deleted=0 LIMIT 1" | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	rows, err := tx.Query(ctx, sqlcmd, sq.PP{"mid": scnMessageID}) | 	return sq.QuerySingle[models.Message](ctx, tx, sqlcmd, sq.PP{"mid": scnMessageID}, sq.SModeExtended, sq.Safe) | ||||||
| 	if err != nil { |  | ||||||
| 		return models.Message{}, err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	msg, err := models.DecodeMessage(ctx, tx, rows) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return models.Message{}, err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	return msg, nil |  | ||||||
| } | } | ||||||
|  |  | ||||||
| func (db *Database) CreateMessage(ctx db.TxContext, senderUserID models.UserID, channel models.Channel, timestampSend *time.Time, title string, content *string, priority int, userMsgId *string, senderIP string, senderName *string, usedKeyID models.KeyTokenID) (models.Message, error) { | func (db *Database) CreateMessage(ctx db.TxContext, senderUserID models.UserID, channel models.Channel, timestampSend *time.Time, title string, content *string, priority int, userMsgId *string, senderIP string, senderName *string, usedKeyID models.KeyTokenID) (models.Message, error) { | ||||||
| @@ -64,21 +40,22 @@ func (db *Database) CreateMessage(ctx db.TxContext, senderUserID models.UserID, | |||||||
| 		return models.Message{}, err | 		return models.Message{}, err | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	entity := models.MessageDB{ | 	entity := models.Message{ | ||||||
| 		MessageID:           models.NewMessageID(), | 		MessageID:           models.NewMessageID(), | ||||||
| 		SenderUserID:        senderUserID, | 		SenderUserID:        senderUserID, | ||||||
| 		ChannelInternalName: channel.InternalName, | 		ChannelInternalName: channel.InternalName, | ||||||
| 		ChannelID:           channel.ChannelID, | 		ChannelID:           channel.ChannelID, | ||||||
| 		SenderIP:            senderIP, | 		SenderIP:            senderIP, | ||||||
| 		SenderName:          senderName, | 		SenderName:          senderName, | ||||||
| 		TimestampReal:       time2DB(time.Now()), | 		TimestampReal:       models.NowSCNTime(), | ||||||
| 		TimestampClient:     time2DBOpt(timestampSend), | 		TimestampClient:     models.NewSCNTimePtr(timestampSend), | ||||||
| 		Title:               title, | 		Title:               title, | ||||||
| 		Content:             content, | 		Content:             content, | ||||||
| 		Priority:            priority, | 		Priority:            priority, | ||||||
| 		UserMessageID:       userMsgId, | 		UserMessageID:       userMsgId, | ||||||
| 		UsedKeyID:           usedKeyID, | 		UsedKeyID:           usedKeyID, | ||||||
| 		Deleted:             bool2DB(false), | 		Deleted:             false, | ||||||
|  | 		MessageExtra:        models.MessageExtra{}, | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	_, err = sq.InsertSingle(ctx, tx, "messages", entity) | 	_, err = sq.InsertSingle(ctx, tx, "messages", entity) | ||||||
| @@ -86,7 +63,7 @@ func (db *Database) CreateMessage(ctx db.TxContext, senderUserID models.UserID, | |||||||
| 		return models.Message{}, err | 		return models.Message{}, err | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	return entity.Model(), nil | 	return entity, nil | ||||||
| } | } | ||||||
|  |  | ||||||
| func (db *Database) DeleteMessage(ctx db.TxContext, messageID models.MessageID) error { | func (db *Database) DeleteMessage(ctx db.TxContext, messageID models.MessageID) error { | ||||||
| @@ -133,12 +110,7 @@ func (db *Database) ListMessages(ctx db.TxContext, filter models.MessageFilter, | |||||||
| 	prepParams["tokts"] = inTok.Timestamp | 	prepParams["tokts"] = inTok.Timestamp | ||||||
| 	prepParams["tokid"] = inTok.Id | 	prepParams["tokid"] = inTok.Id | ||||||
|  |  | ||||||
| 	rows, err := tx.Query(ctx, sqlQuery, prepParams) | 	data, err := sq.QueryAll[models.Message](ctx, tx, sqlQuery, prepParams, sq.SModeExtended, sq.Safe) | ||||||
| 	if err != nil { |  | ||||||
| 		return nil, ct.CursorToken{}, err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	data, err := models.DecodeMessages(ctx, tx, rows) |  | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, ct.CursorToken{}, err | 		return nil, ct.CursorToken{}, err | ||||||
| 	} | 	} | ||||||
|   | |||||||
| @@ -3,10 +3,7 @@ package primary | |||||||
| import ( | import ( | ||||||
| 	"blackforestbytes.com/simplecloudnotifier/db" | 	"blackforestbytes.com/simplecloudnotifier/db" | ||||||
| 	"blackforestbytes.com/simplecloudnotifier/models" | 	"blackforestbytes.com/simplecloudnotifier/models" | ||||||
| 	"database/sql" |  | ||||||
| 	"errors" |  | ||||||
| 	"gogs.mikescher.com/BlackForestBytes/goext/sq" | 	"gogs.mikescher.com/BlackForestBytes/goext/sq" | ||||||
| 	"time" |  | ||||||
| ) | ) | ||||||
|  |  | ||||||
| func (db *Database) CreateSubscription(ctx db.TxContext, subscriberUID models.UserID, channel models.Channel, confirmed bool) (models.Subscription, error) { | func (db *Database) CreateSubscription(ctx db.TxContext, subscriberUID models.UserID, channel models.Channel, confirmed bool) (models.Subscription, error) { | ||||||
| @@ -15,14 +12,14 @@ func (db *Database) CreateSubscription(ctx db.TxContext, subscriberUID models.Us | |||||||
| 		return models.Subscription{}, err | 		return models.Subscription{}, err | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	entity := models.SubscriptionDB{ | 	entity := models.Subscription{ | ||||||
| 		SubscriptionID:      models.NewSubscriptionID(), | 		SubscriptionID:      models.NewSubscriptionID(), | ||||||
| 		SubscriberUserID:    subscriberUID, | 		SubscriberUserID:    subscriberUID, | ||||||
| 		ChannelOwnerUserID:  channel.OwnerUserID, | 		ChannelOwnerUserID:  channel.OwnerUserID, | ||||||
| 		ChannelID:           channel.ChannelID, | 		ChannelID:           channel.ChannelID, | ||||||
| 		ChannelInternalName: channel.InternalName, | 		ChannelInternalName: channel.InternalName, | ||||||
| 		TimestampCreated:    time2DB(time.Now()), | 		TimestampCreated:    models.NowSCNTime(), | ||||||
| 		Confirmed:           bool2DB(confirmed), | 		Confirmed:           confirmed, | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	_, err = sq.InsertSingle(ctx, tx, "subscriptions", entity) | 	_, err = sq.InsertSingle(ctx, tx, "subscriptions", entity) | ||||||
| @@ -30,7 +27,7 @@ func (db *Database) CreateSubscription(ctx db.TxContext, subscriberUID models.Us | |||||||
| 		return models.Subscription{}, err | 		return models.Subscription{}, err | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	return entity.Model(), nil | 	return entity, nil | ||||||
| } | } | ||||||
|  |  | ||||||
| func (db *Database) ListSubscriptions(ctx db.TxContext, filter models.SubscriptionFilter) ([]models.Subscription, error) { | func (db *Database) ListSubscriptions(ctx db.TxContext, filter models.SubscriptionFilter) ([]models.Subscription, error) { | ||||||
| @@ -45,17 +42,7 @@ func (db *Database) ListSubscriptions(ctx db.TxContext, filter models.Subscripti | |||||||
|  |  | ||||||
| 	sqlQuery := "SELECT " + "subscriptions.*" + " FROM subscriptions " + filterJoin + " WHERE ( " + filterCond + " ) " + orderClause | 	sqlQuery := "SELECT " + "subscriptions.*" + " FROM subscriptions " + filterJoin + " WHERE ( " + filterCond + " ) " + orderClause | ||||||
|  |  | ||||||
| 	rows, err := tx.Query(ctx, sqlQuery, prepParams) | 	return sq.QueryAll[models.Subscription](ctx, tx, sqlQuery, prepParams, sq.SModeExtended, sq.Safe) | ||||||
| 	if err != nil { |  | ||||||
| 		return nil, err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	data, err := models.DecodeSubscriptions(ctx, tx, rows) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return nil, err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	return data, nil |  | ||||||
| } | } | ||||||
|  |  | ||||||
| func (db *Database) GetSubscription(ctx db.TxContext, subid models.SubscriptionID) (models.Subscription, error) { | func (db *Database) GetSubscription(ctx db.TxContext, subid models.SubscriptionID) (models.Subscription, error) { | ||||||
| @@ -64,17 +51,7 @@ func (db *Database) GetSubscription(ctx db.TxContext, subid models.SubscriptionI | |||||||
| 		return models.Subscription{}, err | 		return models.Subscription{}, err | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	rows, err := tx.Query(ctx, "SELECT * FROM subscriptions WHERE subscription_id = :sid LIMIT 1", sq.PP{"sid": subid}) | 	return sq.QuerySingle[models.Subscription](ctx, tx, "SELECT * FROM subscriptions WHERE subscription_id = :sid LIMIT 1", sq.PP{"sid": subid}, sq.SModeExtended, sq.Safe) | ||||||
| 	if err != nil { |  | ||||||
| 		return models.Subscription{}, err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	sub, err := models.DecodeSubscription(ctx, tx, rows) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return models.Subscription{}, err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	return sub, nil |  | ||||||
| } | } | ||||||
|  |  | ||||||
| func (db *Database) GetSubscriptionBySubscriber(ctx db.TxContext, subscriberId models.UserID, channelId models.ChannelID) (*models.Subscription, error) { | func (db *Database) GetSubscriptionBySubscriber(ctx db.TxContext, subscriberId models.UserID, channelId models.ChannelID) (*models.Subscription, error) { | ||||||
| @@ -83,23 +60,10 @@ func (db *Database) GetSubscriptionBySubscriber(ctx db.TxContext, subscriberId m | |||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	rows, err := tx.Query(ctx, "SELECT * FROM subscriptions WHERE subscriber_user_id = :suid AND channel_id = :cid LIMIT 1", sq.PP{ | 	return sq.QuerySingleOpt[models.Subscription](ctx, tx, "SELECT * FROM subscriptions WHERE subscriber_user_id = :suid AND channel_id = :cid LIMIT 1", sq.PP{ | ||||||
| 		"suid": subscriberId, | 		"suid": subscriberId, | ||||||
| 		"cid":  channelId, | 		"cid":  channelId, | ||||||
| 	}) | 	}, sq.SModeExtended, sq.Safe) | ||||||
| 	if err != nil { |  | ||||||
| 		return nil, err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	user, err := models.DecodeSubscription(ctx, tx, rows) |  | ||||||
| 	if errors.Is(err, sql.ErrNoRows) { |  | ||||||
| 		return nil, nil |  | ||||||
| 	} |  | ||||||
| 	if err != nil { |  | ||||||
| 		return nil, err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	return &user, nil |  | ||||||
| } | } | ||||||
|  |  | ||||||
| func (db *Database) DeleteSubscription(ctx db.TxContext, subid models.SubscriptionID) error { | func (db *Database) DeleteSubscription(ctx db.TxContext, subid models.SubscriptionID) error { | ||||||
|   | |||||||
| @@ -15,10 +15,10 @@ func (db *Database) CreateUser(ctx db.TxContext, protoken *string, username *str | |||||||
| 		return models.User{}, err | 		return models.User{}, err | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	entity := models.UserDB{ | 	entity := models.User{ | ||||||
| 		UserID:            models.NewUserID(), | 		UserID:            models.NewUserID(), | ||||||
| 		Username:          username, | 		Username:          username, | ||||||
| 		TimestampCreated:  time2DB(time.Now()), | 		TimestampCreated:  models.NowSCNTime(), | ||||||
| 		TimestampLastRead: nil, | 		TimestampLastRead: nil, | ||||||
| 		TimestampLastSent: nil, | 		TimestampLastSent: nil, | ||||||
| 		MessagesSent:      0, | 		MessagesSent:      0, | ||||||
| @@ -26,14 +26,17 @@ func (db *Database) CreateUser(ctx db.TxContext, protoken *string, username *str | |||||||
| 		QuotaUsedDay:      nil, | 		QuotaUsedDay:      nil, | ||||||
| 		IsPro:             protoken != nil, | 		IsPro:             protoken != nil, | ||||||
| 		ProToken:          protoken, | 		ProToken:          protoken, | ||||||
|  | 		UserExtra:         models.UserExtra{}, | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
|  | 	entity.PreMarshal() | ||||||
|  |  | ||||||
| 	_, err = sq.InsertSingle(ctx, tx, "users", entity) | 	_, err = sq.InsertSingle(ctx, tx, "users", entity) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return models.User{}, err | 		return models.User{}, err | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	return entity.Model(), nil | 	return entity, nil | ||||||
| } | } | ||||||
|  |  | ||||||
| func (db *Database) ClearProTokens(ctx db.TxContext, protoken string) error { | func (db *Database) ClearProTokens(ctx db.TxContext, protoken string) error { | ||||||
| @@ -56,17 +59,7 @@ func (db *Database) GetUser(ctx db.TxContext, userid models.UserID) (models.User | |||||||
| 		return models.User{}, err | 		return models.User{}, err | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	rows, err := tx.Query(ctx, "SELECT * FROM users WHERE user_id = :uid LIMIT 1", sq.PP{"uid": userid}) | 	return sq.QuerySingle[models.User](ctx, tx, "SELECT * FROM users WHERE user_id = :uid LIMIT 1", sq.PP{"uid": userid}, sq.SModeExtended, sq.Safe) | ||||||
| 	if err != nil { |  | ||||||
| 		return models.User{}, err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	user, err := models.DecodeUser(ctx, tx, rows) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return models.User{}, err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	return user, nil |  | ||||||
| } | } | ||||||
|  |  | ||||||
| func (db *Database) UpdateUserUsername(ctx db.TxContext, userid models.UserID, username *string) error { | func (db *Database) UpdateUserUsername(ctx db.TxContext, userid models.UserID, username *string) error { | ||||||
| @@ -127,7 +120,7 @@ func (db *Database) IncUserMessageCounter(ctx db.TxContext, user *models.User) e | |||||||
| 		return err | 		return err | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	user.TimestampLastSent = &now | 	user.TimestampLastSent = models.NewSCNTimePtr(&now) | ||||||
| 	user.MessagesSent = user.MessagesSent + 1 | 	user.MessagesSent = user.MessagesSent + 1 | ||||||
|  |  | ||||||
| 	return nil | 	return nil | ||||||
|   | |||||||
| @@ -5,12 +5,13 @@ import ( | |||||||
| 	"blackforestbytes.com/simplecloudnotifier/db/dbtools" | 	"blackforestbytes.com/simplecloudnotifier/db/dbtools" | ||||||
| 	"blackforestbytes.com/simplecloudnotifier/db/schema" | 	"blackforestbytes.com/simplecloudnotifier/db/schema" | ||||||
| 	"blackforestbytes.com/simplecloudnotifier/db/simplectx" | 	"blackforestbytes.com/simplecloudnotifier/db/simplectx" | ||||||
|  | 	"blackforestbytes.com/simplecloudnotifier/models" | ||||||
| 	"context" | 	"context" | ||||||
| 	"database/sql" | 	"database/sql" | ||||||
| 	"errors" | 	"errors" | ||||||
| 	"fmt" | 	"fmt" | ||||||
|  | 	"github.com/glebarez/go-sqlite" | ||||||
| 	"github.com/jmoiron/sqlx" | 	"github.com/jmoiron/sqlx" | ||||||
| 	_ "github.com/mattn/go-sqlite3" |  | ||||||
| 	"github.com/rs/zerolog/log" | 	"github.com/rs/zerolog/log" | ||||||
| 	"gogs.mikescher.com/BlackForestBytes/goext/langext" | 	"gogs.mikescher.com/BlackForestBytes/goext/langext" | ||||||
| 	"gogs.mikescher.com/BlackForestBytes/goext/sq" | 	"gogs.mikescher.com/BlackForestBytes/goext/sq" | ||||||
| @@ -26,7 +27,16 @@ type Database struct { | |||||||
| func NewRequestsDatabase(cfg server.Config) (*Database, error) { | func NewRequestsDatabase(cfg server.Config) (*Database, error) { | ||||||
| 	conf := cfg.DBRequests | 	conf := cfg.DBRequests | ||||||
|  |  | ||||||
| 	url := fmt.Sprintf("file:%s?_journal=%s&_timeout=%d&_fk=%s&_busy_timeout=%d", conf.File, conf.Journal, conf.Timeout.Milliseconds(), langext.FormatBool(conf.CheckForeignKeys, "true", "false"), conf.BusyTimeout.Milliseconds()) | 	url := fmt.Sprintf("file:%s?_pragma=journal_mode(%s)&_pragma=timeout(%d)&_pragma=foreign_keys(%s)&_pragma=busy_timeout(%d)", | ||||||
|  | 		conf.File, | ||||||
|  | 		conf.Journal, | ||||||
|  | 		conf.Timeout.Milliseconds(), | ||||||
|  | 		langext.FormatBool(conf.CheckForeignKeys, "true", "false"), | ||||||
|  | 		conf.BusyTimeout.Milliseconds()) | ||||||
|  |  | ||||||
|  | 	if !langext.InArray("sqlite3", sql.Drivers()) { | ||||||
|  | 		sqlite.RegisterAsSQLITE3() | ||||||
|  | 	} | ||||||
|  |  | ||||||
| 	xdb, err := sqlx.Open("sqlite3", url) | 	xdb, err := sqlx.Open("sqlite3", url) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| @@ -42,7 +52,8 @@ func NewRequestsDatabase(cfg server.Config) (*Database, error) { | |||||||
| 		xdb.SetConnMaxIdleTime(60 * time.Minute) | 		xdb.SetConnMaxIdleTime(60 * time.Minute) | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	qqdb := sq.NewDB(xdb, sq.DBOptions{}) | 	qqdb := sq.NewDB(xdb, sq.DBOptions{RegisterDefaultConverter: langext.PTrue, RegisterCommentTrimmer: langext.PTrue}) | ||||||
|  | 	models.RegisterConverter(qqdb) | ||||||
|  |  | ||||||
| 	if conf.EnableLogger { | 	if conf.EnableLogger { | ||||||
| 		qqdb.AddListener(dbtools.DBLogger{}) | 		qqdb.AddListener(dbtools.DBLogger{}) | ||||||
| @@ -92,7 +103,7 @@ func (db *Database) Migrate(outerctx context.Context) error { | |||||||
| 		schemastr := schema.RequestsSchema[schema.RequestsSchemaVersion].SQL | 		schemastr := schema.RequestsSchema[schema.RequestsSchemaVersion].SQL | ||||||
| 		schemahash := schema.RequestsSchema[schema.RequestsSchemaVersion].Hash | 		schemahash := schema.RequestsSchema[schema.RequestsSchemaVersion].Hash | ||||||
|  |  | ||||||
| 		schemahash, err := sq.HashMattnSqliteSchema(tctx, schemastr) | 		schemahash, err := sq.HashGoSqliteSchema(tctx, schemastr) | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| 			return err | 			return err | ||||||
| 		} | 		} | ||||||
|   | |||||||
| @@ -8,18 +8,17 @@ import ( | |||||||
| 	"time" | 	"time" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| func (db *Database) InsertRequestLog(ctx context.Context, requestid models.RequestID, data models.RequestLog) (models.RequestLog, error) { | func (db *Database) InsertRequestLog(ctx context.Context, requestid models.RequestID, entity models.RequestLog) (models.RequestLog, error) { | ||||||
|  |  | ||||||
| 	entity := data.DB() |  | ||||||
| 	entity.RequestID = requestid | 	entity.RequestID = requestid | ||||||
| 	entity.TimestampCreated = time2DB(time.Now()) | 	entity.TimestampCreated = models.NowSCNTime() | ||||||
|  |  | ||||||
| 	_, err := sq.InsertSingle(ctx, db.db, "requests", entity) | 	_, err := sq.InsertSingle(ctx, db.db, "requests", entity) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return models.RequestLog{}, err | 		return models.RequestLog{}, err | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	return entity.Model(), nil | 	return entity, nil | ||||||
| } | } | ||||||
|  |  | ||||||
| func (db *Database) Cleanup(ctx context.Context, count int, duration time.Duration) (int64, error) { | func (db *Database) Cleanup(ctx context.Context, count int, duration time.Duration) (int64, error) { | ||||||
| @@ -73,12 +72,7 @@ func (db *Database) ListRequestLogs(ctx context.Context, filter models.RequestLo | |||||||
| 	prepParams["tokts"] = inTok.Timestamp | 	prepParams["tokts"] = inTok.Timestamp | ||||||
| 	prepParams["tokid"] = inTok.Id | 	prepParams["tokid"] = inTok.Id | ||||||
|  |  | ||||||
| 	rows, err := db.db.Query(ctx, sqlQuery, prepParams) | 	data, err := sq.QueryAll[models.RequestLog](ctx, db.db, sqlQuery, prepParams, sq.SModeExtended, sq.Safe) | ||||||
| 	if err != nil { |  | ||||||
| 		return nil, ct.CursorToken{}, err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	data, err := models.DecodeRequestLogs(ctx, db.db, rows) |  | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, ct.CursorToken{}, err | 		return nil, ct.CursorToken{}, err | ||||||
| 	} | 	} | ||||||
| @@ -86,7 +80,7 @@ func (db *Database) ListRequestLogs(ctx context.Context, filter models.RequestLo | |||||||
| 	if pageSize == nil || len(data) <= *pageSize { | 	if pageSize == nil || len(data) <= *pageSize { | ||||||
| 		return data, ct.End(), nil | 		return data, ct.End(), nil | ||||||
| 	} else { | 	} else { | ||||||
| 		outToken := ct.Normal(data[*pageSize-1].TimestampCreated, data[*pageSize-1].RequestID.String(), "DESC", filter.Hash()) | 		outToken := ct.Normal(data[*pageSize-1].TimestampCreated.Time(), data[*pageSize-1].RequestID.String(), "DESC", filter.Hash()) | ||||||
| 		return data[0:*pageSize], outToken, nil | 		return data[0:*pageSize], outToken, nil | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
|   | |||||||
| @@ -6,53 +6,61 @@ toolchain go1.22.3 | |||||||
|  |  | ||||||
| require ( | require ( | ||||||
| 	github.com/gin-gonic/gin v1.10.0 | 	github.com/gin-gonic/gin v1.10.0 | ||||||
| 	github.com/go-playground/validator/v10 v10.20.0 | 	github.com/glebarez/go-sqlite v1.22.0 | ||||||
|  | 	github.com/go-playground/validator/v10 v10.22.1 | ||||||
| 	github.com/go-sql-driver/mysql v1.8.1 | 	github.com/go-sql-driver/mysql v1.8.1 | ||||||
| 	github.com/jmoiron/sqlx v1.4.0 | 	github.com/jmoiron/sqlx v1.4.0 | ||||||
| 	github.com/mattn/go-sqlite3 v1.14.22 | 	github.com/mattn/go-sqlite3 v1.14.22 | ||||||
| 	github.com/rs/zerolog v1.33.0 | 	github.com/rs/zerolog v1.33.0 | ||||||
| 	gogs.mikescher.com/BlackForestBytes/goext v0.0.463 | 	gogs.mikescher.com/BlackForestBytes/goext v0.0.513 | ||||||
| 	gopkg.in/loremipsum.v1 v1.1.2 | 	gopkg.in/loremipsum.v1 v1.1.2 | ||||||
| ) | ) | ||||||
|  |  | ||||||
| require ( | require ( | ||||||
| 	filippo.io/edwards25519 v1.1.0 // indirect | 	filippo.io/edwards25519 v1.1.0 // indirect | ||||||
| 	github.com/bytedance/sonic v1.11.8 // indirect | 	github.com/bytedance/sonic v1.12.2 // indirect | ||||||
| 	github.com/bytedance/sonic/loader v0.1.1 // indirect | 	github.com/bytedance/sonic/loader v0.2.0 // indirect | ||||||
| 	github.com/cloudwego/base64x v0.1.4 // indirect | 	github.com/cloudwego/base64x v0.1.4 // indirect | ||||||
| 	github.com/cloudwego/iasm v0.2.0 // indirect | 	github.com/cloudwego/iasm v0.2.0 // indirect | ||||||
| 	github.com/gabriel-vasile/mimetype v1.4.4 // indirect | 	github.com/dustin/go-humanize v1.0.1 // indirect | ||||||
|  | 	github.com/gabriel-vasile/mimetype v1.4.5 // indirect | ||||||
| 	github.com/gin-contrib/sse v0.1.0 // indirect | 	github.com/gin-contrib/sse v0.1.0 // indirect | ||||||
| 	github.com/go-playground/locales v0.14.1 // indirect | 	github.com/go-playground/locales v0.14.1 // indirect | ||||||
| 	github.com/go-playground/universal-translator v0.18.1 // indirect | 	github.com/go-playground/universal-translator v0.18.1 // indirect | ||||||
| 	github.com/goccy/go-json v0.10.3 // indirect | 	github.com/goccy/go-json v0.10.3 // indirect | ||||||
| 	github.com/golang/snappy v0.0.4 // indirect | 	github.com/golang/snappy v0.0.4 // indirect | ||||||
| 	github.com/google/go-cmp v0.5.9 // indirect | 	github.com/google/uuid v1.5.0 // indirect | ||||||
| 	github.com/json-iterator/go v1.1.12 // indirect | 	github.com/json-iterator/go v1.1.12 // indirect | ||||||
| 	github.com/klauspost/compress v1.17.8 // indirect | 	github.com/klauspost/compress v1.17.9 // indirect | ||||||
| 	github.com/klauspost/cpuid/v2 v2.2.7 // indirect | 	github.com/klauspost/cpuid/v2 v2.2.8 // indirect | ||||||
| 	github.com/leodido/go-urn v1.4.0 // indirect | 	github.com/leodido/go-urn v1.4.0 // indirect | ||||||
| 	github.com/mattn/go-colorable v0.1.13 // indirect | 	github.com/mattn/go-colorable v0.1.13 // indirect | ||||||
| 	github.com/mattn/go-isatty v0.0.20 // indirect | 	github.com/mattn/go-isatty v0.0.20 // indirect | ||||||
| 	github.com/modern-go/concurrent v0.0.0-20180306012644-bacd9c7ef1dd // indirect | 	github.com/modern-go/concurrent v0.0.0-20180306012644-bacd9c7ef1dd // indirect | ||||||
| 	github.com/modern-go/reflect2 v1.0.2 // indirect | 	github.com/modern-go/reflect2 v1.0.2 // indirect | ||||||
| 	github.com/montanaflynn/stats v0.7.1 // indirect | 	github.com/montanaflynn/stats v0.7.1 // indirect | ||||||
| 	github.com/pelletier/go-toml/v2 v2.2.2 // indirect | 	github.com/pelletier/go-toml/v2 v2.2.3 // indirect | ||||||
| 	github.com/rs/xid v1.5.0 // indirect | 	github.com/remyoudompheng/bigfft v0.0.0-20230129092748-24d4a6f8daec // indirect | ||||||
|  | 	github.com/rs/xid v1.6.0 // indirect | ||||||
| 	github.com/twitchyliquid64/golang-asm v0.15.1 // indirect | 	github.com/twitchyliquid64/golang-asm v0.15.1 // indirect | ||||||
| 	github.com/ugorji/go/codec v1.2.12 // indirect | 	github.com/ugorji/go/codec v1.2.12 // indirect | ||||||
|  | 	github.com/viney-shih/go-lock v1.1.2 // indirect | ||||||
| 	github.com/xdg-go/pbkdf2 v1.0.0 // indirect | 	github.com/xdg-go/pbkdf2 v1.0.0 // indirect | ||||||
| 	github.com/xdg-go/scram v1.1.2 // indirect | 	github.com/xdg-go/scram v1.1.2 // indirect | ||||||
| 	github.com/xdg-go/stringprep v1.0.4 // indirect | 	github.com/xdg-go/stringprep v1.0.4 // indirect | ||||||
| 	github.com/youmark/pkcs8 v0.0.0-20240424034433-3c2c7870ae76 // indirect | 	github.com/youmark/pkcs8 v0.0.0-20240726163527-a2c0da244d78 // indirect | ||||||
| 	go.mongodb.org/mongo-driver v1.15.0 // indirect | 	go.mongodb.org/mongo-driver v1.16.1 // indirect | ||||||
| 	golang.org/x/arch v0.8.0 // indirect | 	golang.org/x/arch v0.10.0 // indirect | ||||||
| 	golang.org/x/crypto v0.23.0 // indirect | 	golang.org/x/crypto v0.27.0 // indirect | ||||||
| 	golang.org/x/net v0.25.0 // indirect | 	golang.org/x/net v0.29.0 // indirect | ||||||
| 	golang.org/x/sync v0.7.0 // indirect | 	golang.org/x/sync v0.8.0 // indirect | ||||||
| 	golang.org/x/sys v0.20.0 // indirect | 	golang.org/x/sys v0.25.0 // indirect | ||||||
| 	golang.org/x/term v0.20.0 // indirect | 	golang.org/x/term v0.24.0 // indirect | ||||||
| 	golang.org/x/text v0.15.0 // indirect | 	golang.org/x/text v0.18.0 // indirect | ||||||
| 	google.golang.org/protobuf v1.34.1 // indirect | 	google.golang.org/protobuf v1.34.2 // indirect | ||||||
| 	gopkg.in/yaml.v3 v3.0.1 // indirect | 	gopkg.in/yaml.v3 v3.0.1 // indirect | ||||||
|  | 	modernc.org/libc v1.37.6 // indirect | ||||||
|  | 	modernc.org/mathutil v1.6.0 // indirect | ||||||
|  | 	modernc.org/memory v1.7.2 // indirect | ||||||
|  | 	modernc.org/sqlite v1.28.0 // indirect | ||||||
| ) | ) | ||||||
|   | |||||||
| @@ -1,9 +1,10 @@ | |||||||
| filippo.io/edwards25519 v1.1.0 h1:FNf4tywRC1HmFuKW5xopWpigGjJKiJSV0Cqo0cJWDaA= | filippo.io/edwards25519 v1.1.0 h1:FNf4tywRC1HmFuKW5xopWpigGjJKiJSV0Cqo0cJWDaA= | ||||||
| filippo.io/edwards25519 v1.1.0/go.mod h1:BxyFTGdWcka3PhytdK4V28tE5sGfRvvvRV7EaN4VDT4= | filippo.io/edwards25519 v1.1.0/go.mod h1:BxyFTGdWcka3PhytdK4V28tE5sGfRvvvRV7EaN4VDT4= | ||||||
| github.com/bytedance/sonic v1.11.8 h1:Zw/j1KfiS+OYTi9lyB3bb0CFxPJVkM17k1wyDG32LRA= | github.com/bytedance/sonic v1.12.2 h1:oaMFuRTpMHYLpCntGca65YWt5ny+wAceDERTkT2L9lg= | ||||||
| github.com/bytedance/sonic v1.11.8/go.mod h1:LysEHSvpvDySVdC2f87zGWf6CIKJcAvqab1ZaiQtds4= | github.com/bytedance/sonic v1.12.2/go.mod h1:B8Gt/XvtZ3Fqj+iSKMypzymZxw/FVwgIGKzMzT9r/rk= | ||||||
| github.com/bytedance/sonic/loader v0.1.1 h1:c+e5Pt1k/cy5wMveRDyk2X4B9hF4g7an8N3zCYjJFNM= |  | ||||||
| github.com/bytedance/sonic/loader v0.1.1/go.mod h1:ncP89zfokxS5LZrJxl5z0UJcsk4M4yY2JpfqGeCtNLU= | github.com/bytedance/sonic/loader v0.1.1/go.mod h1:ncP89zfokxS5LZrJxl5z0UJcsk4M4yY2JpfqGeCtNLU= | ||||||
|  | github.com/bytedance/sonic/loader v0.2.0 h1:zNprn+lsIP06C/IqCHs3gPQIvnvpKbbxyXQP1iU4kWM= | ||||||
|  | github.com/bytedance/sonic/loader v0.2.0/go.mod h1:ncP89zfokxS5LZrJxl5z0UJcsk4M4yY2JpfqGeCtNLU= | ||||||
| github.com/cloudwego/base64x v0.1.4 h1:jwCgWpFanWmN8xoIUHa2rtzmkd5J2plF/dnLS6Xd/0Y= | github.com/cloudwego/base64x v0.1.4 h1:jwCgWpFanWmN8xoIUHa2rtzmkd5J2plF/dnLS6Xd/0Y= | ||||||
| github.com/cloudwego/base64x v0.1.4/go.mod h1:0zlkT4Wn5C6NdauXdJRhSKRlJvmclQ1hhJgA0rcu/8w= | github.com/cloudwego/base64x v0.1.4/go.mod h1:0zlkT4Wn5C6NdauXdJRhSKRlJvmclQ1hhJgA0rcu/8w= | ||||||
| github.com/cloudwego/iasm v0.2.0 h1:1KNIy1I1H9hNNFEEH3DVnI4UujN+1zjpuk6gwHLTssg= | github.com/cloudwego/iasm v0.2.0 h1:1KNIy1I1H9hNNFEEH3DVnI4UujN+1zjpuk6gwHLTssg= | ||||||
| @@ -14,8 +15,8 @@ github.com/davecgh/go-spew v1.1.1 h1:vj9j/u1bqnvCEfJOwUhtlOARqs3+rkHYY13jYWTU97c | |||||||
| github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= | github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= | ||||||
| github.com/dustin/go-humanize v1.0.1 h1:GzkhY7T5VNhEkwH0PVJgjz+fX1rhBrR7pRT3mDkpeCY= | github.com/dustin/go-humanize v1.0.1 h1:GzkhY7T5VNhEkwH0PVJgjz+fX1rhBrR7pRT3mDkpeCY= | ||||||
| github.com/dustin/go-humanize v1.0.1/go.mod h1:Mu1zIs6XwVuF/gI1OepvI0qD18qycQx+mFykh5fBlto= | github.com/dustin/go-humanize v1.0.1/go.mod h1:Mu1zIs6XwVuF/gI1OepvI0qD18qycQx+mFykh5fBlto= | ||||||
| github.com/gabriel-vasile/mimetype v1.4.4 h1:QjV6pZ7/XZ7ryI2KuyeEDE8wnh7fHP9YnQy+R0LnH8I= | github.com/gabriel-vasile/mimetype v1.4.5 h1:J7wGKdGu33ocBOhGy0z653k/lFKLFDPJMG8Gql0kxn4= | ||||||
| github.com/gabriel-vasile/mimetype v1.4.4/go.mod h1:JwLei5XPtWdGiMFB5Pjle1oEeoSeEuJfJE+TtfvdB/s= | github.com/gabriel-vasile/mimetype v1.4.5/go.mod h1:ibHel+/kbxn9x2407k1izTA1S81ku1z/DlgOW2QE0M4= | ||||||
| github.com/gin-contrib/sse v0.1.0 h1:Y/yl/+YNO8GZSjAhjMsSuLt29uWRFHdHYUb5lYOV9qE= | github.com/gin-contrib/sse v0.1.0 h1:Y/yl/+YNO8GZSjAhjMsSuLt29uWRFHdHYUb5lYOV9qE= | ||||||
| github.com/gin-contrib/sse v0.1.0/go.mod h1:RHrZQHXnP2xjPF+u1gW/2HnVO7nvIa9PG3Gm+fLHvGI= | github.com/gin-contrib/sse v0.1.0/go.mod h1:RHrZQHXnP2xjPF+u1gW/2HnVO7nvIa9PG3Gm+fLHvGI= | ||||||
| github.com/gin-gonic/gin v1.10.0 h1:nTuyha1TYqgedzytsKYqna+DfLos46nTv2ygFy86HFU= | github.com/gin-gonic/gin v1.10.0 h1:nTuyha1TYqgedzytsKYqna+DfLos46nTv2ygFy86HFU= | ||||||
| @@ -28,8 +29,8 @@ github.com/go-playground/locales v0.14.1 h1:EWaQ/wswjilfKLTECiXz7Rh+3BjFhfDFKv/o | |||||||
| github.com/go-playground/locales v0.14.1/go.mod h1:hxrqLVvrK65+Rwrd5Fc6F2O76J/NuW9t0sjnWqG1slY= | github.com/go-playground/locales v0.14.1/go.mod h1:hxrqLVvrK65+Rwrd5Fc6F2O76J/NuW9t0sjnWqG1slY= | ||||||
| github.com/go-playground/universal-translator v0.18.1 h1:Bcnm0ZwsGyWbCzImXv+pAJnYK9S473LQFuzCbDbfSFY= | github.com/go-playground/universal-translator v0.18.1 h1:Bcnm0ZwsGyWbCzImXv+pAJnYK9S473LQFuzCbDbfSFY= | ||||||
| github.com/go-playground/universal-translator v0.18.1/go.mod h1:xekY+UJKNuX9WP91TpwSH2VMlDf28Uj24BCp08ZFTUY= | github.com/go-playground/universal-translator v0.18.1/go.mod h1:xekY+UJKNuX9WP91TpwSH2VMlDf28Uj24BCp08ZFTUY= | ||||||
| github.com/go-playground/validator/v10 v10.20.0 h1:K9ISHbSaI0lyB2eWMPJo+kOS/FBExVwjEviJTixqxL8= | github.com/go-playground/validator/v10 v10.22.1 h1:40JcKH+bBNGFczGuoBYgX4I6m/i27HYW8P9FDk5PbgA= | ||||||
| github.com/go-playground/validator/v10 v10.20.0/go.mod h1:dbuPbCMFw/DrkbEynArYaCwl3amGuJotoKCe95atGMM= | github.com/go-playground/validator/v10 v10.22.1/go.mod h1:dbuPbCMFw/DrkbEynArYaCwl3amGuJotoKCe95atGMM= | ||||||
| github.com/go-sql-driver/mysql v1.8.1 h1:LedoTUt/eveggdHS9qUFC1EFSa8bU2+1pZjSRpvNJ1Y= | github.com/go-sql-driver/mysql v1.8.1 h1:LedoTUt/eveggdHS9qUFC1EFSa8bU2+1pZjSRpvNJ1Y= | ||||||
| github.com/go-sql-driver/mysql v1.8.1/go.mod h1:wEBSXgmK//2ZFJyE+qWnIsVGmvmEKlqwuVSjsCm7DZg= | github.com/go-sql-driver/mysql v1.8.1/go.mod h1:wEBSXgmK//2ZFJyE+qWnIsVGmvmEKlqwuVSjsCm7DZg= | ||||||
| github.com/goccy/go-json v0.10.3 h1:KZ5WoDbxAIgm2HNbYckL0se1fHD6rz5j4ywS6ebzDqA= | github.com/goccy/go-json v0.10.3 h1:KZ5WoDbxAIgm2HNbYckL0se1fHD6rz5j4ywS6ebzDqA= | ||||||
| @@ -37,20 +38,22 @@ github.com/goccy/go-json v0.10.3/go.mod h1:oq7eo15ShAhp70Anwd5lgX2pLfOS3QCiwU/PU | |||||||
| github.com/godbus/dbus/v5 v5.0.4/go.mod h1:xhWf0FNVPg57R7Z0UbKHbJfkEywrmjJnf7w5xrFpKfA= | github.com/godbus/dbus/v5 v5.0.4/go.mod h1:xhWf0FNVPg57R7Z0UbKHbJfkEywrmjJnf7w5xrFpKfA= | ||||||
| github.com/golang/snappy v0.0.4 h1:yAGX7huGHXlcLOEtBnF4w7FQwA26wojNCwOYAEhLjQM= | github.com/golang/snappy v0.0.4 h1:yAGX7huGHXlcLOEtBnF4w7FQwA26wojNCwOYAEhLjQM= | ||||||
| github.com/golang/snappy v0.0.4/go.mod h1:/XxbfmMg8lxefKM7IXC3fBNl/7bRcc72aCRzEWrmP2Q= | github.com/golang/snappy v0.0.4/go.mod h1:/XxbfmMg8lxefKM7IXC3fBNl/7bRcc72aCRzEWrmP2Q= | ||||||
| github.com/google/go-cmp v0.5.9 h1:O2Tfq5qg4qc4AmwVlvv0oLiVAGB7enBSJ2x2DqQFi38= | github.com/google/go-cmp v0.6.0 h1:ofyhxvXcZhMsU5ulbFiLKl/XBFqE1GSq7atu8tAmTRI= | ||||||
| github.com/google/go-cmp v0.5.9/go.mod h1:17dUlkBOakJ0+DkrSSNjCkIjxS6bF9zb3elmeNGIjoY= | github.com/google/go-cmp v0.6.0/go.mod h1:17dUlkBOakJ0+DkrSSNjCkIjxS6bF9zb3elmeNGIjoY= | ||||||
| github.com/google/gofuzz v1.0.0/go.mod h1:dBl0BpW6vV/+mYPU4Po3pmUjxk6FQPldtuIdl/M65Eg= | github.com/google/gofuzz v1.0.0/go.mod h1:dBl0BpW6vV/+mYPU4Po3pmUjxk6FQPldtuIdl/M65Eg= | ||||||
|  | github.com/google/pprof v0.0.0-20221118152302-e6195bd50e26 h1:Xim43kblpZXfIBQsbuBVKCudVG457BR2GZFIz3uw3hQ= | ||||||
|  | github.com/google/pprof v0.0.0-20221118152302-e6195bd50e26/go.mod h1:dDKJzRmX4S37WGHujM7tX//fmj1uioxKzKxz3lo4HJo= | ||||||
| github.com/google/uuid v1.5.0 h1:1p67kYwdtXjb0gL0BPiP1Av9wiZPo5A8z2cWkTZ+eyU= | github.com/google/uuid v1.5.0 h1:1p67kYwdtXjb0gL0BPiP1Av9wiZPo5A8z2cWkTZ+eyU= | ||||||
| github.com/google/uuid v1.5.0/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo= | github.com/google/uuid v1.5.0/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo= | ||||||
| github.com/jmoiron/sqlx v1.4.0 h1:1PLqN7S1UYp5t4SrVVnt4nUVNemrDAtxlulVe+Qgm3o= | github.com/jmoiron/sqlx v1.4.0 h1:1PLqN7S1UYp5t4SrVVnt4nUVNemrDAtxlulVe+Qgm3o= | ||||||
| github.com/jmoiron/sqlx v1.4.0/go.mod h1:ZrZ7UsYB/weZdl2Bxg6jCRO9c3YHl8r3ahlKmRT4JLY= | github.com/jmoiron/sqlx v1.4.0/go.mod h1:ZrZ7UsYB/weZdl2Bxg6jCRO9c3YHl8r3ahlKmRT4JLY= | ||||||
| github.com/json-iterator/go v1.1.12 h1:PV8peI4a0ysnczrg+LtxykD8LfKY9ML6u2jnxaEnrnM= | github.com/json-iterator/go v1.1.12 h1:PV8peI4a0ysnczrg+LtxykD8LfKY9ML6u2jnxaEnrnM= | ||||||
| github.com/json-iterator/go v1.1.12/go.mod h1:e30LSqwooZae/UwlEbR2852Gd8hjQvJoHmT4TnhNGBo= | github.com/json-iterator/go v1.1.12/go.mod h1:e30LSqwooZae/UwlEbR2852Gd8hjQvJoHmT4TnhNGBo= | ||||||
| github.com/klauspost/compress v1.17.8 h1:YcnTYrq7MikUT7k0Yb5eceMmALQPYBW/Xltxn0NAMnU= | github.com/klauspost/compress v1.17.9 h1:6KIumPrER1LHsvBVuDa0r5xaG0Es51mhhB9BQB2qeMA= | ||||||
| github.com/klauspost/compress v1.17.8/go.mod h1:Di0epgTjJY877eYKx5yC51cX2A2Vl2ibi7bDH9ttBbw= | github.com/klauspost/compress v1.17.9/go.mod h1:Di0epgTjJY877eYKx5yC51cX2A2Vl2ibi7bDH9ttBbw= | ||||||
| github.com/klauspost/cpuid/v2 v2.0.9/go.mod h1:FInQzS24/EEf25PyTYn52gqo7WaD8xa0213Md/qVLRg= | github.com/klauspost/cpuid/v2 v2.0.9/go.mod h1:FInQzS24/EEf25PyTYn52gqo7WaD8xa0213Md/qVLRg= | ||||||
| github.com/klauspost/cpuid/v2 v2.2.7 h1:ZWSB3igEs+d0qvnxR/ZBzXVmxkgt8DdzP6m9pfuVLDM= | github.com/klauspost/cpuid/v2 v2.2.8 h1:+StwCXwm9PdpiEkPyzBXIy+M9KUb4ODm0Zarf1kS5BM= | ||||||
| github.com/klauspost/cpuid/v2 v2.2.7/go.mod h1:Lcz8mBdAVJIBVzewtcLocK12l3Y+JytZYpaMropDUws= | github.com/klauspost/cpuid/v2 v2.2.8/go.mod h1:Lcz8mBdAVJIBVzewtcLocK12l3Y+JytZYpaMropDUws= | ||||||
| github.com/knz/go-libedit v1.10.1/go.mod h1:MZTVkCWyz0oBc7JOWP3wNAzd002ZbM/5hgShxwh4x8M= | github.com/knz/go-libedit v1.10.1/go.mod h1:MZTVkCWyz0oBc7JOWP3wNAzd002ZbM/5hgShxwh4x8M= | ||||||
| github.com/leodido/go-urn v1.4.0 h1:WT9HwE9SGECu3lg4d/dIA+jxlljEa1/ffXKmRjqdmIQ= | github.com/leodido/go-urn v1.4.0 h1:WT9HwE9SGECu3lg4d/dIA+jxlljEa1/ffXKmRjqdmIQ= | ||||||
| github.com/leodido/go-urn v1.4.0/go.mod h1:bvxc+MVxLKB4z00jd1z+Dvzr47oO32F/QSNjSBOlFxI= | github.com/leodido/go-urn v1.4.0/go.mod h1:bvxc+MVxLKB4z00jd1z+Dvzr47oO32F/QSNjSBOlFxI= | ||||||
| @@ -71,63 +74,69 @@ github.com/modern-go/reflect2 v1.0.2 h1:xBagoLtFs94CBntxluKeaWgTMpvLxC4ur3nMaC9G | |||||||
| github.com/modern-go/reflect2 v1.0.2/go.mod h1:yWuevngMOJpCy52FWWMvUC8ws7m/LJsjYzDa0/r8luk= | github.com/modern-go/reflect2 v1.0.2/go.mod h1:yWuevngMOJpCy52FWWMvUC8ws7m/LJsjYzDa0/r8luk= | ||||||
| github.com/montanaflynn/stats v0.7.1 h1:etflOAAHORrCC44V+aR6Ftzort912ZU+YLiSTuV8eaE= | github.com/montanaflynn/stats v0.7.1 h1:etflOAAHORrCC44V+aR6Ftzort912ZU+YLiSTuV8eaE= | ||||||
| github.com/montanaflynn/stats v0.7.1/go.mod h1:etXPPgVO6n31NxCd9KQUMvCM+ve0ruNzt6R8Bnaayow= | github.com/montanaflynn/stats v0.7.1/go.mod h1:etXPPgVO6n31NxCd9KQUMvCM+ve0ruNzt6R8Bnaayow= | ||||||
| github.com/pelletier/go-toml/v2 v2.2.2 h1:aYUidT7k73Pcl9nb2gScu7NSrKCSHIDE89b3+6Wq+LM= | github.com/pelletier/go-toml/v2 v2.2.3 h1:YmeHyLY8mFWbdkNWwpr+qIL2bEqT0o95WSdkNHvL12M= | ||||||
| github.com/pelletier/go-toml/v2 v2.2.2/go.mod h1:1t835xjRzz80PqgE6HHgN2JOsmgYu/h4qDAS4n929Rs= | github.com/pelletier/go-toml/v2 v2.2.3/go.mod h1:MfCQTFTvCcUyyvvwm1+G6H/jORL20Xlb6rzQu9GuUkc= | ||||||
| github.com/pkg/errors v0.9.1/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0= | github.com/pkg/errors v0.9.1/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0= | ||||||
| github.com/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZbAQM= | github.com/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZbAQM= | ||||||
| github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4= | github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4= | ||||||
| github.com/remyoudompheng/bigfft v0.0.0-20230129092748-24d4a6f8daec h1:W09IVJc94icq4NjY3clb7Lk8O1qJ8BdBEF8z0ibU0rE= | github.com/remyoudompheng/bigfft v0.0.0-20230129092748-24d4a6f8daec h1:W09IVJc94icq4NjY3clb7Lk8O1qJ8BdBEF8z0ibU0rE= | ||||||
| github.com/remyoudompheng/bigfft v0.0.0-20230129092748-24d4a6f8daec/go.mod h1:qqbHyh8v60DhA7CoWK5oRCqLrMHRGoxYCSS9EjAz6Eo= | github.com/remyoudompheng/bigfft v0.0.0-20230129092748-24d4a6f8daec/go.mod h1:qqbHyh8v60DhA7CoWK5oRCqLrMHRGoxYCSS9EjAz6Eo= | ||||||
| github.com/rs/xid v1.5.0 h1:mKX4bl4iPYJtEIxp6CYiUuLQ/8DYMoz0PUdtGgMFRVc= |  | ||||||
| github.com/rs/xid v1.5.0/go.mod h1:trrq9SKmegXys3aeAKXMUTdJsYXVwGY3RLcfgqegfbg= | github.com/rs/xid v1.5.0/go.mod h1:trrq9SKmegXys3aeAKXMUTdJsYXVwGY3RLcfgqegfbg= | ||||||
|  | github.com/rs/xid v1.6.0 h1:fV591PaemRlL6JfRxGDEPl69wICngIQ3shQtzfy2gxU= | ||||||
|  | github.com/rs/xid v1.6.0/go.mod h1:7XoLgs4eV+QndskICGsho+ADou8ySMSjJKDIan90Nz0= | ||||||
| github.com/rs/zerolog v1.33.0 h1:1cU2KZkvPxNyfgEmhHAz/1A9Bz+llsdYzklWFzgp0r8= | github.com/rs/zerolog v1.33.0 h1:1cU2KZkvPxNyfgEmhHAz/1A9Bz+llsdYzklWFzgp0r8= | ||||||
| github.com/rs/zerolog v1.33.0/go.mod h1:/7mN4D5sKwJLZQ2b/znpjC3/GQWY/xaDXUM0kKWRHss= | github.com/rs/zerolog v1.33.0/go.mod h1:/7mN4D5sKwJLZQ2b/znpjC3/GQWY/xaDXUM0kKWRHss= | ||||||
| github.com/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME= | github.com/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME= | ||||||
| github.com/stretchr/objx v0.4.0/go.mod h1:YvHI0jy2hoMjB+UWwv71VJQ9isScKT/TqJzVSSt89Yw= | github.com/stretchr/objx v0.4.0/go.mod h1:YvHI0jy2hoMjB+UWwv71VJQ9isScKT/TqJzVSSt89Yw= | ||||||
| github.com/stretchr/objx v0.5.0/go.mod h1:Yh+to48EsGEfYuaHDzXPcE3xhTkx73EhmCGUpEOglKo= | github.com/stretchr/objx v0.5.0/go.mod h1:Yh+to48EsGEfYuaHDzXPcE3xhTkx73EhmCGUpEOglKo= | ||||||
| github.com/stretchr/objx v0.5.2/go.mod h1:FRsXN1f5AsAjCGJKqEizvkpNtU+EGNCLh3NxZ/8L+MA= |  | ||||||
| github.com/stretchr/testify v1.3.0/go.mod h1:M5WIy9Dh21IEIfnGCwXGc5bZfKNJtfHm1UVUgZn+9EI= | github.com/stretchr/testify v1.3.0/go.mod h1:M5WIy9Dh21IEIfnGCwXGc5bZfKNJtfHm1UVUgZn+9EI= | ||||||
|  | github.com/stretchr/testify v1.5.1/go.mod h1:5W2xD1RspED5o8YsWQXVCued0rvSQ+mT+I5cxcmMvtA= | ||||||
| github.com/stretchr/testify v1.7.0/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg= | github.com/stretchr/testify v1.7.0/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg= | ||||||
| github.com/stretchr/testify v1.7.1/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg= | github.com/stretchr/testify v1.7.1/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg= | ||||||
| github.com/stretchr/testify v1.8.0/go.mod h1:yNjHg4UonilssWZ8iaSj1OCr/vHnekPRkoO+kdMU+MU= | github.com/stretchr/testify v1.8.0/go.mod h1:yNjHg4UonilssWZ8iaSj1OCr/vHnekPRkoO+kdMU+MU= | ||||||
| github.com/stretchr/testify v1.8.1/go.mod h1:w2LPCIKwWwSfY2zedu0+kehJoqGctiVI29o6fzry7u4= | github.com/stretchr/testify v1.8.1/go.mod h1:w2LPCIKwWwSfY2zedu0+kehJoqGctiVI29o6fzry7u4= | ||||||
| github.com/stretchr/testify v1.8.4/go.mod h1:sz/lmYIOXD/1dqDmKjjqLyZ2RngseejIcXlSw2iwfAo= |  | ||||||
| github.com/stretchr/testify v1.9.0 h1:HtqpIVDClZ4nwg75+f6Lvsy/wHu+3BoSGCbBAcpTsTg= | github.com/stretchr/testify v1.9.0 h1:HtqpIVDClZ4nwg75+f6Lvsy/wHu+3BoSGCbBAcpTsTg= | ||||||
| github.com/stretchr/testify v1.9.0/go.mod h1:r2ic/lqez/lEtzL7wO/rwa5dbSLXVDPFyf8C91i36aY= | github.com/stretchr/testify v1.9.0/go.mod h1:r2ic/lqez/lEtzL7wO/rwa5dbSLXVDPFyf8C91i36aY= | ||||||
| github.com/twitchyliquid64/golang-asm v0.15.1 h1:SU5vSMR7hnwNxj24w34ZyCi/FmDZTkS4MhqMhdFk5YI= | github.com/twitchyliquid64/golang-asm v0.15.1 h1:SU5vSMR7hnwNxj24w34ZyCi/FmDZTkS4MhqMhdFk5YI= | ||||||
| github.com/twitchyliquid64/golang-asm v0.15.1/go.mod h1:a1lVb/DtPvCB8fslRZhAngC2+aY1QWCk3Cedj/Gdt08= | github.com/twitchyliquid64/golang-asm v0.15.1/go.mod h1:a1lVb/DtPvCB8fslRZhAngC2+aY1QWCk3Cedj/Gdt08= | ||||||
| github.com/ugorji/go/codec v1.2.12 h1:9LC83zGrHhuUA9l16C9AHXAqEV/2wBQ4nkvumAE65EE= | github.com/ugorji/go/codec v1.2.12 h1:9LC83zGrHhuUA9l16C9AHXAqEV/2wBQ4nkvumAE65EE= | ||||||
| github.com/ugorji/go/codec v1.2.12/go.mod h1:UNopzCgEMSXjBc6AOMqYvWC1ktqTAfzJZUZgYf6w6lg= | github.com/ugorji/go/codec v1.2.12/go.mod h1:UNopzCgEMSXjBc6AOMqYvWC1ktqTAfzJZUZgYf6w6lg= | ||||||
|  | github.com/viney-shih/go-lock v1.1.2 h1:3TdGTiHZCPqBdTvFbQZQN/TRZzKF3KWw2rFEyKz3YqA= | ||||||
|  | github.com/viney-shih/go-lock v1.1.2/go.mod h1:Yijm78Ljteb3kRiJrbLAxVntkUukGu5uzSxq/xV7OO8= | ||||||
| github.com/xdg-go/pbkdf2 v1.0.0 h1:Su7DPu48wXMwC3bs7MCNG+z4FhcyEuz5dlvchbq0B0c= | github.com/xdg-go/pbkdf2 v1.0.0 h1:Su7DPu48wXMwC3bs7MCNG+z4FhcyEuz5dlvchbq0B0c= | ||||||
| github.com/xdg-go/pbkdf2 v1.0.0/go.mod h1:jrpuAogTd400dnrH08LKmI/xc1MbPOebTwRqcT5RDeI= | github.com/xdg-go/pbkdf2 v1.0.0/go.mod h1:jrpuAogTd400dnrH08LKmI/xc1MbPOebTwRqcT5RDeI= | ||||||
| github.com/xdg-go/scram v1.1.2 h1:FHX5I5B4i4hKRVRBCFRxq1iQRej7WO3hhBuJf+UUySY= | github.com/xdg-go/scram v1.1.2 h1:FHX5I5B4i4hKRVRBCFRxq1iQRej7WO3hhBuJf+UUySY= | ||||||
| github.com/xdg-go/scram v1.1.2/go.mod h1:RT/sEzTbU5y00aCK8UOx6R7YryM0iF1N2MOmC3kKLN4= | github.com/xdg-go/scram v1.1.2/go.mod h1:RT/sEzTbU5y00aCK8UOx6R7YryM0iF1N2MOmC3kKLN4= | ||||||
| github.com/xdg-go/stringprep v1.0.4 h1:XLI/Ng3O1Atzq0oBs3TWm+5ZVgkq2aqdlvP9JtoZ6c8= | github.com/xdg-go/stringprep v1.0.4 h1:XLI/Ng3O1Atzq0oBs3TWm+5ZVgkq2aqdlvP9JtoZ6c8= | ||||||
| github.com/xdg-go/stringprep v1.0.4/go.mod h1:mPGuuIYwz7CmR2bT9j4GbQqutWS1zV24gijq1dTyGkM= | github.com/xdg-go/stringprep v1.0.4/go.mod h1:mPGuuIYwz7CmR2bT9j4GbQqutWS1zV24gijq1dTyGkM= | ||||||
| github.com/youmark/pkcs8 v0.0.0-20240424034433-3c2c7870ae76 h1:tBiBTKHnIjovYoLX/TPkcf+OjqqKGQrPtGT3Foz+Pgo= | github.com/youmark/pkcs8 v0.0.0-20240726163527-a2c0da244d78 h1:ilQV1hzziu+LLM3zUTJ0trRztfwgjqKnBWNtSRkbmwM= | ||||||
| github.com/youmark/pkcs8 v0.0.0-20240424034433-3c2c7870ae76/go.mod h1:SQliXeA7Dhkt//vS29v3zpbEwoa+zb2Cn5xj5uO4K5U= | github.com/youmark/pkcs8 v0.0.0-20240726163527-a2c0da244d78/go.mod h1:aL8wCCfTfSfmXjznFBSZNN13rSJjlIOI1fUNAtF7rmI= | ||||||
| github.com/yuin/goldmark v1.4.13/go.mod h1:6yULJ656Px+3vBD8DxQVa3kxgyrAnzto9xy5taEt/CY= | github.com/yuin/goldmark v1.4.13/go.mod h1:6yULJ656Px+3vBD8DxQVa3kxgyrAnzto9xy5taEt/CY= | ||||||
| go.mongodb.org/mongo-driver v1.15.0 h1:rJCKC8eEliewXjZGf0ddURtl7tTVy1TK3bfl0gkUSLc= | go.mongodb.org/mongo-driver v1.16.1 h1:rIVLL3q0IHM39dvE+z2ulZLp9ENZKThVfuvN/IiN4l8= | ||||||
| go.mongodb.org/mongo-driver v1.15.0/go.mod h1:Vzb0Mk/pa7e6cWw85R4F/endUC3u0U9jGcNU603k65c= | go.mongodb.org/mongo-driver v1.16.1/go.mod h1:oB6AhJQvFQL4LEHyXi6aJzQJtBiTQHiAd83l0GdFaiw= | ||||||
| gogs.mikescher.com/BlackForestBytes/goext v0.0.463 h1:1sdU/jI7gzzucKv3CBefT1Hk5frGAYvgl/ItC9PdoqA= | gogs.mikescher.com/BlackForestBytes/goext v0.0.511 h1:vAEhXdexKlLTNf/mGHzemp/4rzmv7n2jf5l4NK38tIw= | ||||||
| gogs.mikescher.com/BlackForestBytes/goext v0.0.463/go.mod h1:ZEaw70t0Wx044Ifkt8fcDHO/KtD3dwgxclX3OF6ElvA= | gogs.mikescher.com/BlackForestBytes/goext v0.0.511/go.mod h1:9Q9EjraeE3yih7EXgBlnwLLJXWuRZNsl7s5TVTh3aOU= | ||||||
| golang.org/x/arch v0.0.0-20210923205945-b76863e36670/go.mod h1:5om86z9Hs0C8fWVUuoMHwpExlXzs5Tkyp9hOrfG7pp8= | gogs.mikescher.com/BlackForestBytes/goext v0.0.512 h1:cdLUi1bSnGujtx8/K0fPql142aOvUyNPt+8aWMKKDFk= | ||||||
| golang.org/x/arch v0.8.0 h1:3wRIsP3pM4yUptoR96otTUOXI367OS0+c9eeRi9doIc= | gogs.mikescher.com/BlackForestBytes/goext v0.0.512/go.mod h1:9Q9EjraeE3yih7EXgBlnwLLJXWuRZNsl7s5TVTh3aOU= | ||||||
| golang.org/x/arch v0.8.0/go.mod h1:FEVrYAQjsQXMVJ1nsMoVVXPZg6p2JE2mx8psSWTDQys= | gogs.mikescher.com/BlackForestBytes/goext v0.0.513 h1:zGb5n220AYNElzQs611RYXfZlnUw6/VJJesfLftphkQ= | ||||||
|  | gogs.mikescher.com/BlackForestBytes/goext v0.0.513/go.mod h1:9Q9EjraeE3yih7EXgBlnwLLJXWuRZNsl7s5TVTh3aOU= | ||||||
|  | golang.org/x/arch v0.10.0 h1:S3huipmSclq3PJMNe76NGwkBR504WFkQ5dhzWzP8ZW8= | ||||||
|  | golang.org/x/arch v0.10.0/go.mod h1:FEVrYAQjsQXMVJ1nsMoVVXPZg6p2JE2mx8psSWTDQys= | ||||||
| golang.org/x/crypto v0.0.0-20190308221718-c2843e01d9a2/go.mod h1:djNgcEr1/C05ACkg1iLfiJU5Ep61QUkGW8qpdssI0+w= | golang.org/x/crypto v0.0.0-20190308221718-c2843e01d9a2/go.mod h1:djNgcEr1/C05ACkg1iLfiJU5Ep61QUkGW8qpdssI0+w= | ||||||
| golang.org/x/crypto v0.0.0-20210921155107-089bfa567519/go.mod h1:GvvjBRRGRdwPK5ydBHafDWAxML/pGHZbMvKqRZ5+Abc= | golang.org/x/crypto v0.0.0-20210921155107-089bfa567519/go.mod h1:GvvjBRRGRdwPK5ydBHafDWAxML/pGHZbMvKqRZ5+Abc= | ||||||
| golang.org/x/crypto v0.23.0 h1:dIJU/v2J8Mdglj/8rJ6UUOM3Zc9zLZxVZwwxMooUSAI= | golang.org/x/crypto v0.27.0 h1:GXm2NjJrPaiv/h1tb2UH8QfgC/hOf/+z0p6PT8o1w7A= | ||||||
| golang.org/x/crypto v0.23.0/go.mod h1:CKFgDieR+mRhux2Lsu27y0fO304Db0wZe70UKqHu0v8= | golang.org/x/crypto v0.27.0/go.mod h1:1Xngt8kV6Dvbssa53Ziq6Eqn0HqbZi5Z6R0ZpwQzt70= | ||||||
| golang.org/x/mod v0.6.0-dev.0.20220419223038-86c51ed26bb4/go.mod h1:jJ57K6gSWd91VN4djpZkiMVwK6gcyfeH4XE8wZrZaV4= | golang.org/x/mod v0.6.0-dev.0.20220419223038-86c51ed26bb4/go.mod h1:jJ57K6gSWd91VN4djpZkiMVwK6gcyfeH4XE8wZrZaV4= | ||||||
| golang.org/x/net v0.0.0-20190620200207-3b0461eec859/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= | golang.org/x/net v0.0.0-20190620200207-3b0461eec859/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= | ||||||
| golang.org/x/net v0.0.0-20210226172049-e18ecbb05110/go.mod h1:m0MpNAwzfU5UDzcl9v0D8zg8gWTRqZa9RBIspLL5mdg= | golang.org/x/net v0.0.0-20210226172049-e18ecbb05110/go.mod h1:m0MpNAwzfU5UDzcl9v0D8zg8gWTRqZa9RBIspLL5mdg= | ||||||
| golang.org/x/net v0.0.0-20220722155237-a158d28d115b/go.mod h1:XRhObCWvk6IyKnWLug+ECip1KBveYUHfp+8e9klMJ9c= | golang.org/x/net v0.0.0-20220722155237-a158d28d115b/go.mod h1:XRhObCWvk6IyKnWLug+ECip1KBveYUHfp+8e9klMJ9c= | ||||||
| golang.org/x/net v0.25.0 h1:d/OCCoBEUq33pjydKrGQhw7IlUPI2Oylr+8qLx49kac= | golang.org/x/net v0.29.0 h1:5ORfpBpCs4HzDYoodCDBbwHzdR5UrLBZ3sOnUJmFoHo= | ||||||
| golang.org/x/net v0.25.0/go.mod h1:JkAGAh7GEvH74S6FOH42FLoXpXbE/aqXSrIQjXgsiwM= | golang.org/x/net v0.29.0/go.mod h1:gLkgy8jTGERgjzMic6DS9+SP0ajcu6Xu3Orq/SpETg0= | ||||||
| golang.org/x/sync v0.0.0-20190423024810-112230192c58/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= | golang.org/x/sync v0.0.0-20190423024810-112230192c58/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= | ||||||
|  | golang.org/x/sync v0.0.0-20200317015054-43a5402ce75a/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= | ||||||
| golang.org/x/sync v0.0.0-20220722155255-886fb9371eb4/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= | golang.org/x/sync v0.0.0-20220722155255-886fb9371eb4/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= | ||||||
| golang.org/x/sync v0.7.0 h1:YsImfSBoP9QPYL0xyKJPq0gcaJdG3rInoqxTWbfQu9M= | golang.org/x/sync v0.8.0 h1:3NFvSEYkUoMifnESzZl15y791HH1qU2xm6eCJU5ZPXQ= | ||||||
| golang.org/x/sync v0.7.0/go.mod h1:Czt+wKu1gCyEFDUtn0jG5QVvpJ6rzVqr5aXyt9drQfk= | golang.org/x/sync v0.8.0/go.mod h1:Czt+wKu1gCyEFDUtn0jG5QVvpJ6rzVqr5aXyt9drQfk= | ||||||
| golang.org/x/sys v0.0.0-20190215142949-d0b11bdaac8a/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= | golang.org/x/sys v0.0.0-20190215142949-d0b11bdaac8a/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= | ||||||
| golang.org/x/sys v0.0.0-20201119102817-f84b799fce68/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | golang.org/x/sys v0.0.0-20201119102817-f84b799fce68/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||||
| golang.org/x/sys v0.0.0-20210615035016-665e8c7367d1/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | golang.org/x/sys v0.0.0-20210615035016-665e8c7367d1/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | ||||||
| @@ -137,28 +146,29 @@ golang.org/x/sys v0.0.0-20220811171246-fbc7d0a398ab/go.mod h1:oPkhp1MJrh7nUepCBc | |||||||
| golang.org/x/sys v0.5.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | golang.org/x/sys v0.5.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | ||||||
| golang.org/x/sys v0.6.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | golang.org/x/sys v0.6.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | ||||||
| golang.org/x/sys v0.12.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | golang.org/x/sys v0.12.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | ||||||
| golang.org/x/sys v0.20.0 h1:Od9JTbYCk261bKm4M/mw7AklTlFYIa0bIp9BgSm1S8Y= | golang.org/x/sys v0.25.0 h1:r+8e+loiHxRqhXVl6ML1nO3l1+oFoWbnlu2Ehimmi34= | ||||||
| golang.org/x/sys v0.20.0/go.mod h1:/VUhepiaJMQUp4+oa/7Zr1D23ma6VTLIYjOOTFZPUcA= | golang.org/x/sys v0.25.0/go.mod h1:/VUhepiaJMQUp4+oa/7Zr1D23ma6VTLIYjOOTFZPUcA= | ||||||
| golang.org/x/term v0.0.0-20201126162022-7de9c90e9dd1/go.mod h1:bj7SfCRtBDWHUb9snDiAeCFNEtKQo2Wmx5Cou7ajbmo= | golang.org/x/term v0.0.0-20201126162022-7de9c90e9dd1/go.mod h1:bj7SfCRtBDWHUb9snDiAeCFNEtKQo2Wmx5Cou7ajbmo= | ||||||
| golang.org/x/term v0.0.0-20210927222741-03fcf44c2211/go.mod h1:jbD1KX2456YbFQfuXm/mYQcufACuNUgVhRMnK/tPxf8= | golang.org/x/term v0.0.0-20210927222741-03fcf44c2211/go.mod h1:jbD1KX2456YbFQfuXm/mYQcufACuNUgVhRMnK/tPxf8= | ||||||
| golang.org/x/term v0.20.0 h1:VnkxpohqXaOBYJtBmEppKUG6mXpi+4O6purfc2+sMhw= | golang.org/x/term v0.24.0 h1:Mh5cbb+Zk2hqqXNO7S1iTjEphVL+jb8ZWaqh/g+JWkM= | ||||||
| golang.org/x/term v0.20.0/go.mod h1:8UkIAJTvZgivsXaD6/pH6U9ecQzZ45awqEOzuCvwpFY= | golang.org/x/term v0.24.0/go.mod h1:lOBK/LVxemqiMij05LGJ0tzNr8xlmwBRJ81PX6wVLH8= | ||||||
| golang.org/x/text v0.3.0/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ= | golang.org/x/text v0.3.0/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ= | ||||||
| golang.org/x/text v0.3.3/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ= | golang.org/x/text v0.3.3/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ= | ||||||
| golang.org/x/text v0.3.7/go.mod h1:u+2+/6zg+i71rQMx5EYifcz6MCKuco9NR6JIITiCfzQ= | golang.org/x/text v0.3.7/go.mod h1:u+2+/6zg+i71rQMx5EYifcz6MCKuco9NR6JIITiCfzQ= | ||||||
| golang.org/x/text v0.3.8/go.mod h1:E6s5w1FMmriuDzIBO73fBruAKo1PCIq6d2Q6DHfQ8WQ= | golang.org/x/text v0.3.8/go.mod h1:E6s5w1FMmriuDzIBO73fBruAKo1PCIq6d2Q6DHfQ8WQ= | ||||||
| golang.org/x/text v0.15.0 h1:h1V/4gjBv8v9cjcR6+AR5+/cIYK5N/WAgiv4xlsEtAk= | golang.org/x/text v0.18.0 h1:XvMDiNzPAl0jr17s6W9lcaIhGUfUORdGCNsuLmPG224= | ||||||
| golang.org/x/text v0.15.0/go.mod h1:18ZOQIKpY8NJVqYksKHtTdi31H5itFRjB5/qKTNYzSU= | golang.org/x/text v0.18.0/go.mod h1:BuEKDfySbSR4drPmRPG/7iBdf8hvFMuRexcpahXilzY= | ||||||
| golang.org/x/tools v0.0.0-20180917221912-90fa682c2a6e/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ= | golang.org/x/tools v0.0.0-20180917221912-90fa682c2a6e/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ= | ||||||
| golang.org/x/tools v0.0.0-20191119224855-298f0cb1881e/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo= | golang.org/x/tools v0.0.0-20191119224855-298f0cb1881e/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo= | ||||||
| golang.org/x/tools v0.1.12/go.mod h1:hNGJHUnrk76NpqgfD5Aqm5Crs+Hm0VOH/i9J2+nxYbc= | golang.org/x/tools v0.1.12/go.mod h1:hNGJHUnrk76NpqgfD5Aqm5Crs+Hm0VOH/i9J2+nxYbc= | ||||||
| golang.org/x/xerrors v0.0.0-20190717185122-a985d3407aa7/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= | golang.org/x/xerrors v0.0.0-20190717185122-a985d3407aa7/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= | ||||||
| google.golang.org/protobuf v1.34.1 h1:9ddQBjfCyZPOHPUiPxpYESBLc+T8P3E+Vo4IbKZgFWg= | google.golang.org/protobuf v1.34.2 h1:6xV6lTsCfpGD21XK49h7MhtcApnLqkfYgPcdHftf6hg= | ||||||
| google.golang.org/protobuf v1.34.1/go.mod h1:c6P6GXX6sHbq/GpV6MGZEdwhWPcYBgnhAHhKbcUYpos= | google.golang.org/protobuf v1.34.2/go.mod h1:qYOHts0dSfpeUzUFpOMr/WGzszTmLH+DiWniOlNbLDw= | ||||||
| gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405 h1:yhCVgyC4o1eVCa2tZl7eS0r+SDo693bJlVdllGtEeKM= | gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405 h1:yhCVgyC4o1eVCa2tZl7eS0r+SDo693bJlVdllGtEeKM= | ||||||
| gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= | gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= | ||||||
| gopkg.in/loremipsum.v1 v1.1.2 h1:12APklfJKuGszqZsrArW5QoQh03/W+qyCCjvnDuS6Tw= | gopkg.in/loremipsum.v1 v1.1.2 h1:12APklfJKuGszqZsrArW5QoQh03/W+qyCCjvnDuS6Tw= | ||||||
| gopkg.in/loremipsum.v1 v1.1.2/go.mod h1:TuRvzFuzuejXj+odBU6Tubp/EPUyGb9wmSvHenyP2Ts= | gopkg.in/loremipsum.v1 v1.1.2/go.mod h1:TuRvzFuzuejXj+odBU6Tubp/EPUyGb9wmSvHenyP2Ts= | ||||||
|  | gopkg.in/yaml.v2 v2.2.2/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= | ||||||
| gopkg.in/yaml.v3 v3.0.0-20200313102051-9f266ea9e77c/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= | gopkg.in/yaml.v3 v3.0.0-20200313102051-9f266ea9e77c/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= | ||||||
| gopkg.in/yaml.v3 v3.0.1 h1:fxVm/GzAzEWqLHuvctI91KS9hhNmmWOoWu0XTYJS7CA= | gopkg.in/yaml.v3 v3.0.1 h1:fxVm/GzAzEWqLHuvctI91KS9hhNmmWOoWu0XTYJS7CA= | ||||||
| gopkg.in/yaml.v3 v3.0.1/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= | gopkg.in/yaml.v3 v3.0.1/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= | ||||||
| @@ -171,4 +181,3 @@ modernc.org/memory v1.7.2/go.mod h1:NO4NVCQy0N7ln+T9ngWqOQfi7ley4vpwvARR+Hjw95E= | |||||||
| modernc.org/sqlite v1.28.0 h1:Zx+LyDDmXczNnEQdvPuEfcFVA2ZPyaD7UCZDjef3BHQ= | modernc.org/sqlite v1.28.0 h1:Zx+LyDDmXczNnEQdvPuEfcFVA2ZPyaD7UCZDjef3BHQ= | ||||||
| modernc.org/sqlite v1.28.0/go.mod h1:Qxpazz0zH8Z1xCFyi5GSL3FzbtZ3fvbjmywNogldEW0= | modernc.org/sqlite v1.28.0/go.mod h1:Qxpazz0zH8Z1xCFyi5GSL3FzbtZ3fvbjmywNogldEW0= | ||||||
| nullprogram.com/x/optparse v1.0.0/go.mod h1:KdyPE+Igbe0jQUrVfMqDMeJQIJZEuyV7pjYmp6pbG50= | nullprogram.com/x/optparse v1.0.0/go.mod h1:KdyPE+Igbe0jQUrVfMqDMeJQIJZEuyV7pjYmp6pbG50= | ||||||
| rsc.io/pdf v0.1.1/go.mod h1:n8OzWcQ6Sp37PL01nO98y4iUCRdTGarVfzxY20ICaU4= |  | ||||||
|   | |||||||
| @@ -9,6 +9,7 @@ import ( | |||||||
| 	"errors" | 	"errors" | ||||||
| 	"github.com/gin-gonic/gin" | 	"github.com/gin-gonic/gin" | ||||||
| 	"github.com/rs/zerolog/log" | 	"github.com/rs/zerolog/log" | ||||||
|  | 	"gogs.mikescher.com/BlackForestBytes/goext/ginext" | ||||||
| 	"gogs.mikescher.com/BlackForestBytes/goext/sq" | 	"gogs.mikescher.com/BlackForestBytes/goext/sq" | ||||||
| 	"time" | 	"time" | ||||||
| ) | ) | ||||||
| @@ -70,7 +71,10 @@ func (ac *AppContext) Cancel() { | |||||||
| 		} | 		} | ||||||
| 		ac.transaction = nil | 		ac.transaction = nil | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
|  | 	if ac.cancelFunc != nil { | ||||||
| 		ac.cancelFunc() | 		ac.cancelFunc() | ||||||
|  | 	} | ||||||
| } | } | ||||||
|  |  | ||||||
| func (ac *AppContext) RequestURI() string { | func (ac *AppContext) RequestURI() string { | ||||||
| @@ -81,7 +85,7 @@ func (ac *AppContext) RequestURI() string { | |||||||
| 	} | 	} | ||||||
| } | } | ||||||
|  |  | ||||||
| func (ac *AppContext) FinishSuccess(res ginresp.HTTPResponse) ginresp.HTTPResponse { | func (ac *AppContext) _FinishSuccess(res ginext.HTTPResponse) ginext.HTTPResponse { | ||||||
| 	if ac.cancelled { | 	if ac.cancelled { | ||||||
| 		panic("Cannot finish a cancelled request") | 		panic("Cannot finish a cancelled request") | ||||||
| 	} | 	} | ||||||
|   | |||||||
| @@ -2,22 +2,19 @@ package logic | |||||||
|  |  | ||||||
| import ( | import ( | ||||||
| 	scn "blackforestbytes.com/simplecloudnotifier" | 	scn "blackforestbytes.com/simplecloudnotifier" | ||||||
| 	"blackforestbytes.com/simplecloudnotifier/api/apierr" | 	"blackforestbytes.com/simplecloudnotifier/db" | ||||||
| 	"blackforestbytes.com/simplecloudnotifier/api/ginresp" |  | ||||||
| 	"blackforestbytes.com/simplecloudnotifier/db/simplectx" | 	"blackforestbytes.com/simplecloudnotifier/db/simplectx" | ||||||
| 	"blackforestbytes.com/simplecloudnotifier/google" | 	"blackforestbytes.com/simplecloudnotifier/google" | ||||||
| 	"blackforestbytes.com/simplecloudnotifier/models" | 	"blackforestbytes.com/simplecloudnotifier/models" | ||||||
| 	"blackforestbytes.com/simplecloudnotifier/push" | 	"blackforestbytes.com/simplecloudnotifier/push" | ||||||
| 	"context" | 	"context" | ||||||
| 	"errors" | 	"errors" | ||||||
| 	"github.com/gin-gonic/gin" |  | ||||||
| 	"github.com/gin-gonic/gin/binding" |  | ||||||
| 	"github.com/rs/zerolog/log" | 	"github.com/rs/zerolog/log" | ||||||
| 	"gogs.mikescher.com/BlackForestBytes/goext/langext" | 	golock "github.com/viney-shih/go-lock" | ||||||
|  | 	"gogs.mikescher.com/BlackForestBytes/goext/ginext" | ||||||
| 	"gogs.mikescher.com/BlackForestBytes/goext/rext" | 	"gogs.mikescher.com/BlackForestBytes/goext/rext" | ||||||
| 	"gogs.mikescher.com/BlackForestBytes/goext/syncext" | 	"gogs.mikescher.com/BlackForestBytes/goext/syncext" | ||||||
| 	"net" | 	"net" | ||||||
| 	"net/http" |  | ||||||
| 	"os" | 	"os" | ||||||
| 	"os/signal" | 	"os/signal" | ||||||
| 	"regexp" | 	"regexp" | ||||||
| @@ -33,7 +30,7 @@ var rexCompatTitleChannel = rext.W(regexp.MustCompile("^\\[(?P<channel>[A-Za-z\\ | |||||||
|  |  | ||||||
| type Application struct { | type Application struct { | ||||||
| 	Config           scn.Config | 	Config           scn.Config | ||||||
| 	Gin              *gin.Engine | 	Gin              *ginext.GinWrapper | ||||||
| 	Database         *DBPool | 	Database         *DBPool | ||||||
| 	Pusher           push.NotificationClient | 	Pusher           push.NotificationClient | ||||||
| 	AndroidPublisher google.AndroidPublisherClient | 	AndroidPublisher google.AndroidPublisherClient | ||||||
| @@ -42,6 +39,7 @@ type Application struct { | |||||||
| 	Port             string | 	Port             string | ||||||
| 	IsRunning        *syncext.AtomicBool | 	IsRunning        *syncext.AtomicBool | ||||||
| 	RequestLogQueue  chan models.RequestLog | 	RequestLogQueue  chan models.RequestLog | ||||||
|  | 	MainDatabaseLock golock.RWMutex | ||||||
| } | } | ||||||
|  |  | ||||||
| func NewApp(db *DBPool) *Application { | func NewApp(db *DBPool) *Application { | ||||||
| @@ -50,10 +48,11 @@ func NewApp(db *DBPool) *Application { | |||||||
| 		stopChan:         make(chan bool), | 		stopChan:         make(chan bool), | ||||||
| 		IsRunning:        syncext.NewAtomicBool(false), | 		IsRunning:        syncext.NewAtomicBool(false), | ||||||
| 		RequestLogQueue:  make(chan models.RequestLog, 1024), | 		RequestLogQueue:  make(chan models.RequestLog, 1024), | ||||||
|  | 		MainDatabaseLock: golock.NewCASMutex(), | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
|  |  | ||||||
| func (app *Application) Init(cfg scn.Config, g *gin.Engine, fb push.NotificationClient, apc google.AndroidPublisherClient, jobs []Job) { | func (app *Application) Init(cfg scn.Config, g *ginext.GinWrapper, fb push.NotificationClient, apc google.AndroidPublisherClient, jobs []Job) { | ||||||
| 	app.Config = cfg | 	app.Config = cfg | ||||||
| 	app.Gin = g | 	app.Gin = g | ||||||
| 	app.Pusher = fb | 	app.Pusher = fb | ||||||
| @@ -69,38 +68,17 @@ func (app *Application) Stop() { | |||||||
| } | } | ||||||
|  |  | ||||||
| func (app *Application) Run() { | func (app *Application) Run() { | ||||||
| 	httpserver := &http.Server{ |  | ||||||
| 		Addr:    net.JoinHostPort(app.Config.ServerIP, app.Config.ServerPort), |  | ||||||
| 		Handler: app.Gin, |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	errChan := make(chan error) | 	// ================== START HTTP ================== | ||||||
|  |  | ||||||
| 	go func() { | 	addr := net.JoinHostPort(app.Config.ServerIP, app.Config.ServerPort) | ||||||
|  |  | ||||||
| 		ln, err := net.Listen("tcp", httpserver.Addr) |  | ||||||
| 		if err != nil { |  | ||||||
| 			errChan <- err |  | ||||||
| 			return |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		_, port, err := net.SplitHostPort(ln.Addr().String()) |  | ||||||
| 		if err != nil { |  | ||||||
| 			errChan <- err |  | ||||||
| 			return |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		log.Info().Str("address", httpserver.Addr).Msg("HTTP-Server started on http://localhost:" + port) |  | ||||||
|  |  | ||||||
|  | 	errChan, httpserver := app.Gin.ListenAndServeHTTP(addr, func(port string) { | ||||||
| 		app.Port = port | 		app.Port = port | ||||||
|  | 		app.IsRunning.Set(true) | ||||||
|  | 	}) | ||||||
|  |  | ||||||
| 		app.IsRunning.Set(true) // the net.Listener a few lines above is at this point actually already buffering requests | 	// ================== START JOBS ================== | ||||||
|  |  | ||||||
| 		errChan <- httpserver.Serve(ln) |  | ||||||
| 	}() |  | ||||||
|  |  | ||||||
| 	sigstop := make(chan os.Signal, 1) |  | ||||||
| 	signal.Notify(sigstop, os.Interrupt, syscall.SIGTERM) |  | ||||||
|  |  | ||||||
| 	for _, job := range app.Jobs { | 	for _, job := range app.Jobs { | ||||||
| 		err := job.Start() | 		err := job.Start() | ||||||
| @@ -109,6 +87,11 @@ func (app *Application) Run() { | |||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
|  | 	// ================== LISTEN FOR SIGNALS ================== | ||||||
|  |  | ||||||
|  | 	sigstop := make(chan os.Signal, 1) | ||||||
|  | 	signal.Notify(sigstop, os.Interrupt, syscall.SIGTERM) | ||||||
|  |  | ||||||
| 	select { | 	select { | ||||||
| 	case <-sigstop: | 	case <-sigstop: | ||||||
| 		ctx, cancel := context.WithTimeout(context.Background(), 30*time.Second) | 		ctx, cancel := context.WithTimeout(context.Background(), 30*time.Second) | ||||||
| @@ -127,7 +110,7 @@ func (app *Application) Run() { | |||||||
| 	case err := <-errChan: | 	case err := <-errChan: | ||||||
| 		log.Error().Err(err).Msg("HTTP-Server failed") | 		log.Error().Err(err).Msg("HTTP-Server failed") | ||||||
|  |  | ||||||
| 	case _ = <-app.stopChan: | 	case <-app.stopChan: | ||||||
| 		ctx, cancel := context.WithTimeout(context.Background(), 30*time.Second) | 		ctx, cancel := context.WithTimeout(context.Background(), 30*time.Second) | ||||||
| 		defer cancel() | 		defer cancel() | ||||||
|  |  | ||||||
| @@ -142,20 +125,25 @@ func (app *Application) Run() { | |||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
|  | 	// ================== STOP JOBS ================== | ||||||
|  |  | ||||||
| 	for _, job := range app.Jobs { | 	for _, job := range app.Jobs { | ||||||
| 		job.Stop() | 		job.Stop() | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	log.Info().Msg("Manually stopped Jobs") | 	// ================== STOP DB ================== | ||||||
|  |  | ||||||
|  | 	{ | ||||||
| 		ctx, cancel := context.WithTimeout(context.Background(), 30*time.Second) | 		ctx, cancel := context.WithTimeout(context.Background(), 30*time.Second) | ||||||
| 		defer cancel() | 		defer cancel() | ||||||
| 		err := app.Database.Stop(ctx) | 		err := app.Database.Stop(ctx) | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| 		log.Info().Err(err).Msg("Error while stopping the database") | 			log.Err(err).Msg("Failed to stop database") | ||||||
| 		} | 		} | ||||||
|  | 	} | ||||||
|  | 	log.Info().Msg("Stopped Databases") | ||||||
|  |  | ||||||
| 	log.Info().Msg("Manually closed database connection") | 	// ================== FINISH ================== | ||||||
|  |  | ||||||
| 	app.IsRunning.Set(false) | 	app.IsRunning.Set(false) | ||||||
| } | } | ||||||
| @@ -219,77 +207,12 @@ func (app *Application) Migrate() error { | |||||||
| 	return app.Database.Migrate(ctx) | 	return app.Database.Migrate(ctx) | ||||||
| } | } | ||||||
|  |  | ||||||
| type RequestOptions struct { |  | ||||||
| 	IgnoreWrongContentType bool |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (app *Application) StartRequest(g *gin.Context, uri any, query any, body any, form any, opts ...RequestOptions) (*AppContext, *ginresp.HTTPResponse) { |  | ||||||
|  |  | ||||||
| 	ignoreWrongContentType := langext.ArrAny(opts, func(o RequestOptions) bool { return o.IgnoreWrongContentType }) |  | ||||||
|  |  | ||||||
| 	if uri != nil { |  | ||||||
| 		if err := g.ShouldBindUri(uri); err != nil { |  | ||||||
| 			return nil, langext.Ptr(ginresp.APIError(g, 400, apierr.BINDFAIL_URI_PARAM, "Failed to read uri", err)) |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	if query != nil { |  | ||||||
| 		if err := g.ShouldBindQuery(query); err != nil { |  | ||||||
| 			return nil, langext.Ptr(ginresp.APIError(g, 400, apierr.BINDFAIL_QUERY_PARAM, "Failed to read query", err)) |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	if body != nil { |  | ||||||
| 		if g.ContentType() == "application/json" { |  | ||||||
| 			if err := g.ShouldBindJSON(body); err != nil { |  | ||||||
| 				return nil, langext.Ptr(ginresp.APIError(g, 400, apierr.BINDFAIL_BODY_PARAM, "Failed to read body", err)) |  | ||||||
| 			} |  | ||||||
| 		} else { |  | ||||||
| 			if !ignoreWrongContentType { |  | ||||||
| 				return nil, langext.Ptr(ginresp.APIError(g, 400, apierr.BINDFAIL_BODY_PARAM, "missing JSON body", nil)) |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	if form != nil { |  | ||||||
| 		if g.ContentType() == "multipart/form-data" { |  | ||||||
| 			if err := g.ShouldBindWith(form, binding.Form); err != nil { |  | ||||||
| 				return nil, langext.Ptr(ginresp.APIError(g, 400, apierr.BINDFAIL_BODY_PARAM, "Failed to read multipart-form", err)) |  | ||||||
| 			} |  | ||||||
| 		} else if g.ContentType() == "application/x-www-form-urlencoded" { |  | ||||||
| 			if err := g.ShouldBindWith(form, binding.Form); err != nil { |  | ||||||
| 				return nil, langext.Ptr(ginresp.APIError(g, 400, apierr.BINDFAIL_BODY_PARAM, "Failed to read urlencoded-form", err)) |  | ||||||
| 			} |  | ||||||
| 		} else { |  | ||||||
| 			if !ignoreWrongContentType { |  | ||||||
| 				return nil, langext.Ptr(ginresp.APIError(g, 400, apierr.BINDFAIL_BODY_PARAM, "missing form body", nil)) |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	ictx, cancel := context.WithTimeout(context.Background(), app.Config.RequestTimeout) |  | ||||||
| 	actx := CreateAppContext(app, g, ictx, cancel) |  | ||||||
|  |  | ||||||
| 	authheader := g.GetHeader("Authorization") |  | ||||||
|  |  | ||||||
| 	perm, err := app.getPermissions(actx, authheader) |  | ||||||
| 	if err != nil { |  | ||||||
| 		cancel() |  | ||||||
| 		return nil, langext.Ptr(ginresp.APIError(g, 400, apierr.PERM_QUERY_FAIL, "Failed to determine permissions", err)) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	actx.permissions = perm |  | ||||||
| 	g.Set("perm", perm) |  | ||||||
|  |  | ||||||
| 	return actx, nil |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (app *Application) NewSimpleTransactionContext(timeout time.Duration) *simplectx.SimpleContext { | func (app *Application) NewSimpleTransactionContext(timeout time.Duration) *simplectx.SimpleContext { | ||||||
| 	ictx, cancel := context.WithTimeout(context.Background(), timeout) | 	ictx, cancel := context.WithTimeout(context.Background(), timeout) | ||||||
| 	return simplectx.CreateSimpleContext(ictx, cancel) | 	return simplectx.CreateSimpleContext(ictx, cancel) | ||||||
| } | } | ||||||
|  |  | ||||||
| func (app *Application) getPermissions(ctx *AppContext, hdr string) (models.PermissionSet, error) { | func (app *Application) getPermissions(ctx db.TxContext, hdr string) (models.PermissionSet, error) { | ||||||
| 	if hdr == "" { | 	if hdr == "" { | ||||||
| 		return models.NewEmptyPermissions(), nil | 		return models.NewEmptyPermissions(), nil | ||||||
| 	} | 	} | ||||||
|   | |||||||
| @@ -10,6 +10,7 @@ import ( | |||||||
| 	"fmt" | 	"fmt" | ||||||
| 	"github.com/gin-gonic/gin" | 	"github.com/gin-gonic/gin" | ||||||
| 	"github.com/rs/zerolog/log" | 	"github.com/rs/zerolog/log" | ||||||
|  | 	"gogs.mikescher.com/BlackForestBytes/goext/ginext" | ||||||
| 	"gogs.mikescher.com/BlackForestBytes/goext/langext" | 	"gogs.mikescher.com/BlackForestBytes/goext/langext" | ||||||
| 	"gogs.mikescher.com/BlackForestBytes/goext/mathext" | 	"gogs.mikescher.com/BlackForestBytes/goext/mathext" | ||||||
| 	"gogs.mikescher.com/BlackForestBytes/goext/timeext" | 	"gogs.mikescher.com/BlackForestBytes/goext/timeext" | ||||||
| @@ -24,7 +25,7 @@ type SendMessageResponse struct { | |||||||
| 	CompatMessageID int64 | 	CompatMessageID int64 | ||||||
| } | } | ||||||
|  |  | ||||||
| func (app *Application) SendMessage(g *gin.Context, ctx *AppContext, UserID *models.UserID, Key *string, Channel *string, Title *string, Content *string, Priority *int, UserMessageID *string, SendTimestamp *float64, SenderName *string) (*SendMessageResponse, *ginresp.HTTPResponse) { | func (app *Application) SendMessage(g *gin.Context, ctx *AppContext, UserID *models.UserID, Key *string, Channel *string, Title *string, Content *string, Priority *int, UserMessageID *string, SendTimestamp *float64, SenderName *string) (*SendMessageResponse, *ginext.HTTPResponse) { | ||||||
| 	if Title != nil { | 	if Title != nil { | ||||||
| 		Title = langext.Ptr(strings.TrimSpace(*Title)) | 		Title = langext.Ptr(strings.TrimSpace(*Title)) | ||||||
| 	} | 	} | ||||||
|   | |||||||
| @@ -6,10 +6,11 @@ import ( | |||||||
| 	"blackforestbytes.com/simplecloudnotifier/models" | 	"blackforestbytes.com/simplecloudnotifier/models" | ||||||
| 	"database/sql" | 	"database/sql" | ||||||
| 	"errors" | 	"errors" | ||||||
|  | 	"gogs.mikescher.com/BlackForestBytes/goext/ginext" | ||||||
| 	"gogs.mikescher.com/BlackForestBytes/goext/langext" | 	"gogs.mikescher.com/BlackForestBytes/goext/langext" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| func (ac *AppContext) CheckPermissionUserRead(userid models.UserID) *ginresp.HTTPResponse { | func (ac *AppContext) CheckPermissionUserRead(userid models.UserID) *ginext.HTTPResponse { | ||||||
| 	p := ac.permissions | 	p := ac.permissions | ||||||
| 	if p.Token != nil && p.Token.IsUserRead(userid) { | 	if p.Token != nil && p.Token.IsUserRead(userid) { | ||||||
| 		return nil | 		return nil | ||||||
| @@ -18,7 +19,7 @@ func (ac *AppContext) CheckPermissionUserRead(userid models.UserID) *ginresp.HTT | |||||||
| 	return langext.Ptr(ginresp.APIError(ac.ginContext, 401, apierr.USER_AUTH_FAILED, "You are not authorized for this action", nil)) | 	return langext.Ptr(ginresp.APIError(ac.ginContext, 401, apierr.USER_AUTH_FAILED, "You are not authorized for this action", nil)) | ||||||
| } | } | ||||||
|  |  | ||||||
| func (ac *AppContext) CheckPermissionSelfAllMessagesRead() *ginresp.HTTPResponse { | func (ac *AppContext) CheckPermissionSelfAllMessagesRead() *ginext.HTTPResponse { | ||||||
| 	p := ac.permissions | 	p := ac.permissions | ||||||
| 	if p.Token != nil && p.Token.IsAllMessagesRead(p.Token.OwnerUserID) { | 	if p.Token != nil && p.Token.IsAllMessagesRead(p.Token.OwnerUserID) { | ||||||
| 		return nil | 		return nil | ||||||
| @@ -27,7 +28,7 @@ func (ac *AppContext) CheckPermissionSelfAllMessagesRead() *ginresp.HTTPResponse | |||||||
| 	return langext.Ptr(ginresp.APIError(ac.ginContext, 401, apierr.USER_AUTH_FAILED, "You are not authorized for this action", nil)) | 	return langext.Ptr(ginresp.APIError(ac.ginContext, 401, apierr.USER_AUTH_FAILED, "You are not authorized for this action", nil)) | ||||||
| } | } | ||||||
|  |  | ||||||
| func (ac *AppContext) CheckPermissionAllMessagesRead(userid models.UserID) *ginresp.HTTPResponse { | func (ac *AppContext) CheckPermissionAllMessagesRead(userid models.UserID) *ginext.HTTPResponse { | ||||||
| 	p := ac.permissions | 	p := ac.permissions | ||||||
| 	if p.Token != nil && p.Token.IsAllMessagesRead(userid) { | 	if p.Token != nil && p.Token.IsAllMessagesRead(userid) { | ||||||
| 		return nil | 		return nil | ||||||
| @@ -36,7 +37,7 @@ func (ac *AppContext) CheckPermissionAllMessagesRead(userid models.UserID) *ginr | |||||||
| 	return langext.Ptr(ginresp.APIError(ac.ginContext, 401, apierr.USER_AUTH_FAILED, "You are not authorized for this action", nil)) | 	return langext.Ptr(ginresp.APIError(ac.ginContext, 401, apierr.USER_AUTH_FAILED, "You are not authorized for this action", nil)) | ||||||
| } | } | ||||||
|  |  | ||||||
| func (ac *AppContext) CheckPermissionChanMessagesRead(channel models.Channel) *ginresp.HTTPResponse { | func (ac *AppContext) CheckPermissionChanMessagesRead(channel models.Channel) *ginext.HTTPResponse { | ||||||
| 	p := ac.permissions | 	p := ac.permissions | ||||||
| 	if p.Token != nil && p.Token.IsChannelMessagesRead(channel.ChannelID) { | 	if p.Token != nil && p.Token.IsChannelMessagesRead(channel.ChannelID) { | ||||||
|  |  | ||||||
| @@ -63,7 +64,7 @@ func (ac *AppContext) CheckPermissionChanMessagesRead(channel models.Channel) *g | |||||||
| 	return langext.Ptr(ginresp.APIError(ac.ginContext, 401, apierr.USER_AUTH_FAILED, "You are not authorized for this action", nil)) | 	return langext.Ptr(ginresp.APIError(ac.ginContext, 401, apierr.USER_AUTH_FAILED, "You are not authorized for this action", nil)) | ||||||
| } | } | ||||||
|  |  | ||||||
| func (ac *AppContext) CheckPermissionUserAdmin(userid models.UserID) *ginresp.HTTPResponse { | func (ac *AppContext) CheckPermissionUserAdmin(userid models.UserID) *ginext.HTTPResponse { | ||||||
| 	p := ac.permissions | 	p := ac.permissions | ||||||
| 	if p.Token != nil && p.Token.IsAdmin(userid) { | 	if p.Token != nil && p.Token.IsAdmin(userid) { | ||||||
| 		return nil | 		return nil | ||||||
| @@ -72,7 +73,7 @@ func (ac *AppContext) CheckPermissionUserAdmin(userid models.UserID) *ginresp.HT | |||||||
| 	return langext.Ptr(ginresp.APIError(ac.ginContext, 401, apierr.USER_AUTH_FAILED, "You are not authorized for this action", nil)) | 	return langext.Ptr(ginresp.APIError(ac.ginContext, 401, apierr.USER_AUTH_FAILED, "You are not authorized for this action", nil)) | ||||||
| } | } | ||||||
|  |  | ||||||
| func (ac *AppContext) CheckPermissionSend(channel models.Channel, key string) (*models.KeyToken, *ginresp.HTTPResponse) { | func (ac *AppContext) CheckPermissionSend(channel models.Channel, key string) (*models.KeyToken, *ginext.HTTPResponse) { | ||||||
|  |  | ||||||
| 	keytok, err := ac.app.Database.Primary.GetKeyTokenByToken(ac, key) | 	keytok, err := ac.app.Database.Primary.GetKeyTokenByToken(ac, key) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| @@ -107,7 +108,7 @@ func (ac *AppContext) CheckPermissionMessageDelete(msg models.Message) bool { | |||||||
| 	return false | 	return false | ||||||
| } | } | ||||||
|  |  | ||||||
| func (ac *AppContext) CheckPermissionAny() *ginresp.HTTPResponse { | func (ac *AppContext) CheckPermissionAny() *ginext.HTTPResponse { | ||||||
| 	p := ac.permissions | 	p := ac.permissions | ||||||
| 	if p.Token == nil { | 	if p.Token == nil { | ||||||
| 		return langext.Ptr(ginresp.APIError(ac.ginContext, 401, apierr.USER_AUTH_FAILED, "You are not authorized for this action", nil)) | 		return langext.Ptr(ginresp.APIError(ac.ginContext, 401, apierr.USER_AUTH_FAILED, "You are not authorized for this action", nil)) | ||||||
|   | |||||||
							
								
								
									
										264
									
								
								scnserver/logic/request.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										264
									
								
								scnserver/logic/request.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,264 @@ | |||||||
|  | package logic | ||||||
|  |  | ||||||
|  | import ( | ||||||
|  | 	scn "blackforestbytes.com/simplecloudnotifier" | ||||||
|  | 	"blackforestbytes.com/simplecloudnotifier/api/apierr" | ||||||
|  | 	"blackforestbytes.com/simplecloudnotifier/api/ginresp" | ||||||
|  | 	"blackforestbytes.com/simplecloudnotifier/models" | ||||||
|  | 	"context" | ||||||
|  | 	"errors" | ||||||
|  | 	"fmt" | ||||||
|  | 	"github.com/gin-gonic/gin" | ||||||
|  | 	"github.com/mattn/go-sqlite3" | ||||||
|  | 	"github.com/rs/zerolog/log" | ||||||
|  | 	"gogs.mikescher.com/BlackForestBytes/goext/dataext" | ||||||
|  | 	"gogs.mikescher.com/BlackForestBytes/goext/exerr" | ||||||
|  | 	"gogs.mikescher.com/BlackForestBytes/goext/ginext" | ||||||
|  | 	"gogs.mikescher.com/BlackForestBytes/goext/langext" | ||||||
|  | 	"math/rand" | ||||||
|  | 	"runtime/debug" | ||||||
|  | 	"time" | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | type RequestOptions struct { | ||||||
|  | 	IgnoreWrongContentType bool | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (app *Application) DoRequest(gectx *ginext.AppContext, g *gin.Context, lockmode models.TransactionLockMode, fn func(ctx *AppContext, finishSuccess func(r ginext.HTTPResponse) ginext.HTTPResponse) ginext.HTTPResponse) ginext.HTTPResponse { | ||||||
|  |  | ||||||
|  | 	maxRetry := scn.Conf.RequestMaxRetry | ||||||
|  | 	retrySleep := scn.Conf.RequestRetrySleep | ||||||
|  |  | ||||||
|  | 	reqctx := g.Request.Context() | ||||||
|  |  | ||||||
|  | 	t0 := time.Now() | ||||||
|  |  | ||||||
|  | 	for ctr := 1; ; ctr++ { | ||||||
|  |  | ||||||
|  | 		ictx, cancel := context.WithTimeout(gectx, app.Config.RequestTimeout) | ||||||
|  |  | ||||||
|  | 		actx := CreateAppContext(app, g, ictx, cancel) | ||||||
|  |  | ||||||
|  | 		wrap, stackTrace, panicObj := callPanicSafe(func(ctx *AppContext, finishSuccess func(r ginext.HTTPResponse) ginext.HTTPResponse) ginext.HTTPResponse { | ||||||
|  |  | ||||||
|  | 			dl, ok := ctx.Deadline() | ||||||
|  | 			if !ok { | ||||||
|  | 				dl = time.Now().Add(time.Second * 5) | ||||||
|  | 			} | ||||||
|  |  | ||||||
|  | 			if lockmode == models.TLockRead { | ||||||
|  |  | ||||||
|  | 				islock := app.MainDatabaseLock.RTryLockWithTimeout(dl.Sub(time.Now())) | ||||||
|  | 				if !islock { | ||||||
|  | 					return ginresp.APIError(g, 500, apierr.INTERNAL_EXCEPTION, "Failed to lock {MainDatabaseLock} [ro]", nil) | ||||||
|  | 				} | ||||||
|  | 				defer app.MainDatabaseLock.RUnlock() | ||||||
|  |  | ||||||
|  | 			} else if lockmode == models.TLockReadWrite { | ||||||
|  |  | ||||||
|  | 				islock := app.MainDatabaseLock.TryLockWithTimeout(dl.Sub(time.Now())) | ||||||
|  | 				if !islock { | ||||||
|  | 					return ginresp.APIError(g, 500, apierr.INTERNAL_EXCEPTION, "Failed to lock {MainDatabaseLock} [rw]", nil) | ||||||
|  | 				} | ||||||
|  | 				defer app.MainDatabaseLock.Unlock() | ||||||
|  |  | ||||||
|  | 			} | ||||||
|  |  | ||||||
|  | 			authheader := g.GetHeader("Authorization") | ||||||
|  |  | ||||||
|  | 			perm, err := app.getPermissions(actx, authheader) | ||||||
|  | 			if err != nil { | ||||||
|  | 				cancel() | ||||||
|  | 				return ginresp.APIError(g, 400, apierr.PERM_QUERY_FAIL, "Failed to determine permissions", err) | ||||||
|  | 			} | ||||||
|  |  | ||||||
|  | 			actx.permissions = perm | ||||||
|  | 			g.Set("perm", perm) | ||||||
|  |  | ||||||
|  | 			return fn(actx, finishSuccess) | ||||||
|  |  | ||||||
|  | 		}, actx, actx._FinishSuccess) | ||||||
|  | 		if panicObj != nil { | ||||||
|  | 			log.Error().Interface("panicObj", panicObj).Msg("Panic occured (in gin handler)") | ||||||
|  | 			log.Error().Msg(stackTrace) | ||||||
|  | 			wrap = ginresp.APIError(g, 500, apierr.PANIC, "A panic occured in the HTTP handler", errors.New(fmt.Sprintf("%+v\n\n@:\n%s", panicObj, stackTrace))) | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		if g.Writer.Written() { | ||||||
|  | 			if scn.Conf.ReqLogEnabled { | ||||||
|  | 				app.InsertRequestLog(createRequestLog(g, t0, ctr, nil, langext.Ptr("Writing in WrapperFunc is not supported"))) | ||||||
|  | 			} | ||||||
|  | 			panic("Writing in WrapperFunc is not supported") | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		if ctr < maxRetry && isSqlite3Busy(wrap) { | ||||||
|  | 			log.Warn().Int("counter", ctr).Str("url", g.Request.URL.String()).Msg("Retry request (ErrBusy)") | ||||||
|  |  | ||||||
|  | 			err := resetBody(g) | ||||||
|  | 			if err != nil { | ||||||
|  | 				panic(err) | ||||||
|  | 			} | ||||||
|  |  | ||||||
|  | 			time.Sleep(time.Duration(int64(float64(retrySleep) * (0.5 + rand.Float64())))) | ||||||
|  | 			continue | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		if reqctx.Err() == nil { | ||||||
|  | 			if scn.Conf.ReqLogEnabled { | ||||||
|  | 				app.InsertRequestLog(createRequestLog(g, t0, ctr, wrap, nil)) | ||||||
|  | 			} | ||||||
|  |  | ||||||
|  | 			if scw, ok := wrap.(ginext.InspectableHTTPResponse); ok { | ||||||
|  |  | ||||||
|  | 				statuscode := scw.Statuscode() | ||||||
|  | 				if statuscode/100 != 2 { | ||||||
|  | 					log.Warn().Str("url", g.Request.Method+"::"+g.Request.URL.String()).Msg(fmt.Sprintf("Request failed with statuscode %d", statuscode)) | ||||||
|  | 				} | ||||||
|  | 			} else { | ||||||
|  | 				log.Warn().Str("url", g.Request.Method+"::"+g.Request.URL.String()).Msg(fmt.Sprintf("Request failed with statuscode [unknown]")) | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		return wrap | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func callPanicSafe(fn func(ctx *AppContext, finishSuccess func(r ginext.HTTPResponse) ginext.HTTPResponse) ginext.HTTPResponse, actx *AppContext, fnFin func(r ginext.HTTPResponse) ginext.HTTPResponse) (res ginext.HTTPResponse, stackTrace string, panicObj any) { | ||||||
|  | 	defer func() { | ||||||
|  | 		if rec := recover(); rec != nil { | ||||||
|  | 			res = nil | ||||||
|  | 			stackTrace = string(debug.Stack()) | ||||||
|  | 			panicObj = rec | ||||||
|  | 		} | ||||||
|  | 	}() | ||||||
|  |  | ||||||
|  | 	res = fn(actx, fnFin) | ||||||
|  | 	return res, "", nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func createRequestLog(g *gin.Context, t0 time.Time, ctr int, resp ginext.HTTPResponse, panicstr *string) models.RequestLog { | ||||||
|  |  | ||||||
|  | 	t1 := time.Now() | ||||||
|  |  | ||||||
|  | 	ua := g.Request.UserAgent() | ||||||
|  | 	auth := g.Request.Header.Get("Authorization") | ||||||
|  | 	ct := g.Request.Header.Get("Content-Type") | ||||||
|  |  | ||||||
|  | 	var reqbody []byte = nil | ||||||
|  | 	if g.Request.Body != nil { | ||||||
|  | 		brcbody, err := g.Request.Body.(dataext.BufferedReadCloser).BufferedAll() | ||||||
|  | 		if err == nil { | ||||||
|  | 			reqbody = brcbody | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	var strreqbody *string = nil | ||||||
|  | 	if len(reqbody) < scn.Conf.ReqLogMaxBodySize { | ||||||
|  | 		strreqbody = langext.Ptr(string(reqbody)) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	var respbody *string = nil | ||||||
|  |  | ||||||
|  | 	var strrespbody *string = nil | ||||||
|  | 	if resp != nil { | ||||||
|  | 		if resp2, ok := resp.(ginext.InspectableHTTPResponse); ok { | ||||||
|  | 			respbody = resp2.BodyString(g) | ||||||
|  | 			if respbody != nil && len(*respbody) < scn.Conf.ReqLogMaxBodySize { | ||||||
|  | 				strrespbody = respbody | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	permObj, hasPerm := g.Get("perm") | ||||||
|  |  | ||||||
|  | 	hasTok := false | ||||||
|  | 	if hasPerm { | ||||||
|  | 		hasTok = permObj.(models.PermissionSet).Token != nil | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	var statuscode *int64 = nil | ||||||
|  | 	if resp != nil { | ||||||
|  | 		if resp2, ok := resp.(ginext.InspectableHTTPResponse); ok { | ||||||
|  | 			statuscode = langext.Ptr(int64(resp2.Statuscode())) | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	var contentType = "" | ||||||
|  | 	if resp != nil { | ||||||
|  | 		if resp2, ok := resp.(ginext.InspectableHTTPResponse); ok { | ||||||
|  | 			contentType = resp2.ContentType() | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return models.RequestLog{ | ||||||
|  | 		Method:              g.Request.Method, | ||||||
|  | 		URI:                 g.Request.URL.String(), | ||||||
|  | 		UserAgent:           langext.Conditional(ua == "", nil, &ua), | ||||||
|  | 		Authentication:      langext.Conditional(auth == "", nil, &auth), | ||||||
|  | 		RequestBody:         strreqbody, | ||||||
|  | 		RequestBodySize:     int64(len(reqbody)), | ||||||
|  | 		RequestContentType:  ct, | ||||||
|  | 		RemoteIP:            g.RemoteIP(), | ||||||
|  | 		KeyID:               langext.ConditionalFn10(hasTok, func() *models.KeyTokenID { return langext.Ptr(permObj.(models.PermissionSet).Token.KeyTokenID) }, nil), | ||||||
|  | 		UserID:              langext.ConditionalFn10(hasTok, func() *models.UserID { return langext.Ptr(permObj.(models.PermissionSet).Token.OwnerUserID) }, nil), | ||||||
|  | 		Permissions:         langext.ConditionalFn10(hasTok, func() *string { return langext.Ptr(permObj.(models.PermissionSet).Token.Permissions.String()) }, nil), | ||||||
|  | 		ResponseStatuscode:  statuscode, | ||||||
|  | 		ResponseBodySize:    langext.ConditionalFn10(strrespbody != nil, func() *int64 { return langext.Ptr(int64(len(*respbody))) }, nil), | ||||||
|  | 		ResponseBody:        strrespbody, | ||||||
|  | 		ResponseContentType: contentType, | ||||||
|  | 		RetryCount:          int64(ctr), | ||||||
|  | 		Panicked:            panicstr != nil, | ||||||
|  | 		PanicStr:            panicstr, | ||||||
|  | 		ProcessingTime:      models.SCNDuration(t1.Sub(t0)), | ||||||
|  | 		TimestampStart:      models.NewSCNTime(t0), | ||||||
|  | 		TimestampFinish:     models.NewSCNTime(t1), | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func resetBody(g *gin.Context) error { | ||||||
|  | 	if g.Request.Body == nil { | ||||||
|  | 		return nil | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	err := g.Request.Body.(dataext.BufferedReadCloser).Reset() | ||||||
|  | 	if err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func isSqlite3Busy(r ginext.HTTPResponse) bool { | ||||||
|  | 	if errwrap, ok := r.(interface{ Unwrap() error }); ok && errwrap != nil { | ||||||
|  | 		orig := exerr.OriginalError(errwrap.Unwrap()) | ||||||
|  |  | ||||||
|  | 		var sqlite3Err sqlite3.Error | ||||||
|  | 		if errors.As(orig, &sqlite3Err) { | ||||||
|  | 			if sqlite3Err.Code == 5 { // [5] == SQLITE_BUSY | ||||||
|  | 				return true | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	return false | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func BuildGinRequestError(g *gin.Context, fieldtype string, err error) ginext.HTTPResponse { | ||||||
|  | 	switch fieldtype { | ||||||
|  | 	case "URI": | ||||||
|  | 		return ginresp.APIError(g, 400, apierr.BINDFAIL_URI_PARAM, "Failed to read uri", err) | ||||||
|  | 	case "QUERY": | ||||||
|  | 		return ginresp.APIError(g, 400, apierr.BINDFAIL_QUERY_PARAM, "Failed to read query", err) | ||||||
|  | 	case "JSON": | ||||||
|  | 		return ginresp.APIError(g, 400, apierr.BINDFAIL_BODY_PARAM, "Failed to read JSON body", err) | ||||||
|  | 	case "BODY": | ||||||
|  | 		return ginresp.APIError(g, 400, apierr.BINDFAIL_BODY_PARAM, "Failed to read query", err) | ||||||
|  | 	case "FORM": | ||||||
|  | 		return ginresp.APIError(g, 400, apierr.BINDFAIL_BODY_PARAM, "Failed to read multipart-form / urlencoded-form", err) | ||||||
|  | 	case "HEADER": | ||||||
|  | 		return ginresp.APIError(g, 400, apierr.BINDFAIL_HEADER_PARAM, "Failed to read header", err) | ||||||
|  | 	case "INIT": | ||||||
|  | 		return ginresp.APIError(g, 400, apierr.INTERNAL_EXCEPTION, "Failed to init context", err) | ||||||
|  | 	default: | ||||||
|  | 		return ginresp.APIError(g, 400, apierr.INTERNAL_EXCEPTION, "Failed to init", err) | ||||||
|  | 	} | ||||||
|  | } | ||||||
| @@ -1,37 +1,28 @@ | |||||||
| package models | package models | ||||||
|  |  | ||||||
| import ( |  | ||||||
| 	"context" |  | ||||||
| 	"github.com/jmoiron/sqlx" |  | ||||||
| 	"gogs.mikescher.com/BlackForestBytes/goext/langext" |  | ||||||
| 	"gogs.mikescher.com/BlackForestBytes/goext/sq" |  | ||||||
| 	"time" |  | ||||||
| ) |  | ||||||
|  |  | ||||||
| type Channel struct { | type Channel struct { | ||||||
| 	ChannelID         ChannelID | 	ChannelID         ChannelID `db:"channel_id"         json:"channel_id"` | ||||||
| 	OwnerUserID       UserID | 	OwnerUserID       UserID    `db:"owner_user_id"      json:"owner_user_id"` | ||||||
| 	InternalName      string | 	InternalName      string    `db:"internal_name"      json:"internal_name"` | ||||||
| 	DisplayName       string | 	DisplayName       string    `db:"display_name"       json:"display_name"` | ||||||
| 	DescriptionName   *string | 	DescriptionName   *string   `db:"description_name"   json:"description_name"` | ||||||
| 	SubscribeKey      string | 	SubscribeKey      string    `db:"subscribe_key"      json:"subscribe_key"      jsonfilter:"INCLUDE_KEY"` // can be nil, depending on endpoint | ||||||
| 	TimestampCreated  time.Time | 	TimestampCreated  SCNTime   `db:"timestamp_created"  json:"timestamp_created"` | ||||||
| 	TimestampLastSent *time.Time | 	TimestampLastSent *SCNTime  `db:"timestamp_lastsent" json:"timestamp_lastsent"` | ||||||
| 	MessagesSent      int | 	MessagesSent      int       `db:"messages_sent"      json:"messages_sent"` | ||||||
| } | } | ||||||
|  |  | ||||||
| func (c Channel) JSON(includeKey bool) ChannelJSON { | type ChannelWithSubscription struct { | ||||||
| 	return ChannelJSON{ | 	Channel | ||||||
| 		ChannelID:         c.ChannelID, | 	Subscription *Subscription `db:"sub" json:"subscription"` | ||||||
| 		OwnerUserID:       c.OwnerUserID, | } | ||||||
| 		InternalName:      c.InternalName, |  | ||||||
| 		DisplayName:       c.DisplayName, | type ChannelPreview struct { | ||||||
| 		DescriptionName:   c.DescriptionName, | 	ChannelID       ChannelID `json:"channel_id"` | ||||||
| 		SubscribeKey:      langext.Conditional(includeKey, langext.Ptr(c.SubscribeKey), nil), | 	OwnerUserID     UserID    `json:"owner_user_id"` | ||||||
| 		TimestampCreated:  c.TimestampCreated.Format(time.RFC3339Nano), | 	InternalName    string    `json:"internal_name"` | ||||||
| 		TimestampLastSent: timeOptFmt(c.TimestampLastSent, time.RFC3339Nano), | 	DisplayName     string    `json:"display_name"` | ||||||
| 		MessagesSent:      c.MessagesSent, | 	DescriptionName *string   `json:"description_name"` | ||||||
| 	} |  | ||||||
| } | } | ||||||
|  |  | ||||||
| func (c Channel) WithSubscription(sub *Subscription) ChannelWithSubscription { | func (c Channel) WithSubscription(sub *Subscription) ChannelWithSubscription { | ||||||
| @@ -41,8 +32,8 @@ func (c Channel) WithSubscription(sub *Subscription) ChannelWithSubscription { | |||||||
| 	} | 	} | ||||||
| } | } | ||||||
|  |  | ||||||
| func (c Channel) JSONPreview() ChannelPreviewJSON { | func (c Channel) Preview() ChannelPreview { | ||||||
| 	return ChannelPreviewJSON{ | 	return ChannelPreview{ | ||||||
| 		ChannelID:       c.ChannelID, | 		ChannelID:       c.ChannelID, | ||||||
| 		OwnerUserID:     c.OwnerUserID, | 		OwnerUserID:     c.OwnerUserID, | ||||||
| 		InternalName:    c.InternalName, | 		InternalName:    c.InternalName, | ||||||
| @@ -50,118 +41,3 @@ func (c Channel) JSONPreview() ChannelPreviewJSON { | |||||||
| 		DescriptionName: c.DescriptionName, | 		DescriptionName: c.DescriptionName, | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
|  |  | ||||||
| type ChannelWithSubscription struct { |  | ||||||
| 	Channel |  | ||||||
| 	Subscription *Subscription |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (c ChannelWithSubscription) JSON(includeChannelKey bool) ChannelWithSubscriptionJSON { |  | ||||||
| 	var sub *SubscriptionJSON = nil |  | ||||||
| 	if c.Subscription != nil { |  | ||||||
| 		sub = langext.Ptr(c.Subscription.JSON()) |  | ||||||
| 	} |  | ||||||
| 	return ChannelWithSubscriptionJSON{ |  | ||||||
| 		ChannelJSON:  c.Channel.JSON(includeChannelKey), |  | ||||||
| 		Subscription: sub, |  | ||||||
| 	} |  | ||||||
| } |  | ||||||
|  |  | ||||||
| type ChannelJSON struct { |  | ||||||
| 	ChannelID         ChannelID `json:"channel_id"` |  | ||||||
| 	OwnerUserID       UserID    `json:"owner_user_id"` |  | ||||||
| 	InternalName      string    `json:"internal_name"` |  | ||||||
| 	DisplayName       string    `json:"display_name"` |  | ||||||
| 	DescriptionName   *string   `json:"description_name"` |  | ||||||
| 	SubscribeKey      *string   `json:"subscribe_key"` // can be nil, depending on endpoint |  | ||||||
| 	TimestampCreated  string    `json:"timestamp_created"` |  | ||||||
| 	TimestampLastSent *string   `json:"timestamp_lastsent"` |  | ||||||
| 	MessagesSent      int       `json:"messages_sent"` |  | ||||||
| } |  | ||||||
|  |  | ||||||
| type ChannelWithSubscriptionJSON struct { |  | ||||||
| 	ChannelJSON |  | ||||||
| 	Subscription *SubscriptionJSON `json:"subscription"` |  | ||||||
| } |  | ||||||
|  |  | ||||||
| type ChannelPreviewJSON struct { |  | ||||||
| 	ChannelID       ChannelID `json:"channel_id"` |  | ||||||
| 	OwnerUserID     UserID    `json:"owner_user_id"` |  | ||||||
| 	InternalName    string    `json:"internal_name"` |  | ||||||
| 	DisplayName     string    `json:"display_name"` |  | ||||||
| 	DescriptionName *string   `json:"description_name"` |  | ||||||
| } |  | ||||||
|  |  | ||||||
| type ChannelDB struct { |  | ||||||
| 	ChannelID         ChannelID `db:"channel_id"` |  | ||||||
| 	OwnerUserID       UserID    `db:"owner_user_id"` |  | ||||||
| 	InternalName      string    `db:"internal_name"` |  | ||||||
| 	DisplayName       string    `db:"display_name"` |  | ||||||
| 	DescriptionName   *string   `db:"description_name"` |  | ||||||
| 	SubscribeKey      string    `db:"subscribe_key"` |  | ||||||
| 	TimestampCreated  int64     `db:"timestamp_created"` |  | ||||||
| 	TimestampLastSent *int64    `db:"timestamp_lastsent"` |  | ||||||
| 	MessagesSent      int       `db:"messages_sent"` |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (c ChannelDB) Model() Channel { |  | ||||||
| 	return Channel{ |  | ||||||
| 		ChannelID:         c.ChannelID, |  | ||||||
| 		OwnerUserID:       c.OwnerUserID, |  | ||||||
| 		InternalName:      c.InternalName, |  | ||||||
| 		DisplayName:       c.DisplayName, |  | ||||||
| 		DescriptionName:   c.DescriptionName, |  | ||||||
| 		SubscribeKey:      c.SubscribeKey, |  | ||||||
| 		TimestampCreated:  timeFromMilli(c.TimestampCreated), |  | ||||||
| 		TimestampLastSent: timeOptFromMilli(c.TimestampLastSent), |  | ||||||
| 		MessagesSent:      c.MessagesSent, |  | ||||||
| 	} |  | ||||||
| } |  | ||||||
|  |  | ||||||
| type ChannelWithSubscriptionDB struct { |  | ||||||
| 	ChannelDB |  | ||||||
| 	Subscription *SubscriptionDB `db:"sub"` |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (c ChannelWithSubscriptionDB) Model() ChannelWithSubscription { |  | ||||||
| 	var sub *Subscription = nil |  | ||||||
| 	if c.Subscription != nil { |  | ||||||
| 		sub = langext.Ptr(c.Subscription.Model()) |  | ||||||
| 	} |  | ||||||
| 	return ChannelWithSubscription{ |  | ||||||
| 		Channel:      c.ChannelDB.Model(), |  | ||||||
| 		Subscription: sub, |  | ||||||
| 	} |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func DecodeChannel(ctx context.Context, q sq.Queryable, r *sqlx.Rows) (Channel, error) { |  | ||||||
| 	data, err := sq.ScanSingle[ChannelDB](ctx, q, r, sq.SModeFast, sq.Safe, true) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return Channel{}, err |  | ||||||
| 	} |  | ||||||
| 	return data.Model(), nil |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func DecodeChannels(ctx context.Context, q sq.Queryable, r *sqlx.Rows) ([]Channel, error) { |  | ||||||
| 	data, err := sq.ScanAll[ChannelDB](ctx, q, r, sq.SModeFast, sq.Safe, true) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return nil, err |  | ||||||
| 	} |  | ||||||
| 	return langext.ArrMap(data, func(v ChannelDB) Channel { return v.Model() }), nil |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func DecodeChannelWithSubscription(ctx context.Context, q sq.Queryable, r *sqlx.Rows) (ChannelWithSubscription, error) { |  | ||||||
| 	data, err := sq.ScanSingle[ChannelWithSubscriptionDB](ctx, q, r, sq.SModeExtended, sq.Safe, true) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return ChannelWithSubscription{}, err |  | ||||||
| 	} |  | ||||||
| 	return data.Model(), nil |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func DecodeChannelsWithSubscription(ctx context.Context, q sq.Queryable, r *sqlx.Rows) ([]ChannelWithSubscription, error) { |  | ||||||
| 	data, err := sq.ScanAll[ChannelWithSubscriptionDB](ctx, q, r, sq.SModeExtended, sq.Safe, true) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return nil, err |  | ||||||
| 	} |  | ||||||
| 	return langext.ArrMap(data, func(v ChannelWithSubscriptionDB) ChannelWithSubscription { return v.Model() }), nil |  | ||||||
| } |  | ||||||
|   | |||||||
| @@ -1,13 +1,5 @@ | |||||||
| package models | package models | ||||||
|  |  | ||||||
| import ( |  | ||||||
| 	"context" |  | ||||||
| 	"github.com/jmoiron/sqlx" |  | ||||||
| 	"gogs.mikescher.com/BlackForestBytes/goext/langext" |  | ||||||
| 	"gogs.mikescher.com/BlackForestBytes/goext/sq" |  | ||||||
| 	"time" |  | ||||||
| ) |  | ||||||
|  |  | ||||||
| type ClientType string //@enum:type | type ClientType string //@enum:type | ||||||
|  |  | ||||||
| const ( | const ( | ||||||
| @@ -19,76 +11,12 @@ const ( | |||||||
| ) | ) | ||||||
|  |  | ||||||
| type Client struct { | type Client struct { | ||||||
| 	ClientID         ClientID | 	ClientID         ClientID   `db:"client_id"         json:"client_id"` | ||||||
| 	UserID           UserID | 	UserID           UserID     `db:"user_id"           json:"user_id"` | ||||||
| 	Type             ClientType | 	Type             ClientType `db:"type"              json:"type"` | ||||||
| 	FCMToken         string | 	FCMToken         string     `db:"fcm_token"         json:"fcm_token"` | ||||||
| 	TimestampCreated time.Time | 	TimestampCreated SCNTime    `db:"timestamp_created" json:"timestamp_created"` | ||||||
| 	AgentModel       string | 	AgentModel       string     `db:"agent_model"       json:"agent_model"` | ||||||
| 	AgentVersion     string | 	AgentVersion     string     `db:"agent_version"     json:"agent_version"` | ||||||
| 	Name             *string | 	Name             *string    `db:"name"              json:"name"` | ||||||
| } |  | ||||||
|  |  | ||||||
| func (c Client) JSON() ClientJSON { |  | ||||||
| 	return ClientJSON{ |  | ||||||
| 		ClientID:         c.ClientID, |  | ||||||
| 		UserID:           c.UserID, |  | ||||||
| 		Type:             c.Type, |  | ||||||
| 		FCMToken:         c.FCMToken, |  | ||||||
| 		TimestampCreated: c.TimestampCreated.Format(time.RFC3339Nano), |  | ||||||
| 		AgentModel:       c.AgentModel, |  | ||||||
| 		AgentVersion:     c.AgentVersion, |  | ||||||
| 		Name:             c.Name, |  | ||||||
| 	} |  | ||||||
| } |  | ||||||
|  |  | ||||||
| type ClientJSON struct { |  | ||||||
| 	ClientID         ClientID   `json:"client_id"` |  | ||||||
| 	UserID           UserID     `json:"user_id"` |  | ||||||
| 	Type             ClientType `json:"type"` |  | ||||||
| 	FCMToken         string     `json:"fcm_token"` |  | ||||||
| 	TimestampCreated string     `json:"timestamp_created"` |  | ||||||
| 	AgentModel       string     `json:"agent_model"` |  | ||||||
| 	AgentVersion     string     `json:"agent_version"` |  | ||||||
| 	Name             *string    `json:"name"` |  | ||||||
| } |  | ||||||
|  |  | ||||||
| type ClientDB struct { |  | ||||||
| 	ClientID         ClientID   `db:"client_id"` |  | ||||||
| 	UserID           UserID     `db:"user_id"` |  | ||||||
| 	Type             ClientType `db:"type"` |  | ||||||
| 	FCMToken         string     `db:"fcm_token"` |  | ||||||
| 	TimestampCreated int64      `db:"timestamp_created"` |  | ||||||
| 	AgentModel       string     `db:"agent_model"` |  | ||||||
| 	AgentVersion     string     `db:"agent_version"` |  | ||||||
| 	Name             *string    `db:"name"` |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (c ClientDB) Model() Client { |  | ||||||
| 	return Client{ |  | ||||||
| 		ClientID:         c.ClientID, |  | ||||||
| 		UserID:           c.UserID, |  | ||||||
| 		Type:             c.Type, |  | ||||||
| 		FCMToken:         c.FCMToken, |  | ||||||
| 		TimestampCreated: timeFromMilli(c.TimestampCreated), |  | ||||||
| 		AgentModel:       c.AgentModel, |  | ||||||
| 		AgentVersion:     c.AgentVersion, |  | ||||||
| 		Name:             c.Name, |  | ||||||
| 	} |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func DecodeClient(ctx context.Context, q sq.Queryable, r *sqlx.Rows) (Client, error) { |  | ||||||
| 	data, err := sq.ScanSingle[ClientDB](ctx, q, r, sq.SModeFast, sq.Safe, true) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return Client{}, err |  | ||||||
| 	} |  | ||||||
| 	return data.Model(), nil |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func DecodeClients(ctx context.Context, q sq.Queryable, r *sqlx.Rows) ([]Client, error) { |  | ||||||
| 	data, err := sq.ScanAll[ClientDB](ctx, q, r, sq.SModeFast, sq.Safe, true) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return nil, err |  | ||||||
| 	} |  | ||||||
| 	return langext.ArrMap(data, func(v ClientDB) Client { return v.Model() }), nil |  | ||||||
| } | } | ||||||
|   | |||||||
| @@ -1,13 +1,5 @@ | |||||||
| package models | package models | ||||||
|  |  | ||||||
| import ( |  | ||||||
| 	"context" |  | ||||||
| 	"github.com/jmoiron/sqlx" |  | ||||||
| 	"gogs.mikescher.com/BlackForestBytes/goext/langext" |  | ||||||
| 	"gogs.mikescher.com/BlackForestBytes/goext/sq" |  | ||||||
| 	"time" |  | ||||||
| ) |  | ||||||
|  |  | ||||||
| type DeliveryStatus string //@enum:type | type DeliveryStatus string //@enum:type | ||||||
|  |  | ||||||
| const ( | const ( | ||||||
| @@ -17,90 +9,18 @@ const ( | |||||||
| ) | ) | ||||||
|  |  | ||||||
| type Delivery struct { | type Delivery struct { | ||||||
| 	DeliveryID         DeliveryID | 	DeliveryID         DeliveryID     `db:"delivery_id"         json:"delivery_id"` | ||||||
| 	MessageID          MessageID | 	MessageID          MessageID      `db:"message_id"          json:"message_id"` | ||||||
| 	ReceiverUserID     UserID | 	ReceiverUserID     UserID         `db:"receiver_user_id"    json:"receiver_user_id"` | ||||||
| 	ReceiverClientID   ClientID | 	ReceiverClientID   ClientID       `db:"receiver_client_id"  json:"receiver_client_id"` | ||||||
| 	TimestampCreated   time.Time | 	TimestampCreated   SCNTime        `db:"timestamp_created"   json:"timestamp_created"` | ||||||
| 	TimestampFinalized *time.Time | 	TimestampFinalized *SCNTime       `db:"timestamp_finalized" json:"timestamp_finalized"` | ||||||
| 	Status             DeliveryStatus | 	Status             DeliveryStatus `db:"status"              json:"status"` | ||||||
| 	RetryCount         int | 	RetryCount         int            `db:"retry_count"         json:"retry_count"` | ||||||
| 	NextDelivery       *time.Time | 	NextDelivery       *SCNTime       `db:"next_delivery"       json:"next_delivery"` | ||||||
| 	FCMMessageID       *string | 	FCMMessageID       *string        `db:"fcm_message_id"      json:"fcm_message_id"` | ||||||
| } |  | ||||||
|  |  | ||||||
| func (d Delivery) JSON() DeliveryJSON { |  | ||||||
| 	return DeliveryJSON{ |  | ||||||
| 		DeliveryID:         d.DeliveryID, |  | ||||||
| 		MessageID:          d.MessageID, |  | ||||||
| 		ReceiverUserID:     d.ReceiverUserID, |  | ||||||
| 		ReceiverClientID:   d.ReceiverClientID, |  | ||||||
| 		TimestampCreated:   d.TimestampCreated.Format(time.RFC3339Nano), |  | ||||||
| 		TimestampFinalized: timeOptFmt(d.TimestampFinalized, time.RFC3339Nano), |  | ||||||
| 		Status:             d.Status, |  | ||||||
| 		RetryCount:         d.RetryCount, |  | ||||||
| 		NextDelivery:       timeOptFmt(d.NextDelivery, time.RFC3339Nano), |  | ||||||
| 		FCMMessageID:       d.FCMMessageID, |  | ||||||
| 	} |  | ||||||
| } | } | ||||||
|  |  | ||||||
| func (d Delivery) MaxRetryCount() int { | func (d Delivery) MaxRetryCount() int { | ||||||
| 	return 5 | 	return 5 | ||||||
| } | } | ||||||
|  |  | ||||||
| type DeliveryJSON struct { |  | ||||||
| 	DeliveryID         DeliveryID     `json:"delivery_id"` |  | ||||||
| 	MessageID          MessageID      `json:"message_id"` |  | ||||||
| 	ReceiverUserID     UserID         `json:"receiver_user_id"` |  | ||||||
| 	ReceiverClientID   ClientID       `json:"receiver_client_id"` |  | ||||||
| 	TimestampCreated   string         `json:"timestamp_created"` |  | ||||||
| 	TimestampFinalized *string        `json:"timestamp_finalized"` |  | ||||||
| 	Status             DeliveryStatus `json:"status"` |  | ||||||
| 	RetryCount         int            `json:"retry_count"` |  | ||||||
| 	NextDelivery       *string        `json:"next_delivery"` |  | ||||||
| 	FCMMessageID       *string        `json:"fcm_message_id"` |  | ||||||
| } |  | ||||||
|  |  | ||||||
| type DeliveryDB struct { |  | ||||||
| 	DeliveryID         DeliveryID     `db:"delivery_id"` |  | ||||||
| 	MessageID          MessageID      `db:"message_id"` |  | ||||||
| 	ReceiverUserID     UserID         `db:"receiver_user_id"` |  | ||||||
| 	ReceiverClientID   ClientID       `db:"receiver_client_id"` |  | ||||||
| 	TimestampCreated   int64          `db:"timestamp_created"` |  | ||||||
| 	TimestampFinalized *int64         `db:"timestamp_finalized"` |  | ||||||
| 	Status             DeliveryStatus `db:"status"` |  | ||||||
| 	RetryCount         int            `db:"retry_count"` |  | ||||||
| 	NextDelivery       *int64         `db:"next_delivery"` |  | ||||||
| 	FCMMessageID       *string        `db:"fcm_message_id"` |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (d DeliveryDB) Model() Delivery { |  | ||||||
| 	return Delivery{ |  | ||||||
| 		DeliveryID:         d.DeliveryID, |  | ||||||
| 		MessageID:          d.MessageID, |  | ||||||
| 		ReceiverUserID:     d.ReceiverUserID, |  | ||||||
| 		ReceiverClientID:   d.ReceiverClientID, |  | ||||||
| 		TimestampCreated:   timeFromMilli(d.TimestampCreated), |  | ||||||
| 		TimestampFinalized: timeOptFromMilli(d.TimestampFinalized), |  | ||||||
| 		Status:             d.Status, |  | ||||||
| 		RetryCount:         d.RetryCount, |  | ||||||
| 		NextDelivery:       timeOptFromMilli(d.NextDelivery), |  | ||||||
| 		FCMMessageID:       d.FCMMessageID, |  | ||||||
| 	} |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func DecodeDelivery(ctx context.Context, q sq.Queryable, r *sqlx.Rows) (Delivery, error) { |  | ||||||
| 	data, err := sq.ScanSingle[DeliveryDB](ctx, q, r, sq.SModeFast, sq.Safe, true) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return Delivery{}, err |  | ||||||
| 	} |  | ||||||
| 	return data.Model(), nil |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func DecodeDeliveries(ctx context.Context, q sq.Queryable, r *sqlx.Rows) ([]Delivery, error) { |  | ||||||
| 	data, err := sq.ScanAll[DeliveryDB](ctx, q, r, sq.SModeFast, sq.Safe, true) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return nil, err |  | ||||||
| 	} |  | ||||||
| 	return langext.ArrMap(data, func(v DeliveryDB) Delivery { return v.Model() }), nil |  | ||||||
| } |  | ||||||
|   | |||||||
							
								
								
									
										35
									
								
								scnserver/models/duration.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										35
									
								
								scnserver/models/duration.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,35 @@ | |||||||
|  | package models | ||||||
|  |  | ||||||
|  | import ( | ||||||
|  | 	"encoding/json" | ||||||
|  | 	"gogs.mikescher.com/BlackForestBytes/goext/timeext" | ||||||
|  | 	"time" | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | type SCNDuration time.Duration | ||||||
|  |  | ||||||
|  | func (t SCNDuration) MarshalToDB(v SCNDuration) (int64, error) { | ||||||
|  | 	return v.Duration().Milliseconds(), nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (t SCNDuration) UnmarshalToModel(v int64) (SCNDuration, error) { | ||||||
|  | 	return SCNDuration(timeext.FromMilliseconds(v)), nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (t SCNDuration) Duration() time.Duration { | ||||||
|  | 	return time.Duration(t) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (t *SCNDuration) UnmarshalJSON(data []byte) error { | ||||||
|  | 	flt := float64(0) | ||||||
|  | 	if err := json.Unmarshal(data, &flt); err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  | 	d0 := timeext.FromSeconds(flt) | ||||||
|  | 	*t = SCNDuration(d0) | ||||||
|  | 	return nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (t SCNDuration) MarshalJSON() ([]byte, error) { | ||||||
|  | 	return json.Marshal(t.Duration().Seconds()) | ||||||
|  | } | ||||||
| @@ -5,7 +5,7 @@ package models | |||||||
| import "gogs.mikescher.com/BlackForestBytes/goext/langext" | import "gogs.mikescher.com/BlackForestBytes/goext/langext" | ||||||
| import "gogs.mikescher.com/BlackForestBytes/goext/enums" | import "gogs.mikescher.com/BlackForestBytes/goext/enums" | ||||||
|  |  | ||||||
| const ChecksumEnumGenerator = "e500346e3f60b3abf78558ec3df128c3be2a1cefa71c4f1feba9293d14eb85d1" // GoExtVersion: 0.0.463 | const ChecksumEnumGenerator = "902919af7c6d46bd6701b33e47308bad93d50cd10cdacaac739e5242819c4d7b" // GoExtVersion: 0.0.512 | ||||||
|  |  | ||||||
| // ================================ ClientType ================================ | // ================================ ClientType ================================ | ||||||
| // | // | ||||||
| @@ -283,6 +283,86 @@ func TokenPermValuesDescriptionMeta() []enums.EnumDescriptionMetaValue { | |||||||
| 	} | 	} | ||||||
| } | } | ||||||
|  |  | ||||||
|  | // ================================ TransactionLockMode ================================ | ||||||
|  | // | ||||||
|  | // File:       lock.go | ||||||
|  | // StringEnum: true | ||||||
|  | // DescrEnum:  false | ||||||
|  | // DataEnum:   false | ||||||
|  | // | ||||||
|  |  | ||||||
|  | var __TransactionLockModeValues = []TransactionLockMode{ | ||||||
|  | 	TLockNone, | ||||||
|  | 	TLockRead, | ||||||
|  | 	TLockReadWrite, | ||||||
|  | } | ||||||
|  |  | ||||||
|  | var __TransactionLockModeVarnames = map[TransactionLockMode]string{ | ||||||
|  | 	TLockNone:      "TLockNone", | ||||||
|  | 	TLockRead:      "TLockRead", | ||||||
|  | 	TLockReadWrite: "TLockReadWrite", | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (e TransactionLockMode) Valid() bool { | ||||||
|  | 	return langext.InArray(e, __TransactionLockModeValues) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (e TransactionLockMode) Values() []TransactionLockMode { | ||||||
|  | 	return __TransactionLockModeValues | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (e TransactionLockMode) ValuesAny() []any { | ||||||
|  | 	return langext.ArrCastToAny(__TransactionLockModeValues) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (e TransactionLockMode) ValuesMeta() []enums.EnumMetaValue { | ||||||
|  | 	return TransactionLockModeValuesMeta() | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (e TransactionLockMode) String() string { | ||||||
|  | 	return string(e) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (e TransactionLockMode) VarName() string { | ||||||
|  | 	if d, ok := __TransactionLockModeVarnames[e]; ok { | ||||||
|  | 		return d | ||||||
|  | 	} | ||||||
|  | 	return "" | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (e TransactionLockMode) TypeName() string { | ||||||
|  | 	return "TransactionLockMode" | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (e TransactionLockMode) PackageName() string { | ||||||
|  | 	return "models" | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (e TransactionLockMode) Meta() enums.EnumMetaValue { | ||||||
|  | 	return enums.EnumMetaValue{VarName: e.VarName(), Value: e, Description: nil} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func ParseTransactionLockMode(vv string) (TransactionLockMode, bool) { | ||||||
|  | 	for _, ev := range __TransactionLockModeValues { | ||||||
|  | 		if string(ev) == vv { | ||||||
|  | 			return ev, true | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	return "", false | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func TransactionLockModeValues() []TransactionLockMode { | ||||||
|  | 	return __TransactionLockModeValues | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func TransactionLockModeValuesMeta() []enums.EnumMetaValue { | ||||||
|  | 	return []enums.EnumMetaValue{ | ||||||
|  | 		TLockNone.Meta(), | ||||||
|  | 		TLockRead.Meta(), | ||||||
|  | 		TLockReadWrite.Meta(), | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
| // ================================ ================= ================================ | // ================================ ================= ================================ | ||||||
|  |  | ||||||
| func AllPackageEnums() []enums.Enum { | func AllPackageEnums() []enums.Enum { | ||||||
| @@ -290,5 +370,6 @@ func AllPackageEnums() []enums.Enum { | |||||||
| 		ClientTypeAndroid,   // ClientType | 		ClientTypeAndroid,   // ClientType | ||||||
| 		DeliveryStatusRetry, // DeliveryStatus | 		DeliveryStatusRetry, // DeliveryStatus | ||||||
| 		PermAdmin,           // TokenPerm | 		PermAdmin,           // TokenPerm | ||||||
|  | 		TLockNone,           // TransactionLockMode | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
|   | |||||||
| @@ -15,7 +15,7 @@ import "reflect" | |||||||
| import "regexp" | import "regexp" | ||||||
| import "strings" | import "strings" | ||||||
|  |  | ||||||
| const ChecksumCharsetIDGenerator = "e500346e3f60b3abf78558ec3df128c3be2a1cefa71c4f1feba9293d14eb85d1" // GoExtVersion: 0.0.463 | const ChecksumCharsetIDGenerator = "902919af7c6d46bd6701b33e47308bad93d50cd10cdacaac739e5242819c4d7b" // GoExtVersion: 0.0.512 | ||||||
|  |  | ||||||
| const idlen = 24 | const idlen = 24 | ||||||
|  |  | ||||||
|   | |||||||
| @@ -1,12 +1,9 @@ | |||||||
| package models | package models | ||||||
|  |  | ||||||
| import ( | import ( | ||||||
| 	"context" | 	"encoding/json" | ||||||
| 	"github.com/jmoiron/sqlx" |  | ||||||
| 	"gogs.mikescher.com/BlackForestBytes/goext/langext" | 	"gogs.mikescher.com/BlackForestBytes/goext/langext" | ||||||
| 	"gogs.mikescher.com/BlackForestBytes/goext/sq" |  | ||||||
| 	"strings" | 	"strings" | ||||||
| 	"time" |  | ||||||
| ) | ) | ||||||
|  |  | ||||||
| type TokenPerm string //@enum:type | type TokenPerm string //@enum:type | ||||||
| @@ -45,17 +42,53 @@ func ParseTokenPermissionList(input string) TokenPermissionList { | |||||||
| 	return r | 	return r | ||||||
| } | } | ||||||
|  |  | ||||||
|  | func (e TokenPermissionList) MarshalToDB(v TokenPermissionList) (string, error) { | ||||||
|  | 	return v.String(), nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (e TokenPermissionList) UnmarshalToModel(v string) (TokenPermissionList, error) { | ||||||
|  | 	return ParseTokenPermissionList(v), nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (t TokenPermissionList) MarshalJSON() ([]byte, error) { | ||||||
|  | 	return json.Marshal(t.String()) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | type ChannelIDArr []ChannelID | ||||||
|  |  | ||||||
|  | func (t ChannelIDArr) MarshalToDB(v ChannelIDArr) (string, error) { | ||||||
|  | 	return strings.Join(langext.ArrMap(v, func(v ChannelID) string { return v.String() }), ";"), nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (t ChannelIDArr) UnmarshalToModel(v string) (ChannelIDArr, error) { | ||||||
|  | 	channels := make([]ChannelID, 0) | ||||||
|  | 	if strings.TrimSpace(v) != "" { | ||||||
|  | 		channels = langext.ArrMap(strings.Split(v, ";"), func(v string) ChannelID { return ChannelID(v) }) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return channels, nil | ||||||
|  | } | ||||||
|  |  | ||||||
| type KeyToken struct { | type KeyToken struct { | ||||||
| 	KeyTokenID        KeyTokenID | 	KeyTokenID        KeyTokenID          `db:"keytoken_id"          json:"keytoken_id"` | ||||||
| 	Name              string | 	Name              string              `db:"name"                 json:"name"` | ||||||
| 	TimestampCreated  time.Time | 	TimestampCreated  SCNTime             `db:"timestamp_created"    json:"timestamp_created"` | ||||||
| 	TimestampLastUsed *time.Time | 	TimestampLastUsed *SCNTime            `db:"timestamp_lastused"   json:"timestamp_lastused"` | ||||||
| 	OwnerUserID       UserID | 	OwnerUserID       UserID              `db:"owner_user_id"        json:"owner_user_id"` | ||||||
| 	AllChannels       bool | 	AllChannels       bool                `db:"all_channels"         json:"all_channels"` | ||||||
| 	Channels          []ChannelID // can also be owned by other user (needs active subscription) | 	Channels          ChannelIDArr        `db:"channels"             json:"channels"` | ||||||
| 	Token             string | 	Token             string              `db:"token"                json:"token"               jsonfilter:"INCLUDE_TOKEN"` | ||||||
| 	Permissions       TokenPermissionList | 	Permissions       TokenPermissionList `db:"permissions"          json:"permissions"` | ||||||
| 	MessagesSent      int | 	MessagesSent      int                 `db:"messages_sent"        json:"messages_sent"` | ||||||
|  | } | ||||||
|  |  | ||||||
|  | type KeyTokenPreview struct { | ||||||
|  | 	KeyTokenID  KeyTokenID  `json:"keytoken_id"` | ||||||
|  | 	Name        string      `json:"name"` | ||||||
|  | 	OwnerUserID UserID      `json:"owner_user_id"` | ||||||
|  | 	AllChannels bool        `json:"all_channels"` | ||||||
|  | 	Channels    []ChannelID `json:"channels"` | ||||||
|  | 	Permissions string      `json:"permissions"` | ||||||
| } | } | ||||||
|  |  | ||||||
| func (k KeyToken) IsUserRead(uid UserID) bool { | func (k KeyToken) IsUserRead(uid UserID) bool { | ||||||
| @@ -78,22 +111,8 @@ func (k KeyToken) IsChannelMessagesSend(c Channel) bool { | |||||||
| 	return (k.AllChannels == true || langext.InArray(c.ChannelID, k.Channels)) && k.OwnerUserID == c.OwnerUserID && k.Permissions.Any(PermAdmin, PermChannelSend) | 	return (k.AllChannels == true || langext.InArray(c.ChannelID, k.Channels)) && k.OwnerUserID == c.OwnerUserID && k.Permissions.Any(PermAdmin, PermChannelSend) | ||||||
| } | } | ||||||
|  |  | ||||||
| func (k KeyToken) JSON() KeyTokenJSON { | func (k KeyToken) Preview() KeyTokenPreview { | ||||||
| 	return KeyTokenJSON{ | 	return KeyTokenPreview{ | ||||||
| 		KeyTokenID:        k.KeyTokenID, |  | ||||||
| 		Name:              k.Name, |  | ||||||
| 		TimestampCreated:  k.TimestampCreated, |  | ||||||
| 		TimestampLastUsed: k.TimestampLastUsed, |  | ||||||
| 		OwnerUserID:       k.OwnerUserID, |  | ||||||
| 		AllChannels:       k.AllChannels, |  | ||||||
| 		Channels:          k.Channels, |  | ||||||
| 		Permissions:       k.Permissions.String(), |  | ||||||
| 		MessagesSent:      k.MessagesSent, |  | ||||||
| 	} |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (k KeyToken) JSONPreview() KeyTokenPreviewJSON { |  | ||||||
| 	return KeyTokenPreviewJSON{ |  | ||||||
| 		KeyTokenID:  k.KeyTokenID, | 		KeyTokenID:  k.KeyTokenID, | ||||||
| 		Name:        k.Name, | 		Name:        k.Name, | ||||||
| 		OwnerUserID: k.OwnerUserID, | 		OwnerUserID: k.OwnerUserID, | ||||||
| @@ -102,86 +121,3 @@ func (k KeyToken) JSONPreview() KeyTokenPreviewJSON { | |||||||
| 		Permissions: k.Permissions.String(), | 		Permissions: k.Permissions.String(), | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
|  |  | ||||||
| type KeyTokenJSON struct { |  | ||||||
| 	KeyTokenID        KeyTokenID  `json:"keytoken_id"` |  | ||||||
| 	Name              string      `json:"name"` |  | ||||||
| 	TimestampCreated  time.Time   `json:"timestamp_created"` |  | ||||||
| 	TimestampLastUsed *time.Time  `json:"timestamp_lastused"` |  | ||||||
| 	OwnerUserID       UserID      `json:"owner_user_id"` |  | ||||||
| 	AllChannels       bool        `json:"all_channels"` |  | ||||||
| 	Channels          []ChannelID `json:"channels"` |  | ||||||
| 	Permissions       string      `json:"permissions"` |  | ||||||
| 	MessagesSent      int         `json:"messages_sent"` |  | ||||||
| } |  | ||||||
|  |  | ||||||
| type KeyTokenWithTokenJSON struct { |  | ||||||
| 	KeyTokenJSON |  | ||||||
| 	Token string `json:"token"` |  | ||||||
| } |  | ||||||
|  |  | ||||||
| type KeyTokenPreviewJSON struct { |  | ||||||
| 	KeyTokenID  KeyTokenID  `json:"keytoken_id"` |  | ||||||
| 	Name        string      `json:"name"` |  | ||||||
| 	OwnerUserID UserID      `json:"owner_user_id"` |  | ||||||
| 	AllChannels bool        `json:"all_channels"` |  | ||||||
| 	Channels    []ChannelID `json:"channels"` |  | ||||||
| 	Permissions string      `json:"permissions"` |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (j KeyTokenJSON) WithToken(tok string) KeyTokenWithTokenJSON { |  | ||||||
| 	return KeyTokenWithTokenJSON{ |  | ||||||
| 		KeyTokenJSON: j, |  | ||||||
| 		Token:        tok, |  | ||||||
| 	} |  | ||||||
| } |  | ||||||
|  |  | ||||||
| type KeyTokenDB struct { |  | ||||||
| 	KeyTokenID        KeyTokenID `db:"keytoken_id"` |  | ||||||
| 	Name              string     `db:"name"` |  | ||||||
| 	TimestampCreated  int64      `db:"timestamp_created"` |  | ||||||
| 	TimestampLastUsed *int64     `db:"timestamp_lastused"` |  | ||||||
| 	OwnerUserID       UserID     `db:"owner_user_id"` |  | ||||||
| 	AllChannels       bool       `db:"all_channels"` |  | ||||||
| 	Channels          string     `db:"channels"` |  | ||||||
| 	Token             string     `db:"token"` |  | ||||||
| 	Permissions       string     `db:"permissions"` |  | ||||||
| 	MessagesSent      int        `db:"messages_sent"` |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (k KeyTokenDB) Model() KeyToken { |  | ||||||
|  |  | ||||||
| 	channels := make([]ChannelID, 0) |  | ||||||
| 	if strings.TrimSpace(k.Channels) != "" { |  | ||||||
| 		channels = langext.ArrMap(strings.Split(k.Channels, ";"), func(v string) ChannelID { return ChannelID(v) }) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	return KeyToken{ |  | ||||||
| 		KeyTokenID:        k.KeyTokenID, |  | ||||||
| 		Name:              k.Name, |  | ||||||
| 		TimestampCreated:  timeFromMilli(k.TimestampCreated), |  | ||||||
| 		TimestampLastUsed: timeOptFromMilli(k.TimestampLastUsed), |  | ||||||
| 		OwnerUserID:       k.OwnerUserID, |  | ||||||
| 		AllChannels:       k.AllChannels, |  | ||||||
| 		Channels:          channels, |  | ||||||
| 		Token:             k.Token, |  | ||||||
| 		Permissions:       ParseTokenPermissionList(k.Permissions), |  | ||||||
| 		MessagesSent:      k.MessagesSent, |  | ||||||
| 	} |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func DecodeKeyToken(ctx context.Context, q sq.Queryable, r *sqlx.Rows) (KeyToken, error) { |  | ||||||
| 	data, err := sq.ScanSingle[KeyTokenDB](ctx, q, r, sq.SModeFast, sq.Safe, true) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return KeyToken{}, err |  | ||||||
| 	} |  | ||||||
| 	return data.Model(), nil |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func DecodeKeyTokens(ctx context.Context, q sq.Queryable, r *sqlx.Rows) ([]KeyToken, error) { |  | ||||||
| 	data, err := sq.ScanAll[KeyTokenDB](ctx, q, r, sq.SModeFast, sq.Safe, true) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return nil, err |  | ||||||
| 	} |  | ||||||
| 	return langext.ArrMap(data, func(v KeyTokenDB) KeyToken { return v.Model() }), nil |  | ||||||
| } |  | ||||||
|   | |||||||
							
								
								
									
										9
									
								
								scnserver/models/lock.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										9
									
								
								scnserver/models/lock.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,9 @@ | |||||||
|  | package models | ||||||
|  |  | ||||||
|  | type TransactionLockMode string //@enum:type | ||||||
|  |  | ||||||
|  | const ( | ||||||
|  | 	TLockNone      TransactionLockMode = "NONE" | ||||||
|  | 	TLockRead      TransactionLockMode = "READ" | ||||||
|  | 	TLockReadWrite TransactionLockMode = "READ_WRITE" | ||||||
|  | ) | ||||||
| @@ -1,11 +1,8 @@ | |||||||
| package models | package models | ||||||
|  |  | ||||||
| import ( | import ( | ||||||
| 	"context" |  | ||||||
| 	"fmt" | 	"fmt" | ||||||
| 	"github.com/jmoiron/sqlx" |  | ||||||
| 	"gogs.mikescher.com/BlackForestBytes/goext/langext" | 	"gogs.mikescher.com/BlackForestBytes/goext/langext" | ||||||
| 	"gogs.mikescher.com/BlackForestBytes/goext/sq" |  | ||||||
| 	"time" | 	"time" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| @@ -15,60 +12,45 @@ const ( | |||||||
| ) | ) | ||||||
|  |  | ||||||
| type Message struct { | type Message struct { | ||||||
| 	MessageID           MessageID | 	MessageID           MessageID  `db:"message_id"              json:"message_id"` | ||||||
| 	SenderUserID        UserID // user that sent the message (this is also the owner of the channel that contains it) | 	SenderUserID        UserID     `db:"sender_user_id"          json:"sender_user_id"` // user that sent the message (this is also the owner of the channel that contains it) | ||||||
| 	ChannelInternalName string | 	ChannelInternalName string     `db:"channel_internal_name"   json:"channel_internal_name"` | ||||||
| 	ChannelID           ChannelID | 	ChannelID           ChannelID  `db:"channel_id"              json:"channel_id"` | ||||||
| 	SenderName          *string | 	SenderName          *string    `db:"sender_name"             json:"sender_name"` | ||||||
| 	SenderIP            string | 	SenderIP            string     `db:"sender_ip"               json:"sender_ip"` | ||||||
| 	TimestampReal       time.Time | 	TimestampReal       SCNTime    `db:"timestamp_real"          json:"-"` | ||||||
| 	TimestampClient     *time.Time | 	TimestampClient     *SCNTime   `db:"timestamp_client"        json:"-"` | ||||||
| 	Title               string | 	Title               string     `db:"title"                   json:"title"` | ||||||
| 	Content             *string | 	Content             *string    `db:"content"                 json:"content"` | ||||||
| 	Priority            int | 	Priority            int        `db:"priority"                json:"priority"` | ||||||
| 	UserMessageID       *string | 	UserMessageID       *string    `db:"usr_message_id"          json:"usr_message_id"` | ||||||
| 	UsedKeyID           KeyTokenID | 	UsedKeyID           KeyTokenID `db:"used_key_id"             json:"used_key_id"` | ||||||
| 	Deleted             bool | 	Deleted             bool       `db:"deleted"                 json:"-"` | ||||||
|  |  | ||||||
|  | 	MessageExtra `db:"-"` // fields that are not in DB and are set on PreMarshal | ||||||
| } | } | ||||||
|  |  | ||||||
| func (m Message) FullJSON() MessageJSON { | type MessageExtra struct { | ||||||
| 	return MessageJSON{ | 	Timestamp SCNTime `db:"-" json:"timestamp"` | ||||||
| 		MessageID:           m.MessageID, | 	Trimmed   bool    `db:"-" json:"trimmed"` | ||||||
| 		SenderUserID:        m.SenderUserID, |  | ||||||
| 		ChannelInternalName: m.ChannelInternalName, |  | ||||||
| 		ChannelID:           m.ChannelID, |  | ||||||
| 		SenderName:          m.SenderName, |  | ||||||
| 		SenderIP:            m.SenderIP, |  | ||||||
| 		Timestamp:           m.Timestamp().Format(time.RFC3339Nano), |  | ||||||
| 		Title:               m.Title, |  | ||||||
| 		Content:             m.Content, |  | ||||||
| 		Priority:            m.Priority, |  | ||||||
| 		UserMessageID:       m.UserMessageID, |  | ||||||
| 		UsedKeyID:           m.UsedKeyID, |  | ||||||
| 		Trimmed:             false, |  | ||||||
| 	} |  | ||||||
| } | } | ||||||
|  |  | ||||||
| func (m Message) TrimmedJSON() MessageJSON { | func (u *Message) PreMarshal() Message { | ||||||
| 	return MessageJSON{ | 	u.MessageExtra.Timestamp = NewSCNTime(u.Timestamp()) | ||||||
| 		MessageID:           m.MessageID, | 	return *u | ||||||
| 		SenderUserID:        m.SenderUserID, | } | ||||||
| 		ChannelInternalName: m.ChannelInternalName, |  | ||||||
| 		ChannelID:           m.ChannelID, | func (m Message) Trim() Message { | ||||||
| 		SenderName:          m.SenderName, | 	r := m | ||||||
| 		SenderIP:            m.SenderIP, | 	if !r.Trimmed && r.NeedsTrim() { | ||||||
| 		Timestamp:           m.Timestamp().Format(time.RFC3339Nano), | 		r.Content = r.TrimmedContent() | ||||||
| 		Title:               m.Title, | 		r.MessageExtra.Trimmed = true | ||||||
| 		Content:             m.TrimmedContent(), |  | ||||||
| 		Priority:            m.Priority, |  | ||||||
| 		UserMessageID:       m.UserMessageID, |  | ||||||
| 		UsedKeyID:           m.UsedKeyID, |  | ||||||
| 		Trimmed:             m.NeedsTrim(), |  | ||||||
| 	} | 	} | ||||||
|  | 	return r.PreMarshal() | ||||||
| } | } | ||||||
|  |  | ||||||
| func (m Message) Timestamp() time.Time { | func (m Message) Timestamp() time.Time { | ||||||
| 	return langext.Coalesce(m.TimestampClient, m.TimestampReal) | 	return langext.Coalesce(m.TimestampClient, m.TimestampReal).Time() | ||||||
| } | } | ||||||
|  |  | ||||||
| func (m Message) NeedsTrim() bool { | func (m Message) NeedsTrim() bool { | ||||||
| @@ -102,71 +84,3 @@ func (m Message) FormatNotificationTitle(user User, channel Channel) string { | |||||||
|  |  | ||||||
| 	return fmt.Sprintf("[%s] %s", channel.DisplayName, m.Title) | 	return fmt.Sprintf("[%s] %s", channel.DisplayName, m.Title) | ||||||
| } | } | ||||||
|  |  | ||||||
| type MessageJSON struct { |  | ||||||
| 	MessageID           MessageID  `json:"message_id"` |  | ||||||
| 	SenderUserID        UserID     `json:"sender_user_id"` |  | ||||||
| 	ChannelInternalName string     `json:"channel_internal_name"` |  | ||||||
| 	ChannelID           ChannelID  `json:"channel_id"` |  | ||||||
| 	SenderName          *string    `json:"sender_name"` |  | ||||||
| 	SenderIP            string     `json:"sender_ip"` |  | ||||||
| 	Timestamp           string     `json:"timestamp"` |  | ||||||
| 	Title               string     `json:"title"` |  | ||||||
| 	Content             *string    `json:"content"` |  | ||||||
| 	Priority            int        `json:"priority"` |  | ||||||
| 	UserMessageID       *string    `json:"usr_message_id"` |  | ||||||
| 	UsedKeyID           KeyTokenID `json:"used_key_id"` |  | ||||||
| 	Trimmed             bool       `json:"trimmed"` |  | ||||||
| } |  | ||||||
|  |  | ||||||
| type MessageDB struct { |  | ||||||
| 	MessageID           MessageID  `db:"message_id"` |  | ||||||
| 	SenderUserID        UserID     `db:"sender_user_id"` |  | ||||||
| 	ChannelInternalName string     `db:"channel_internal_name"` |  | ||||||
| 	ChannelID           ChannelID  `db:"channel_id"` |  | ||||||
| 	SenderName          *string    `db:"sender_name"` |  | ||||||
| 	SenderIP            string     `db:"sender_ip"` |  | ||||||
| 	TimestampReal       int64      `db:"timestamp_real"` |  | ||||||
| 	TimestampClient     *int64     `db:"timestamp_client"` |  | ||||||
| 	Title               string     `db:"title"` |  | ||||||
| 	Content             *string    `db:"content"` |  | ||||||
| 	Priority            int        `db:"priority"` |  | ||||||
| 	UserMessageID       *string    `db:"usr_message_id"` |  | ||||||
| 	UsedKeyID           KeyTokenID `db:"used_key_id"` |  | ||||||
| 	Deleted             int        `db:"deleted"` |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (m MessageDB) Model() Message { |  | ||||||
| 	return Message{ |  | ||||||
| 		MessageID:           m.MessageID, |  | ||||||
| 		SenderUserID:        m.SenderUserID, |  | ||||||
| 		ChannelInternalName: m.ChannelInternalName, |  | ||||||
| 		ChannelID:           m.ChannelID, |  | ||||||
| 		SenderName:          m.SenderName, |  | ||||||
| 		SenderIP:            m.SenderIP, |  | ||||||
| 		TimestampReal:       timeFromMilli(m.TimestampReal), |  | ||||||
| 		TimestampClient:     timeOptFromMilli(m.TimestampClient), |  | ||||||
| 		Title:               m.Title, |  | ||||||
| 		Content:             m.Content, |  | ||||||
| 		Priority:            m.Priority, |  | ||||||
| 		UserMessageID:       m.UserMessageID, |  | ||||||
| 		UsedKeyID:           m.UsedKeyID, |  | ||||||
| 		Deleted:             m.Deleted != 0, |  | ||||||
| 	} |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func DecodeMessage(ctx context.Context, q sq.Queryable, r *sqlx.Rows) (Message, error) { |  | ||||||
| 	data, err := sq.ScanSingle[MessageDB](ctx, q, r, sq.SModeFast, sq.Safe, true) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return Message{}, err |  | ||||||
| 	} |  | ||||||
| 	return data.Model(), nil |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func DecodeMessages(ctx context.Context, q sq.Queryable, r *sqlx.Rows) ([]Message, error) { |  | ||||||
| 	data, err := sq.ScanAll[MessageDB](ctx, q, r, sq.SModeFast, sq.Safe, true) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return nil, err |  | ||||||
| 	} |  | ||||||
| 	return langext.ArrMap(data, func(v MessageDB) Message { return v.Model() }), nil |  | ||||||
| } |  | ||||||
|   | |||||||
| @@ -1,188 +1,27 @@ | |||||||
| package models | package models | ||||||
|  |  | ||||||
| import ( |  | ||||||
| 	"context" |  | ||||||
| 	"github.com/jmoiron/sqlx" |  | ||||||
| 	"gogs.mikescher.com/BlackForestBytes/goext/langext" |  | ||||||
| 	"gogs.mikescher.com/BlackForestBytes/goext/sq" |  | ||||||
| 	"gogs.mikescher.com/BlackForestBytes/goext/timeext" |  | ||||||
| 	"time" |  | ||||||
| ) |  | ||||||
|  |  | ||||||
| type RequestLog struct { | type RequestLog struct { | ||||||
| 	RequestID           RequestID | 	RequestID           RequestID   `db:"request_id"             json:"requestLog_id"` | ||||||
| 	Method              string | 	Method              string      `db:"method"                 json:"method"` | ||||||
| 	URI                 string | 	URI                 string      `db:"uri"                    json:"uri"` | ||||||
| 	UserAgent           *string | 	UserAgent           *string     `db:"user_agent"             json:"user_agent"` | ||||||
| 	Authentication      *string | 	Authentication      *string     `db:"authentication"         json:"authentication"` | ||||||
| 	RequestBody         *string | 	RequestBody         *string     `db:"request_body"           json:"request_body"` | ||||||
| 	RequestBodySize     int64 | 	RequestBodySize     int64       `db:"request_body_size"      json:"request_body_size"` | ||||||
| 	RequestContentType  string | 	RequestContentType  string      `db:"request_content_type"   json:"request_content_type"` | ||||||
| 	RemoteIP            string | 	RemoteIP            string      `db:"remote_ip"              json:"remote_ip"` | ||||||
| 	KeyID               *KeyTokenID | 	KeyID               *KeyTokenID `db:"key_id"                 json:"key_id"` | ||||||
| 	UserID              *UserID | 	UserID              *UserID     `db:"userid"                 json:"userid"` | ||||||
| 	Permissions         *string | 	Permissions         *string     `db:"permissions"            json:"permissions"` | ||||||
| 	ResponseStatuscode  *int64 | 	ResponseStatuscode  *int64      `db:"response_statuscode"    json:"response_statuscode"` | ||||||
| 	ResponseBodySize    *int64 | 	ResponseBodySize    *int64      `db:"response_body_size"     json:"response_body_size"` | ||||||
| 	ResponseBody        *string | 	ResponseBody        *string     `db:"response_body"          json:"response_body"` | ||||||
| 	ResponseContentType string | 	ResponseContentType string      `db:"response_content_type"  json:"response_content_type"` | ||||||
| 	RetryCount          int64 | 	RetryCount          int64       `db:"retry_count"            json:"retry_count"` | ||||||
| 	Panicked            bool | 	Panicked            bool        `db:"panicked"               json:"panicked"` | ||||||
| 	PanicStr            *string | 	PanicStr            *string     `db:"panic_str"              json:"panic_str"` | ||||||
| 	ProcessingTime      time.Duration | 	ProcessingTime      SCNDuration `db:"processing_time"        json:"processing_time"` | ||||||
| 	TimestampCreated    time.Time | 	TimestampCreated    SCNTime     `db:"timestamp_created"      json:"timestamp_created"` | ||||||
| 	TimestampStart      time.Time | 	TimestampStart      SCNTime     `db:"timestamp_start"        json:"timestamp_start"` | ||||||
| 	TimestampFinish     time.Time | 	TimestampFinish     SCNTime     `db:"timestamp_finish"       json:"timestamp_finish"` | ||||||
| } |  | ||||||
|  |  | ||||||
| func (c RequestLog) JSON() RequestLogJSON { |  | ||||||
| 	return RequestLogJSON{ |  | ||||||
| 		RequestID:           c.RequestID, |  | ||||||
| 		Method:              c.Method, |  | ||||||
| 		URI:                 c.URI, |  | ||||||
| 		UserAgent:           c.UserAgent, |  | ||||||
| 		Authentication:      c.Authentication, |  | ||||||
| 		RequestBody:         c.RequestBody, |  | ||||||
| 		RequestBodySize:     c.RequestBodySize, |  | ||||||
| 		RequestContentType:  c.RequestContentType, |  | ||||||
| 		RemoteIP:            c.RemoteIP, |  | ||||||
| 		KeyID:               c.KeyID, |  | ||||||
| 		UserID:              c.UserID, |  | ||||||
| 		Permissions:         c.Permissions, |  | ||||||
| 		ResponseStatuscode:  c.ResponseStatuscode, |  | ||||||
| 		ResponseBodySize:    c.ResponseBodySize, |  | ||||||
| 		ResponseBody:        c.ResponseBody, |  | ||||||
| 		ResponseContentType: c.ResponseContentType, |  | ||||||
| 		RetryCount:          c.RetryCount, |  | ||||||
| 		Panicked:            c.Panicked, |  | ||||||
| 		PanicStr:            c.PanicStr, |  | ||||||
| 		ProcessingTime:      c.ProcessingTime.Seconds(), |  | ||||||
| 		TimestampCreated:    c.TimestampCreated.Format(time.RFC3339Nano), |  | ||||||
| 		TimestampStart:      c.TimestampStart.Format(time.RFC3339Nano), |  | ||||||
| 		TimestampFinish:     c.TimestampFinish.Format(time.RFC3339Nano), |  | ||||||
| 	} |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (c RequestLog) DB() RequestLogDB { |  | ||||||
| 	return RequestLogDB{ |  | ||||||
| 		RequestID:           c.RequestID, |  | ||||||
| 		Method:              c.Method, |  | ||||||
| 		URI:                 c.URI, |  | ||||||
| 		UserAgent:           c.UserAgent, |  | ||||||
| 		Authentication:      c.Authentication, |  | ||||||
| 		RequestBody:         c.RequestBody, |  | ||||||
| 		RequestBodySize:     c.RequestBodySize, |  | ||||||
| 		RequestContentType:  c.RequestContentType, |  | ||||||
| 		RemoteIP:            c.RemoteIP, |  | ||||||
| 		KeyID:               c.KeyID, |  | ||||||
| 		UserID:              c.UserID, |  | ||||||
| 		Permissions:         c.Permissions, |  | ||||||
| 		ResponseStatuscode:  c.ResponseStatuscode, |  | ||||||
| 		ResponseBodySize:    c.ResponseBodySize, |  | ||||||
| 		ResponseBody:        c.ResponseBody, |  | ||||||
| 		ResponseContentType: c.ResponseContentType, |  | ||||||
| 		RetryCount:          c.RetryCount, |  | ||||||
| 		Panicked:            langext.Conditional[int64](c.Panicked, 1, 0), |  | ||||||
| 		PanicStr:            c.PanicStr, |  | ||||||
| 		ProcessingTime:      c.ProcessingTime.Milliseconds(), |  | ||||||
| 		TimestampCreated:    c.TimestampCreated.UnixMilli(), |  | ||||||
| 		TimestampStart:      c.TimestampStart.UnixMilli(), |  | ||||||
| 		TimestampFinish:     c.TimestampFinish.UnixMilli(), |  | ||||||
| 	} |  | ||||||
| } |  | ||||||
|  |  | ||||||
| type RequestLogJSON struct { |  | ||||||
| 	RequestID           RequestID   `json:"requestLog_id"` |  | ||||||
| 	Method              string      `json:"method"` |  | ||||||
| 	URI                 string      `json:"uri"` |  | ||||||
| 	UserAgent           *string     `json:"user_agent"` |  | ||||||
| 	Authentication      *string     `json:"authentication"` |  | ||||||
| 	RequestBody         *string     `json:"request_body"` |  | ||||||
| 	RequestBodySize     int64       `json:"request_body_size"` |  | ||||||
| 	RequestContentType  string      `json:"request_content_type"` |  | ||||||
| 	RemoteIP            string      `json:"remote_ip"` |  | ||||||
| 	KeyID               *KeyTokenID `json:"key_id"` |  | ||||||
| 	UserID              *UserID     `json:"userid"` |  | ||||||
| 	Permissions         *string     `json:"permissions"` |  | ||||||
| 	ResponseStatuscode  *int64      `json:"response_statuscode"` |  | ||||||
| 	ResponseBodySize    *int64      `json:"response_body_size"` |  | ||||||
| 	ResponseBody        *string     `json:"response_body"` |  | ||||||
| 	ResponseContentType string      `json:"response_content_type"` |  | ||||||
| 	RetryCount          int64       `json:"retry_count"` |  | ||||||
| 	Panicked            bool        `json:"panicked"` |  | ||||||
| 	PanicStr            *string     `json:"panic_str"` |  | ||||||
| 	ProcessingTime      float64     `json:"processing_time"` |  | ||||||
| 	TimestampCreated    string      `json:"timestamp_created"` |  | ||||||
| 	TimestampStart      string      `json:"timestamp_start"` |  | ||||||
| 	TimestampFinish     string      `json:"timestamp_finish"` |  | ||||||
| } |  | ||||||
|  |  | ||||||
| type RequestLogDB struct { |  | ||||||
| 	RequestID           RequestID   `db:"request_id"` |  | ||||||
| 	Method              string      `db:"method"` |  | ||||||
| 	URI                 string      `db:"uri"` |  | ||||||
| 	UserAgent           *string     `db:"user_agent"` |  | ||||||
| 	Authentication      *string     `db:"authentication"` |  | ||||||
| 	RequestBody         *string     `db:"request_body"` |  | ||||||
| 	RequestBodySize     int64       `db:"request_body_size"` |  | ||||||
| 	RequestContentType  string      `db:"request_content_type"` |  | ||||||
| 	RemoteIP            string      `db:"remote_ip"` |  | ||||||
| 	KeyID               *KeyTokenID `db:"key_id"` |  | ||||||
| 	UserID              *UserID     `db:"userid"` |  | ||||||
| 	Permissions         *string     `db:"permissions"` |  | ||||||
| 	ResponseStatuscode  *int64      `db:"response_statuscode"` |  | ||||||
| 	ResponseBodySize    *int64      `db:"response_body_size"` |  | ||||||
| 	ResponseBody        *string     `db:"response_body"` |  | ||||||
| 	ResponseContentType string      `db:"response_content_type"` |  | ||||||
| 	RetryCount          int64       `db:"retry_count"` |  | ||||||
| 	Panicked            int64       `db:"panicked"` |  | ||||||
| 	PanicStr            *string     `db:"panic_str"` |  | ||||||
| 	ProcessingTime      int64       `db:"processing_time"` |  | ||||||
| 	TimestampCreated    int64       `db:"timestamp_created"` |  | ||||||
| 	TimestampStart      int64       `db:"timestamp_start"` |  | ||||||
| 	TimestampFinish     int64       `db:"timestamp_finish"` |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (c RequestLogDB) Model() RequestLog { |  | ||||||
| 	return RequestLog{ |  | ||||||
| 		RequestID:           c.RequestID, |  | ||||||
| 		Method:              c.Method, |  | ||||||
| 		URI:                 c.URI, |  | ||||||
| 		UserAgent:           c.UserAgent, |  | ||||||
| 		Authentication:      c.Authentication, |  | ||||||
| 		RequestBody:         c.RequestBody, |  | ||||||
| 		RequestBodySize:     c.RequestBodySize, |  | ||||||
| 		RequestContentType:  c.RequestContentType, |  | ||||||
| 		RemoteIP:            c.RemoteIP, |  | ||||||
| 		KeyID:               c.KeyID, |  | ||||||
| 		UserID:              c.UserID, |  | ||||||
| 		Permissions:         c.Permissions, |  | ||||||
| 		ResponseStatuscode:  c.ResponseStatuscode, |  | ||||||
| 		ResponseBodySize:    c.ResponseBodySize, |  | ||||||
| 		ResponseBody:        c.ResponseBody, |  | ||||||
| 		ResponseContentType: c.ResponseContentType, |  | ||||||
| 		RetryCount:          c.RetryCount, |  | ||||||
| 		Panicked:            c.Panicked != 0, |  | ||||||
| 		PanicStr:            c.PanicStr, |  | ||||||
| 		ProcessingTime:      timeext.FromMilliseconds(c.ProcessingTime), |  | ||||||
| 		TimestampCreated:    timeFromMilli(c.TimestampCreated), |  | ||||||
| 		TimestampStart:      timeFromMilli(c.TimestampStart), |  | ||||||
| 		TimestampFinish:     timeFromMilli(c.TimestampFinish), |  | ||||||
| 	} |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func DecodeRequestLog(ctx context.Context, q sq.Queryable, r *sqlx.Rows) (RequestLog, error) { |  | ||||||
| 	data, err := sq.ScanSingle[RequestLogDB](ctx, q, r, sq.SModeFast, sq.Safe, true) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return RequestLog{}, err |  | ||||||
| 	} |  | ||||||
| 	return data.Model(), nil |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func DecodeRequestLogs(ctx context.Context, q sq.Queryable, r *sqlx.Rows) ([]RequestLog, error) { |  | ||||||
| 	data, err := sq.ScanAll[RequestLogDB](ctx, q, r, sq.SModeFast, sq.Safe, true) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return nil, err |  | ||||||
| 	} |  | ||||||
| 	return langext.ArrMap(data, func(v RequestLogDB) RequestLog { return v.Model() }), nil |  | ||||||
| } | } | ||||||
|   | |||||||
| @@ -1,13 +1,5 @@ | |||||||
| package models | package models | ||||||
|  |  | ||||||
| import ( |  | ||||||
| 	"context" |  | ||||||
| 	"github.com/jmoiron/sqlx" |  | ||||||
| 	"gogs.mikescher.com/BlackForestBytes/goext/langext" |  | ||||||
| 	"gogs.mikescher.com/BlackForestBytes/goext/sq" |  | ||||||
| 	"time" |  | ||||||
| ) |  | ||||||
|  |  | ||||||
| // [!] subscriptions are read-access to channels, | // [!] subscriptions are read-access to channels, | ||||||
| // | // | ||||||
| // The set of subscriptions specifies which messages the ListMessages() API call returns | // The set of subscriptions specifies which messages the ListMessages() API call returns | ||||||
| @@ -16,71 +8,11 @@ import ( | |||||||
| // (use keytokens for write-access) | // (use keytokens for write-access) | ||||||
|  |  | ||||||
| type Subscription struct { | type Subscription struct { | ||||||
| 	SubscriptionID      SubscriptionID | 	SubscriptionID      SubscriptionID `db:"subscription_id"          json:"subscription_id"` | ||||||
| 	SubscriberUserID    UserID | 	SubscriberUserID    UserID         `db:"subscriber_user_id"       json:"subscriber_user_id"` | ||||||
| 	ChannelOwnerUserID  UserID | 	ChannelOwnerUserID  UserID         `db:"channel_owner_user_id"    json:"channel_owner_user_id"` | ||||||
| 	ChannelID           ChannelID | 	ChannelID           ChannelID      `db:"channel_id"               json:"channel_id"` | ||||||
| 	ChannelInternalName string | 	ChannelInternalName string         `db:"channel_internal_name"    json:"channel_internal_name"` | ||||||
| 	TimestampCreated    time.Time | 	TimestampCreated    SCNTime        `db:"timestamp_created"        json:"timestamp_created"` | ||||||
| 	Confirmed           bool | 	Confirmed           bool           `db:"confirmed"                json:"confirmed"` | ||||||
| } |  | ||||||
|  |  | ||||||
| func (s Subscription) JSON() SubscriptionJSON { |  | ||||||
| 	return SubscriptionJSON{ |  | ||||||
| 		SubscriptionID:      s.SubscriptionID, |  | ||||||
| 		SubscriberUserID:    s.SubscriberUserID, |  | ||||||
| 		ChannelOwnerUserID:  s.ChannelOwnerUserID, |  | ||||||
| 		ChannelID:           s.ChannelID, |  | ||||||
| 		ChannelInternalName: s.ChannelInternalName, |  | ||||||
| 		TimestampCreated:    s.TimestampCreated.Format(time.RFC3339Nano), |  | ||||||
| 		Confirmed:           s.Confirmed, |  | ||||||
| 	} |  | ||||||
| } |  | ||||||
|  |  | ||||||
| type SubscriptionJSON struct { |  | ||||||
| 	SubscriptionID      SubscriptionID `json:"subscription_id"` |  | ||||||
| 	SubscriberUserID    UserID         `json:"subscriber_user_id"` |  | ||||||
| 	ChannelOwnerUserID  UserID         `json:"channel_owner_user_id"` |  | ||||||
| 	ChannelID           ChannelID      `json:"channel_id"` |  | ||||||
| 	ChannelInternalName string         `json:"channel_internal_name"` |  | ||||||
| 	TimestampCreated    string         `json:"timestamp_created"` |  | ||||||
| 	Confirmed           bool           `json:"confirmed"` |  | ||||||
| } |  | ||||||
|  |  | ||||||
| type SubscriptionDB struct { |  | ||||||
| 	SubscriptionID      SubscriptionID `db:"subscription_id"` |  | ||||||
| 	SubscriberUserID    UserID         `db:"subscriber_user_id"` |  | ||||||
| 	ChannelOwnerUserID  UserID         `db:"channel_owner_user_id"` |  | ||||||
| 	ChannelID           ChannelID      `db:"channel_id"` |  | ||||||
| 	ChannelInternalName string         `db:"channel_internal_name"` |  | ||||||
| 	TimestampCreated    int64          `db:"timestamp_created"` |  | ||||||
| 	Confirmed           int            `db:"confirmed"` |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (s SubscriptionDB) Model() Subscription { |  | ||||||
| 	return Subscription{ |  | ||||||
| 		SubscriptionID:      s.SubscriptionID, |  | ||||||
| 		SubscriberUserID:    s.SubscriberUserID, |  | ||||||
| 		ChannelOwnerUserID:  s.ChannelOwnerUserID, |  | ||||||
| 		ChannelID:           s.ChannelID, |  | ||||||
| 		ChannelInternalName: s.ChannelInternalName, |  | ||||||
| 		TimestampCreated:    timeFromMilli(s.TimestampCreated), |  | ||||||
| 		Confirmed:           s.Confirmed != 0, |  | ||||||
| 	} |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func DecodeSubscription(ctx context.Context, q sq.Queryable, r *sqlx.Rows) (Subscription, error) { |  | ||||||
| 	data, err := sq.ScanSingle[SubscriptionDB](ctx, q, r, sq.SModeFast, sq.Safe, true) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return Subscription{}, err |  | ||||||
| 	} |  | ||||||
| 	return data.Model(), nil |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func DecodeSubscriptions(ctx context.Context, q sq.Queryable, r *sqlx.Rows) ([]Subscription, error) { |  | ||||||
| 	data, err := sq.ScanAll[SubscriptionDB](ctx, q, r, sq.SModeFast, sq.Safe, true) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return nil, err |  | ||||||
| 	} |  | ||||||
| 	return langext.ArrMap(data, func(v SubscriptionDB) Subscription { return v.Model() }), nil |  | ||||||
| } | } | ||||||
|   | |||||||
							
								
								
									
										65
									
								
								scnserver/models/time.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										65
									
								
								scnserver/models/time.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,65 @@ | |||||||
|  | package models | ||||||
|  |  | ||||||
|  | import ( | ||||||
|  | 	"encoding/json" | ||||||
|  | 	"gogs.mikescher.com/BlackForestBytes/goext/langext" | ||||||
|  | 	"gogs.mikescher.com/BlackForestBytes/goext/rfctime" | ||||||
|  | 	"time" | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | type SCNTime time.Time | ||||||
|  |  | ||||||
|  | func (t SCNTime) MarshalToDB(v SCNTime) (int64, error) { | ||||||
|  | 	return v.Time().UnixMilli(), nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (t SCNTime) UnmarshalToModel(v int64) (SCNTime, error) { | ||||||
|  | 	return NewSCNTime(time.UnixMilli(v)), nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (t SCNTime) Time() time.Time { | ||||||
|  | 	return time.Time(t) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (t *SCNTime) UnmarshalJSON(data []byte) error { | ||||||
|  | 	str := "" | ||||||
|  | 	if err := json.Unmarshal(data, &str); err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  | 	t0, err := time.Parse(time.RFC3339Nano, str) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  | 	*t = SCNTime(t0) | ||||||
|  | 	return nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (t SCNTime) MarshalJSON() ([]byte, error) { | ||||||
|  | 	str := t.Time().Format(time.RFC3339Nano) | ||||||
|  | 	return json.Marshal(str) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func NewSCNTime(t time.Time) SCNTime { | ||||||
|  | 	return SCNTime(t) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func NewSCNTimePtr(t *time.Time) *SCNTime { | ||||||
|  | 	if t == nil { | ||||||
|  | 		return nil | ||||||
|  | 	} | ||||||
|  | 	return langext.Ptr(SCNTime(*t)) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func NowSCNTime() SCNTime { | ||||||
|  | 	return SCNTime(time.Now()) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func tt(v rfctime.AnyTime) time.Time { | ||||||
|  | 	if r, ok := v.(time.Time); ok { | ||||||
|  | 		return r | ||||||
|  | 	} | ||||||
|  | 	if r, ok := v.(rfctime.RFCTime); ok { | ||||||
|  | 		return r.Time() | ||||||
|  | 	} | ||||||
|  | 	return time.Unix(0, v.UnixNano()).In(v.Location()) | ||||||
|  | } | ||||||
| @@ -2,38 +2,63 @@ package models | |||||||
|  |  | ||||||
| import ( | import ( | ||||||
| 	scn "blackforestbytes.com/simplecloudnotifier" | 	scn "blackforestbytes.com/simplecloudnotifier" | ||||||
| 	"context" |  | ||||||
| 	"github.com/jmoiron/sqlx" |  | ||||||
| 	"gogs.mikescher.com/BlackForestBytes/goext/langext" |  | ||||||
| 	"gogs.mikescher.com/BlackForestBytes/goext/sq" |  | ||||||
| 	"time" |  | ||||||
| ) | ) | ||||||
|  |  | ||||||
| type User struct { | type User struct { | ||||||
| 	UserID            UserID | 	UserID            UserID   `db:"user_id"              json:"user_id"` | ||||||
| 	Username          *string | 	Username          *string  `db:"username"             json:"username"` | ||||||
| 	TimestampCreated  time.Time | 	TimestampCreated  SCNTime  `db:"timestamp_created"    json:"timestamp_created"` | ||||||
| 	TimestampLastRead *time.Time | 	TimestampLastRead *SCNTime `db:"timestamp_lastread"   json:"timestamp_lastread"` | ||||||
| 	TimestampLastSent *time.Time | 	TimestampLastSent *SCNTime `db:"timestamp_lastsent"   json:"timestamp_lastsent"` | ||||||
| 	MessagesSent      int | 	MessagesSent      int      `db:"messages_sent"        json:"messages_sent"` | ||||||
| 	QuotaUsed         int | 	QuotaUsed         int      `db:"quota_used"           json:"quota_used"` | ||||||
| 	QuotaUsedDay      *string | 	QuotaUsedDay      *string  `db:"quota_used_day"       json:"-"` | ||||||
| 	IsPro             bool | 	IsPro             bool     `db:"is_pro"               json:"is_pro"` | ||||||
| 	ProToken          *string | 	ProToken          *string  `db:"pro_token"            json:"-"` | ||||||
|  |  | ||||||
|  | 	UserExtra `db:"-"` // fields that are not in DB and are set on PreMarshal | ||||||
| } | } | ||||||
|  |  | ||||||
| func (u User) JSON() UserJSON { | type UserExtra struct { | ||||||
| 	return UserJSON{ | 	QuotaRemaining              int    `json:"quota_remaining"` | ||||||
| 		UserID:                      u.UserID, | 	QuotaPerDay                 int    `json:"quota_max"` | ||||||
| 		Username:                    u.Username, | 	DefaultChannel              string `json:"default_channel"` | ||||||
| 		TimestampCreated:            u.TimestampCreated.Format(time.RFC3339Nano), | 	MaxBodySize                 int    `json:"max_body_size"` | ||||||
| 		TimestampLastRead:           timeOptFmt(u.TimestampLastRead, time.RFC3339Nano), | 	MaxTitleLength              int    `json:"max_title_length"` | ||||||
| 		TimestampLastSent:           timeOptFmt(u.TimestampLastSent, time.RFC3339Nano), | 	DefaultPriority             int    `json:"default_priority"` | ||||||
| 		MessagesSent:                u.MessagesSent, | 	MaxChannelNameLength        int    `json:"max_channel_name_length"` | ||||||
| 		QuotaUsed:                   u.QuotaUsedToday(), | 	MaxChannelDescriptionLength int    `json:"max_channel_description_length"` | ||||||
|  | 	MaxSenderNameLength         int    `json:"max_sender_name_length"` | ||||||
|  | 	MaxUserMessageIDLength      int    `json:"max_user_message_id_length"` | ||||||
|  | } | ||||||
|  |  | ||||||
|  | type UserPreview struct { | ||||||
|  | 	UserID   UserID  `json:"user_id"` | ||||||
|  | 	Username *string `json:"username"` | ||||||
|  | } | ||||||
|  |  | ||||||
|  | type UserWithClientsAndKeys struct { | ||||||
|  | 	User | ||||||
|  | 	Clients  []Client `json:"clients"` | ||||||
|  | 	SendKey  string   `json:"send_key"` | ||||||
|  | 	ReadKey  string   `json:"read_key"` | ||||||
|  | 	AdminKey string   `json:"admin_key"` | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (u User) WithClients(clients []Client, ak string, sk string, rk string) UserWithClientsAndKeys { | ||||||
|  | 	return UserWithClientsAndKeys{ | ||||||
|  | 		User:     u.PreMarshal(), | ||||||
|  | 		Clients:  clients, | ||||||
|  | 		SendKey:  sk, | ||||||
|  | 		ReadKey:  rk, | ||||||
|  | 		AdminKey: ak, | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (u *User) PreMarshal() User { | ||||||
|  | 	u.UserExtra = UserExtra{ | ||||||
| 		QuotaPerDay:                 u.QuotaPerDay(), | 		QuotaPerDay:                 u.QuotaPerDay(), | ||||||
| 		QuotaRemaining:              u.QuotaRemainingToday(), | 		QuotaRemaining:              u.QuotaRemainingToday(), | ||||||
| 		IsPro:                       u.IsPro, |  | ||||||
| 		DefaultChannel:              u.DefaultChannel(), | 		DefaultChannel:              u.DefaultChannel(), | ||||||
| 		MaxBodySize:                 u.MaxContentLength(), | 		MaxBodySize:                 u.MaxContentLength(), | ||||||
| 		MaxTitleLength:              u.MaxTitleLength(), | 		MaxTitleLength:              u.MaxTitleLength(), | ||||||
| @@ -43,16 +68,7 @@ func (u User) JSON() UserJSON { | |||||||
| 		MaxSenderNameLength:         u.MaxSenderNameLength(), | 		MaxSenderNameLength:         u.MaxSenderNameLength(), | ||||||
| 		MaxUserMessageIDLength:      u.MaxUserMessageIDLength(), | 		MaxUserMessageIDLength:      u.MaxUserMessageIDLength(), | ||||||
| 	} | 	} | ||||||
| } | 	return *u | ||||||
|  |  | ||||||
| func (u User) JSONWithClients(clients []Client, ak string, sk string, rk string) UserJSONWithClientsAndKeys { |  | ||||||
| 	return UserJSONWithClientsAndKeys{ |  | ||||||
| 		UserJSON: u.JSON(), |  | ||||||
| 		Clients:  langext.ArrMap(clients, func(v Client) ClientJSON { return v.JSON() }), |  | ||||||
| 		SendKey:  sk, |  | ||||||
| 		ReadKey:  rk, |  | ||||||
| 		AdminKey: ak, |  | ||||||
| 	} |  | ||||||
| } | } | ||||||
|  |  | ||||||
| func (u User) MaxContentLength() int { | func (u User) MaxContentLength() int { | ||||||
| @@ -116,86 +132,9 @@ func (u User) MaxTimestampDiffHours() int { | |||||||
| 	return 24 | 	return 24 | ||||||
| } | } | ||||||
|  |  | ||||||
| func (u User) JSONPreview() UserPreviewJSON { | func (u User) JSONPreview() UserPreview { | ||||||
| 	return UserPreviewJSON{ | 	return UserPreview{ | ||||||
| 		UserID:   u.UserID, | 		UserID:   u.UserID, | ||||||
| 		Username: u.Username, | 		Username: u.Username, | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
|  |  | ||||||
| type UserJSON struct { |  | ||||||
| 	UserID                      UserID  `json:"user_id"` |  | ||||||
| 	Username                    *string `json:"username"` |  | ||||||
| 	TimestampCreated            string  `json:"timestamp_created"` |  | ||||||
| 	TimestampLastRead           *string `json:"timestamp_lastread"` |  | ||||||
| 	TimestampLastSent           *string `json:"timestamp_lastsent"` |  | ||||||
| 	MessagesSent                int     `json:"messages_sent"` |  | ||||||
| 	QuotaUsed                   int     `json:"quota_used"` |  | ||||||
| 	QuotaRemaining              int     `json:"quota_remaining"` |  | ||||||
| 	QuotaPerDay                 int     `json:"quota_max"` |  | ||||||
| 	IsPro                       bool    `json:"is_pro"` |  | ||||||
| 	DefaultChannel              string  `json:"default_channel"` |  | ||||||
| 	MaxBodySize                 int     `json:"max_body_size"` |  | ||||||
| 	MaxTitleLength              int     `json:"max_title_length"` |  | ||||||
| 	DefaultPriority             int     `json:"default_priority"` |  | ||||||
| 	MaxChannelNameLength        int     `json:"max_channel_name_length"` |  | ||||||
| 	MaxChannelDescriptionLength int     `json:"max_channel_description_length"` |  | ||||||
| 	MaxSenderNameLength         int     `json:"max_sender_name_length"` |  | ||||||
| 	MaxUserMessageIDLength      int     `json:"max_user_message_id_length"` |  | ||||||
| } |  | ||||||
|  |  | ||||||
| type UserPreviewJSON struct { |  | ||||||
| 	UserID   UserID  `json:"user_id"` |  | ||||||
| 	Username *string `json:"username"` |  | ||||||
| } |  | ||||||
|  |  | ||||||
| type UserJSONWithClientsAndKeys struct { |  | ||||||
| 	UserJSON |  | ||||||
| 	Clients  []ClientJSON `json:"clients"` |  | ||||||
| 	SendKey  string       `json:"send_key"` |  | ||||||
| 	ReadKey  string       `json:"read_key"` |  | ||||||
| 	AdminKey string       `json:"admin_key"` |  | ||||||
| } |  | ||||||
|  |  | ||||||
| type UserDB struct { |  | ||||||
| 	UserID            UserID  `db:"user_id"` |  | ||||||
| 	Username          *string `db:"username"` |  | ||||||
| 	TimestampCreated  int64   `db:"timestamp_created"` |  | ||||||
| 	TimestampLastRead *int64  `db:"timestamp_lastread"` |  | ||||||
| 	TimestampLastSent *int64  `db:"timestamp_lastsent"` |  | ||||||
| 	MessagesSent      int     `db:"messages_sent"` |  | ||||||
| 	QuotaUsed         int     `db:"quota_used"` |  | ||||||
| 	QuotaUsedDay      *string `db:"quota_used_day"` |  | ||||||
| 	IsPro             bool    `db:"is_pro"` |  | ||||||
| 	ProToken          *string `db:"pro_token"` |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (u UserDB) Model() User { |  | ||||||
| 	return User{ |  | ||||||
| 		UserID:            u.UserID, |  | ||||||
| 		Username:          u.Username, |  | ||||||
| 		TimestampCreated:  timeFromMilli(u.TimestampCreated), |  | ||||||
| 		TimestampLastRead: timeOptFromMilli(u.TimestampLastRead), |  | ||||||
| 		TimestampLastSent: timeOptFromMilli(u.TimestampLastSent), |  | ||||||
| 		MessagesSent:      u.MessagesSent, |  | ||||||
| 		QuotaUsed:         u.QuotaUsed, |  | ||||||
| 		QuotaUsedDay:      u.QuotaUsedDay, |  | ||||||
| 		IsPro:             u.IsPro, |  | ||||||
| 	} |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func DecodeUser(ctx context.Context, q sq.Queryable, r *sqlx.Rows) (User, error) { |  | ||||||
| 	data, err := sq.ScanSingle[UserDB](ctx, q, r, sq.SModeFast, sq.Safe, true) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return User{}, err |  | ||||||
| 	} |  | ||||||
| 	return data.Model(), nil |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func DecodeUsers(ctx context.Context, q sq.Queryable, r *sqlx.Rows) ([]User, error) { |  | ||||||
| 	data, err := sq.ScanAll[UserDB](ctx, q, r, sq.SModeFast, sq.Safe, true) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return nil, err |  | ||||||
| 	} |  | ||||||
| 	return langext.ArrMap(data, func(v UserDB) User { return v.Model() }), nil |  | ||||||
| } |  | ||||||
|   | |||||||
| @@ -2,6 +2,7 @@ package models | |||||||
|  |  | ||||||
| import ( | import ( | ||||||
| 	"gogs.mikescher.com/BlackForestBytes/goext/langext" | 	"gogs.mikescher.com/BlackForestBytes/goext/langext" | ||||||
|  | 	"gogs.mikescher.com/BlackForestBytes/goext/sq" | ||||||
| 	"time" | 	"time" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| @@ -23,3 +24,10 @@ func timeOptFromMilli(millis *int64) *time.Time { | |||||||
| func timeFromMilli(millis int64) time.Time { | func timeFromMilli(millis int64) time.Time { | ||||||
| 	return time.UnixMilli(millis) | 	return time.UnixMilli(millis) | ||||||
| } | } | ||||||
|  |  | ||||||
|  | func RegisterConverter(db sq.DB) { | ||||||
|  | 	db.RegisterConverter(sq.NewAutoDBTypeConverter(SCNTime{})) | ||||||
|  | 	db.RegisterConverter(sq.NewAutoDBTypeConverter(SCNDuration(0))) | ||||||
|  | 	db.RegisterConverter(sq.NewAutoDBTypeConverter(TokenPermissionList{})) | ||||||
|  | 	db.RegisterConverter(sq.NewAutoDBTypeConverter(ChannelIDArr{})) | ||||||
|  | } | ||||||
|   | |||||||
| @@ -1,10 +1,10 @@ | |||||||
| package swagger | package swagger | ||||||
|  |  | ||||||
| import ( | import ( | ||||||
| 	"blackforestbytes.com/simplecloudnotifier/api/ginresp" |  | ||||||
| 	"embed" | 	"embed" | ||||||
| 	_ "embed" | 	_ "embed" | ||||||
| 	"github.com/gin-gonic/gin" | 	"github.com/gin-gonic/gin" | ||||||
|  | 	"gogs.mikescher.com/BlackForestBytes/goext/ginext" | ||||||
| 	"net/http" | 	"net/http" | ||||||
| 	"strings" | 	"strings" | ||||||
| ) | ) | ||||||
| @@ -46,26 +46,28 @@ func getAsset(fn string) ([]byte, string, bool) { | |||||||
| 	return data, mime, true | 	return data, mime, true | ||||||
| } | } | ||||||
|  |  | ||||||
| func Handle(g *gin.Context) ginresp.HTTPResponse { | func Handle(pctx ginext.PreContext) ginext.HTTPResponse { | ||||||
| 	type uri struct { | 	type uri struct { | ||||||
| 		Filename string `uri:"sub"` | 		Filename string `uri:"sub"` | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	var u uri | 	var u uri | ||||||
| 	if err := g.ShouldBindUri(&u); err != nil { | 	ctx, _, errResp := pctx.URI(&u).Start() | ||||||
| 		return ginresp.JSON(http.StatusBadRequest, gin.H{"error": err.Error()}) | 	if errResp != nil { | ||||||
|  | 		return *errResp | ||||||
| 	} | 	} | ||||||
|  | 	defer ctx.Cancel() | ||||||
|  |  | ||||||
| 	u.Filename = strings.TrimLeft(u.Filename, "/") | 	u.Filename = strings.TrimLeft(u.Filename, "/") | ||||||
|  |  | ||||||
| 	if u.Filename == "" { | 	if u.Filename == "" { | ||||||
| 		index, _, _ := getAsset("index.html") | 		index, _, _ := getAsset("index.html") | ||||||
| 		return ginresp.Data(http.StatusOK, "text/html", index) | 		return ginext.Data(http.StatusOK, "text/html", index) | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	if data, mime, ok := getAsset(u.Filename); ok { | 	if data, mime, ok := getAsset(u.Filename); ok { | ||||||
| 		return ginresp.Data(http.StatusOK, mime, data) | 		return ginext.Data(http.StatusOK, mime, data) | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	return ginresp.JSON(http.StatusNotFound, gin.H{"error": "AssetNotFound", "filename": u.Filename}) | 	return ginext.JSON(http.StatusNotFound, gin.H{"error": "AssetNotFound", "filename": u.Filename}) | ||||||
| } | } | ||||||
|   | |||||||
| @@ -19,63 +19,39 @@ | |||||||
|                 "parameters": [ |                 "parameters": [ | ||||||
|                     { |                     { | ||||||
|                         "type": "string", |                         "type": "string", | ||||||
|                         "example": "test", |  | ||||||
|                         "name": "channel", |  | ||||||
|                         "in": "query" |  | ||||||
|                     }, |  | ||||||
|                     { |  | ||||||
|                         "type": "string", |  | ||||||
|                         "example": "This is a message", |  | ||||||
|                         "name": "content", |                         "name": "content", | ||||||
|                         "in": "query" |                         "in": "query" | ||||||
|                     }, |                     }, | ||||||
|                     { |                     { | ||||||
|                         "type": "string", |                         "type": "string", | ||||||
|                         "example": "P3TNH8mvv14fm", |  | ||||||
|                         "name": "key", |  | ||||||
|                         "in": "query" |  | ||||||
|                     }, |  | ||||||
|                     { |  | ||||||
|                         "type": "string", |  | ||||||
|                         "example": "db8b0e6a-a08c-4646", |  | ||||||
|                         "name": "msg_id", |                         "name": "msg_id", | ||||||
|                         "in": "query" |                         "in": "query" | ||||||
|                     }, |                     }, | ||||||
|                     { |                     { | ||||||
|                         "enum": [ |  | ||||||
|                             0, |  | ||||||
|                             1, |  | ||||||
|                             2 |  | ||||||
|                         ], |  | ||||||
|                         "type": "integer", |                         "type": "integer", | ||||||
|                         "example": 1, |  | ||||||
|                         "name": "priority", |                         "name": "priority", | ||||||
|                         "in": "query" |                         "in": "query" | ||||||
|                     }, |                     }, | ||||||
|                     { |  | ||||||
|                         "type": "string", |  | ||||||
|                         "example": "example-server", |  | ||||||
|                         "name": "sender_name", |  | ||||||
|                         "in": "query" |  | ||||||
|                     }, |  | ||||||
|                     { |                     { | ||||||
|                         "type": "number", |                         "type": "number", | ||||||
|                         "example": 1669824037, |  | ||||||
|                         "name": "timestamp", |                         "name": "timestamp", | ||||||
|                         "in": "query" |                         "in": "query" | ||||||
|                     }, |                     }, | ||||||
|                     { |                     { | ||||||
|                         "type": "string", |                         "type": "string", | ||||||
|                         "example": "Hello World", |  | ||||||
|                         "name": "title", |                         "name": "title", | ||||||
|                         "in": "query" |                         "in": "query" | ||||||
|                     }, |                     }, | ||||||
|                     { |                     { | ||||||
|                         "type": "string", |                         "type": "integer", | ||||||
|                         "example": "7725", |  | ||||||
|                         "name": "user_id", |                         "name": "user_id", | ||||||
|                         "in": "query" |                         "in": "query" | ||||||
|                     }, |                     }, | ||||||
|  |                     { | ||||||
|  |                         "type": "string", | ||||||
|  |                         "name": "user_key", | ||||||
|  |                         "in": "query" | ||||||
|  |                     }, | ||||||
|                     { |                     { | ||||||
|                         "description": " ", |                         "description": " ", | ||||||
|                         "name": "post_body", |                         "name": "post_body", | ||||||
| @@ -86,62 +62,38 @@ | |||||||
|                     }, |                     }, | ||||||
|                     { |                     { | ||||||
|                         "type": "string", |                         "type": "string", | ||||||
|                         "example": "test", |  | ||||||
|                         "name": "channel", |  | ||||||
|                         "in": "formData" |  | ||||||
|                     }, |  | ||||||
|                     { |  | ||||||
|                         "type": "string", |  | ||||||
|                         "example": "This is a message", |  | ||||||
|                         "name": "content", |                         "name": "content", | ||||||
|                         "in": "formData" |                         "in": "formData" | ||||||
|                     }, |                     }, | ||||||
|                     { |                     { | ||||||
|                         "type": "string", |                         "type": "string", | ||||||
|                         "example": "P3TNH8mvv14fm", |  | ||||||
|                         "name": "key", |  | ||||||
|                         "in": "formData" |  | ||||||
|                     }, |  | ||||||
|                     { |  | ||||||
|                         "type": "string", |  | ||||||
|                         "example": "db8b0e6a-a08c-4646", |  | ||||||
|                         "name": "msg_id", |                         "name": "msg_id", | ||||||
|                         "in": "formData" |                         "in": "formData" | ||||||
|                     }, |                     }, | ||||||
|                     { |                     { | ||||||
|                         "enum": [ |  | ||||||
|                             0, |  | ||||||
|                             1, |  | ||||||
|                             2 |  | ||||||
|                         ], |  | ||||||
|                         "type": "integer", |                         "type": "integer", | ||||||
|                         "example": 1, |  | ||||||
|                         "name": "priority", |                         "name": "priority", | ||||||
|                         "in": "formData" |                         "in": "formData" | ||||||
|                     }, |                     }, | ||||||
|                     { |  | ||||||
|                         "type": "string", |  | ||||||
|                         "example": "example-server", |  | ||||||
|                         "name": "sender_name", |  | ||||||
|                         "in": "formData" |  | ||||||
|                     }, |  | ||||||
|                     { |                     { | ||||||
|                         "type": "number", |                         "type": "number", | ||||||
|                         "example": 1669824037, |  | ||||||
|                         "name": "timestamp", |                         "name": "timestamp", | ||||||
|                         "in": "formData" |                         "in": "formData" | ||||||
|                     }, |                     }, | ||||||
|                     { |                     { | ||||||
|                         "type": "string", |                         "type": "string", | ||||||
|                         "example": "Hello World", |  | ||||||
|                         "name": "title", |                         "name": "title", | ||||||
|                         "in": "formData" |                         "in": "formData" | ||||||
|                     }, |                     }, | ||||||
|                     { |                     { | ||||||
|                         "type": "string", |                         "type": "integer", | ||||||
|                         "example": "7725", |  | ||||||
|                         "name": "user_id", |                         "name": "user_id", | ||||||
|                         "in": "formData" |                         "in": "formData" | ||||||
|  |                     }, | ||||||
|  |                     { | ||||||
|  |                         "type": "string", | ||||||
|  |                         "name": "user_key", | ||||||
|  |                         "in": "formData" | ||||||
|                     } |                     } | ||||||
|                 ], |                 ], | ||||||
|                 "responses": { |                 "responses": { | ||||||
| @@ -978,7 +930,7 @@ | |||||||
|                     "200": { |                     "200": { | ||||||
|                         "description": "OK", |                         "description": "OK", | ||||||
|                         "schema": { |                         "schema": { | ||||||
|                             "$ref": "#/definitions/models.MessageJSON" |                             "$ref": "#/definitions/models.Message" | ||||||
|                         } |                         } | ||||||
|                     }, |                     }, | ||||||
|                     "400": { |                     "400": { | ||||||
| @@ -1027,7 +979,7 @@ | |||||||
|                     "200": { |                     "200": { | ||||||
|                         "description": "OK", |                         "description": "OK", | ||||||
|                         "schema": { |                         "schema": { | ||||||
|                             "$ref": "#/definitions/models.MessageJSON" |                             "$ref": "#/definitions/models.Message" | ||||||
|                         } |                         } | ||||||
|                     }, |                     }, | ||||||
|                     "400": { |                     "400": { | ||||||
| @@ -1077,7 +1029,7 @@ | |||||||
|                     "200": { |                     "200": { | ||||||
|                         "description": "OK", |                         "description": "OK", | ||||||
|                         "schema": { |                         "schema": { | ||||||
|                             "$ref": "#/definitions/models.ChannelPreviewJSON" |                             "$ref": "#/definitions/models.ChannelPreview" | ||||||
|                         } |                         } | ||||||
|                     }, |                     }, | ||||||
|                     "400": { |                     "400": { | ||||||
| @@ -1127,7 +1079,7 @@ | |||||||
|                     "200": { |                     "200": { | ||||||
|                         "description": "OK", |                         "description": "OK", | ||||||
|                         "schema": { |                         "schema": { | ||||||
|                             "$ref": "#/definitions/models.KeyTokenPreviewJSON" |                             "$ref": "#/definitions/models.KeyTokenPreview" | ||||||
|                         } |                         } | ||||||
|                     }, |                     }, | ||||||
|                     "400": { |                     "400": { | ||||||
| @@ -1177,7 +1129,7 @@ | |||||||
|                     "200": { |                     "200": { | ||||||
|                         "description": "OK", |                         "description": "OK", | ||||||
|                         "schema": { |                         "schema": { | ||||||
|                             "$ref": "#/definitions/models.UserPreviewJSON" |                             "$ref": "#/definitions/models.UserPreview" | ||||||
|                         } |                         } | ||||||
|                     }, |                     }, | ||||||
|                     "400": { |                     "400": { | ||||||
| @@ -1228,7 +1180,7 @@ | |||||||
|                     "200": { |                     "200": { | ||||||
|                         "description": "OK", |                         "description": "OK", | ||||||
|                         "schema": { |                         "schema": { | ||||||
|                             "$ref": "#/definitions/models.UserJSONWithClientsAndKeys" |                             "$ref": "#/definitions/models.UserWithClientsAndKeys" | ||||||
|                         } |                         } | ||||||
|                     }, |                     }, | ||||||
|                     "400": { |                     "400": { | ||||||
| @@ -1266,7 +1218,7 @@ | |||||||
|                     "200": { |                     "200": { | ||||||
|                         "description": "OK", |                         "description": "OK", | ||||||
|                         "schema": { |                         "schema": { | ||||||
|                             "$ref": "#/definitions/models.UserJSON" |                             "$ref": "#/definitions/models.User" | ||||||
|                         } |                         } | ||||||
|                     }, |                     }, | ||||||
|                     "400": { |                     "400": { | ||||||
| @@ -1331,7 +1283,7 @@ | |||||||
|                     "200": { |                     "200": { | ||||||
|                         "description": "OK", |                         "description": "OK", | ||||||
|                         "schema": { |                         "schema": { | ||||||
|                             "$ref": "#/definitions/models.UserJSON" |                             "$ref": "#/definitions/models.User" | ||||||
|                         } |                         } | ||||||
|                     }, |                     }, | ||||||
|                     "400": { |                     "400": { | ||||||
| @@ -1445,7 +1397,7 @@ | |||||||
|                     "200": { |                     "200": { | ||||||
|                         "description": "OK", |                         "description": "OK", | ||||||
|                         "schema": { |                         "schema": { | ||||||
|                             "$ref": "#/definitions/models.ChannelWithSubscriptionJSON" |                             "$ref": "#/definitions/models.ChannelWithSubscription" | ||||||
|                         } |                         } | ||||||
|                     }, |                     }, | ||||||
|                     "400": { |                     "400": { | ||||||
| @@ -1502,7 +1454,7 @@ | |||||||
|                     "200": { |                     "200": { | ||||||
|                         "description": "OK", |                         "description": "OK", | ||||||
|                         "schema": { |                         "schema": { | ||||||
|                             "$ref": "#/definitions/models.ChannelWithSubscriptionJSON" |                             "$ref": "#/definitions/models.ChannelWithSubscription" | ||||||
|                         } |                         } | ||||||
|                     }, |                     }, | ||||||
|                     "400": { |                     "400": { | ||||||
| @@ -1581,7 +1533,7 @@ | |||||||
|                     "200": { |                     "200": { | ||||||
|                         "description": "OK", |                         "description": "OK", | ||||||
|                         "schema": { |                         "schema": { | ||||||
|                             "$ref": "#/definitions/models.ChannelWithSubscriptionJSON" |                             "$ref": "#/definitions/models.ChannelWithSubscription" | ||||||
|                         } |                         } | ||||||
|                     }, |                     }, | ||||||
|                     "400": { |                     "400": { | ||||||
| @@ -1816,7 +1768,7 @@ | |||||||
|                     "200": { |                     "200": { | ||||||
|                         "description": "OK", |                         "description": "OK", | ||||||
|                         "schema": { |                         "schema": { | ||||||
|                             "$ref": "#/definitions/models.ClientJSON" |                             "$ref": "#/definitions/models.Client" | ||||||
|                         } |                         } | ||||||
|                     }, |                     }, | ||||||
|                     "400": { |                     "400": { | ||||||
| @@ -1867,7 +1819,7 @@ | |||||||
|                     "200": { |                     "200": { | ||||||
|                         "description": "OK", |                         "description": "OK", | ||||||
|                         "schema": { |                         "schema": { | ||||||
|                             "$ref": "#/definitions/models.ClientJSON" |                             "$ref": "#/definitions/models.Client" | ||||||
|                         } |                         } | ||||||
|                     }, |                     }, | ||||||
|                     "400": { |                     "400": { | ||||||
| @@ -1922,7 +1874,7 @@ | |||||||
|                     "200": { |                     "200": { | ||||||
|                         "description": "OK", |                         "description": "OK", | ||||||
|                         "schema": { |                         "schema": { | ||||||
|                             "$ref": "#/definitions/models.ClientJSON" |                             "$ref": "#/definitions/models.Client" | ||||||
|                         } |                         } | ||||||
|                     }, |                     }, | ||||||
|                     "400": { |                     "400": { | ||||||
| @@ -1994,7 +1946,7 @@ | |||||||
|                     "200": { |                     "200": { | ||||||
|                         "description": "OK", |                         "description": "OK", | ||||||
|                         "schema": { |                         "schema": { | ||||||
|                             "$ref": "#/definitions/models.ClientJSON" |                             "$ref": "#/definitions/models.Client" | ||||||
|                         } |                         } | ||||||
|                     }, |                     }, | ||||||
|                     "400": { |                     "400": { | ||||||
| @@ -2101,7 +2053,7 @@ | |||||||
|                     "200": { |                     "200": { | ||||||
|                         "description": "OK", |                         "description": "OK", | ||||||
|                         "schema": { |                         "schema": { | ||||||
|                             "$ref": "#/definitions/models.KeyTokenJSON" |                             "$ref": "#/definitions/models.KeyToken" | ||||||
|                         } |                         } | ||||||
|                     }, |                     }, | ||||||
|                     "400": { |                     "400": { | ||||||
| @@ -2159,7 +2111,7 @@ | |||||||
|                     "200": { |                     "200": { | ||||||
|                         "description": "OK", |                         "description": "OK", | ||||||
|                         "schema": { |                         "schema": { | ||||||
|                             "$ref": "#/definitions/models.KeyTokenWithTokenJSON" |                             "$ref": "#/definitions/models.KeyToken" | ||||||
|                         } |                         } | ||||||
|                     }, |                     }, | ||||||
|                     "400": { |                     "400": { | ||||||
| @@ -2217,7 +2169,7 @@ | |||||||
|                     "200": { |                     "200": { | ||||||
|                         "description": "OK", |                         "description": "OK", | ||||||
|                         "schema": { |                         "schema": { | ||||||
|                             "$ref": "#/definitions/models.KeyTokenJSON" |                             "$ref": "#/definitions/models.KeyToken" | ||||||
|                         } |                         } | ||||||
|                     }, |                     }, | ||||||
|                     "400": { |                     "400": { | ||||||
| @@ -2273,7 +2225,7 @@ | |||||||
|                     "200": { |                     "200": { | ||||||
|                         "description": "OK", |                         "description": "OK", | ||||||
|                         "schema": { |                         "schema": { | ||||||
|                             "$ref": "#/definitions/models.KeyTokenJSON" |                             "$ref": "#/definitions/models.KeyToken" | ||||||
|                         } |                         } | ||||||
|                     }, |                     }, | ||||||
|                     "400": { |                     "400": { | ||||||
| @@ -2336,7 +2288,7 @@ | |||||||
|                     "200": { |                     "200": { | ||||||
|                         "description": "OK", |                         "description": "OK", | ||||||
|                         "schema": { |                         "schema": { | ||||||
|                             "$ref": "#/definitions/models.KeyTokenJSON" |                             "$ref": "#/definitions/models.KeyToken" | ||||||
|                         } |                         } | ||||||
|                     }, |                     }, | ||||||
|                     "400": { |                     "400": { | ||||||
| @@ -2458,7 +2410,7 @@ | |||||||
|                     "200": { |                     "200": { | ||||||
|                         "description": "OK", |                         "description": "OK", | ||||||
|                         "schema": { |                         "schema": { | ||||||
|                             "$ref": "#/definitions/models.SubscriptionJSON" |                             "$ref": "#/definitions/models.Subscription" | ||||||
|                         } |                         } | ||||||
|                     }, |                     }, | ||||||
|                     "400": { |                     "400": { | ||||||
| @@ -2509,7 +2461,7 @@ | |||||||
|                     "200": { |                     "200": { | ||||||
|                         "description": "OK", |                         "description": "OK", | ||||||
|                         "schema": { |                         "schema": { | ||||||
|                             "$ref": "#/definitions/models.SubscriptionJSON" |                             "$ref": "#/definitions/models.Subscription" | ||||||
|                         } |                         } | ||||||
|                     }, |                     }, | ||||||
|                     "400": { |                     "400": { | ||||||
| @@ -2564,7 +2516,7 @@ | |||||||
|                     "200": { |                     "200": { | ||||||
|                         "description": "OK", |                         "description": "OK", | ||||||
|                         "schema": { |                         "schema": { | ||||||
|                             "$ref": "#/definitions/models.SubscriptionJSON" |                             "$ref": "#/definitions/models.Subscription" | ||||||
|                         } |                         } | ||||||
|                     }, |                     }, | ||||||
|                     "400": { |                     "400": { | ||||||
| @@ -2627,7 +2579,7 @@ | |||||||
|                     "200": { |                     "200": { | ||||||
|                         "description": "OK", |                         "description": "OK", | ||||||
|                         "schema": { |                         "schema": { | ||||||
|                             "$ref": "#/definitions/models.SubscriptionJSON" |                             "$ref": "#/definitions/models.Subscription" | ||||||
|                         } |                         } | ||||||
|                     }, |                     }, | ||||||
|                     "400": { |                     "400": { | ||||||
| @@ -2765,63 +2717,39 @@ | |||||||
|                 "parameters": [ |                 "parameters": [ | ||||||
|                     { |                     { | ||||||
|                         "type": "string", |                         "type": "string", | ||||||
|                         "example": "test", |  | ||||||
|                         "name": "channel", |  | ||||||
|                         "in": "query" |  | ||||||
|                     }, |  | ||||||
|                     { |  | ||||||
|                         "type": "string", |  | ||||||
|                         "example": "This is a message", |  | ||||||
|                         "name": "content", |                         "name": "content", | ||||||
|                         "in": "query" |                         "in": "query" | ||||||
|                     }, |                     }, | ||||||
|                     { |                     { | ||||||
|                         "type": "string", |                         "type": "string", | ||||||
|                         "example": "P3TNH8mvv14fm", |  | ||||||
|                         "name": "key", |  | ||||||
|                         "in": "query" |  | ||||||
|                     }, |  | ||||||
|                     { |  | ||||||
|                         "type": "string", |  | ||||||
|                         "example": "db8b0e6a-a08c-4646", |  | ||||||
|                         "name": "msg_id", |                         "name": "msg_id", | ||||||
|                         "in": "query" |                         "in": "query" | ||||||
|                     }, |                     }, | ||||||
|                     { |                     { | ||||||
|                         "enum": [ |  | ||||||
|                             0, |  | ||||||
|                             1, |  | ||||||
|                             2 |  | ||||||
|                         ], |  | ||||||
|                         "type": "integer", |                         "type": "integer", | ||||||
|                         "example": 1, |  | ||||||
|                         "name": "priority", |                         "name": "priority", | ||||||
|                         "in": "query" |                         "in": "query" | ||||||
|                     }, |                     }, | ||||||
|                     { |  | ||||||
|                         "type": "string", |  | ||||||
|                         "example": "example-server", |  | ||||||
|                         "name": "sender_name", |  | ||||||
|                         "in": "query" |  | ||||||
|                     }, |  | ||||||
|                     { |                     { | ||||||
|                         "type": "number", |                         "type": "number", | ||||||
|                         "example": 1669824037, |  | ||||||
|                         "name": "timestamp", |                         "name": "timestamp", | ||||||
|                         "in": "query" |                         "in": "query" | ||||||
|                     }, |                     }, | ||||||
|                     { |                     { | ||||||
|                         "type": "string", |                         "type": "string", | ||||||
|                         "example": "Hello World", |  | ||||||
|                         "name": "title", |                         "name": "title", | ||||||
|                         "in": "query" |                         "in": "query" | ||||||
|                     }, |                     }, | ||||||
|                     { |                     { | ||||||
|                         "type": "string", |                         "type": "integer", | ||||||
|                         "example": "7725", |  | ||||||
|                         "name": "user_id", |                         "name": "user_id", | ||||||
|                         "in": "query" |                         "in": "query" | ||||||
|                     }, |                     }, | ||||||
|  |                     { | ||||||
|  |                         "type": "string", | ||||||
|  |                         "name": "user_key", | ||||||
|  |                         "in": "query" | ||||||
|  |                     }, | ||||||
|                     { |                     { | ||||||
|                         "description": " ", |                         "description": " ", | ||||||
|                         "name": "post_body", |                         "name": "post_body", | ||||||
| @@ -2832,62 +2760,38 @@ | |||||||
|                     }, |                     }, | ||||||
|                     { |                     { | ||||||
|                         "type": "string", |                         "type": "string", | ||||||
|                         "example": "test", |  | ||||||
|                         "name": "channel", |  | ||||||
|                         "in": "formData" |  | ||||||
|                     }, |  | ||||||
|                     { |  | ||||||
|                         "type": "string", |  | ||||||
|                         "example": "This is a message", |  | ||||||
|                         "name": "content", |                         "name": "content", | ||||||
|                         "in": "formData" |                         "in": "formData" | ||||||
|                     }, |                     }, | ||||||
|                     { |                     { | ||||||
|                         "type": "string", |                         "type": "string", | ||||||
|                         "example": "P3TNH8mvv14fm", |  | ||||||
|                         "name": "key", |  | ||||||
|                         "in": "formData" |  | ||||||
|                     }, |  | ||||||
|                     { |  | ||||||
|                         "type": "string", |  | ||||||
|                         "example": "db8b0e6a-a08c-4646", |  | ||||||
|                         "name": "msg_id", |                         "name": "msg_id", | ||||||
|                         "in": "formData" |                         "in": "formData" | ||||||
|                     }, |                     }, | ||||||
|                     { |                     { | ||||||
|                         "enum": [ |  | ||||||
|                             0, |  | ||||||
|                             1, |  | ||||||
|                             2 |  | ||||||
|                         ], |  | ||||||
|                         "type": "integer", |                         "type": "integer", | ||||||
|                         "example": 1, |  | ||||||
|                         "name": "priority", |                         "name": "priority", | ||||||
|                         "in": "formData" |                         "in": "formData" | ||||||
|                     }, |                     }, | ||||||
|                     { |  | ||||||
|                         "type": "string", |  | ||||||
|                         "example": "example-server", |  | ||||||
|                         "name": "sender_name", |  | ||||||
|                         "in": "formData" |  | ||||||
|                     }, |  | ||||||
|                     { |                     { | ||||||
|                         "type": "number", |                         "type": "number", | ||||||
|                         "example": 1669824037, |  | ||||||
|                         "name": "timestamp", |                         "name": "timestamp", | ||||||
|                         "in": "formData" |                         "in": "formData" | ||||||
|                     }, |                     }, | ||||||
|                     { |                     { | ||||||
|                         "type": "string", |                         "type": "string", | ||||||
|                         "example": "Hello World", |  | ||||||
|                         "name": "title", |                         "name": "title", | ||||||
|                         "in": "formData" |                         "in": "formData" | ||||||
|                     }, |                     }, | ||||||
|                     { |                     { | ||||||
|                         "type": "string", |                         "type": "integer", | ||||||
|                         "example": "7725", |  | ||||||
|                         "name": "user_id", |                         "name": "user_id", | ||||||
|                         "in": "formData" |                         "in": "formData" | ||||||
|  |                     }, | ||||||
|  |                     { | ||||||
|  |                         "type": "string", | ||||||
|  |                         "name": "user_key", | ||||||
|  |                         "in": "formData" | ||||||
|                     } |                     } | ||||||
|                 ], |                 ], | ||||||
|                 "responses": { |                 "responses": { | ||||||
| @@ -2935,121 +2839,73 @@ | |||||||
|                 "parameters": [ |                 "parameters": [ | ||||||
|                     { |                     { | ||||||
|                         "type": "string", |                         "type": "string", | ||||||
|                         "example": "test", |  | ||||||
|                         "name": "channel", |  | ||||||
|                         "in": "query" |  | ||||||
|                     }, |  | ||||||
|                     { |  | ||||||
|                         "type": "string", |  | ||||||
|                         "example": "This is a message", |  | ||||||
|                         "name": "content", |                         "name": "content", | ||||||
|                         "in": "query" |                         "in": "query" | ||||||
|                     }, |                     }, | ||||||
|                     { |                     { | ||||||
|                         "type": "string", |                         "type": "string", | ||||||
|                         "example": "P3TNH8mvv14fm", |  | ||||||
|                         "name": "key", |  | ||||||
|                         "in": "query" |  | ||||||
|                     }, |  | ||||||
|                     { |  | ||||||
|                         "type": "string", |  | ||||||
|                         "example": "db8b0e6a-a08c-4646", |  | ||||||
|                         "name": "msg_id", |                         "name": "msg_id", | ||||||
|                         "in": "query" |                         "in": "query" | ||||||
|                     }, |                     }, | ||||||
|                     { |                     { | ||||||
|                         "enum": [ |  | ||||||
|                             0, |  | ||||||
|                             1, |  | ||||||
|                             2 |  | ||||||
|                         ], |  | ||||||
|                         "type": "integer", |                         "type": "integer", | ||||||
|                         "example": 1, |  | ||||||
|                         "name": "priority", |                         "name": "priority", | ||||||
|                         "in": "query" |                         "in": "query" | ||||||
|                     }, |                     }, | ||||||
|                     { |  | ||||||
|                         "type": "string", |  | ||||||
|                         "example": "example-server", |  | ||||||
|                         "name": "sender_name", |  | ||||||
|                         "in": "query" |  | ||||||
|                     }, |  | ||||||
|                     { |                     { | ||||||
|                         "type": "number", |                         "type": "number", | ||||||
|                         "example": 1669824037, |  | ||||||
|                         "name": "timestamp", |                         "name": "timestamp", | ||||||
|                         "in": "query" |                         "in": "query" | ||||||
|                     }, |                     }, | ||||||
|                     { |                     { | ||||||
|                         "type": "string", |                         "type": "string", | ||||||
|                         "example": "Hello World", |  | ||||||
|                         "name": "title", |                         "name": "title", | ||||||
|                         "in": "query" |                         "in": "query" | ||||||
|                     }, |                     }, | ||||||
|                     { |                     { | ||||||
|                         "type": "string", |                         "type": "integer", | ||||||
|                         "example": "7725", |  | ||||||
|                         "name": "user_id", |                         "name": "user_id", | ||||||
|                         "in": "query" |                         "in": "query" | ||||||
|                     }, |                     }, | ||||||
|                     { |                     { | ||||||
|                         "type": "string", |                         "type": "string", | ||||||
|                         "example": "test", |                         "name": "user_key", | ||||||
|                         "name": "channel", |                         "in": "query" | ||||||
|                         "in": "formData" |  | ||||||
|                     }, |                     }, | ||||||
|                     { |                     { | ||||||
|                         "type": "string", |                         "type": "string", | ||||||
|                         "example": "This is a message", |  | ||||||
|                         "name": "content", |                         "name": "content", | ||||||
|                         "in": "formData" |                         "in": "formData" | ||||||
|                     }, |                     }, | ||||||
|                     { |                     { | ||||||
|                         "type": "string", |                         "type": "string", | ||||||
|                         "example": "P3TNH8mvv14fm", |  | ||||||
|                         "name": "key", |  | ||||||
|                         "in": "formData" |  | ||||||
|                     }, |  | ||||||
|                     { |  | ||||||
|                         "type": "string", |  | ||||||
|                         "example": "db8b0e6a-a08c-4646", |  | ||||||
|                         "name": "msg_id", |                         "name": "msg_id", | ||||||
|                         "in": "formData" |                         "in": "formData" | ||||||
|                     }, |                     }, | ||||||
|                     { |                     { | ||||||
|                         "enum": [ |  | ||||||
|                             0, |  | ||||||
|                             1, |  | ||||||
|                             2 |  | ||||||
|                         ], |  | ||||||
|                         "type": "integer", |                         "type": "integer", | ||||||
|                         "example": 1, |  | ||||||
|                         "name": "priority", |                         "name": "priority", | ||||||
|                         "in": "formData" |                         "in": "formData" | ||||||
|                     }, |                     }, | ||||||
|                     { |  | ||||||
|                         "type": "string", |  | ||||||
|                         "example": "example-server", |  | ||||||
|                         "name": "sender_name", |  | ||||||
|                         "in": "formData" |  | ||||||
|                     }, |  | ||||||
|                     { |                     { | ||||||
|                         "type": "number", |                         "type": "number", | ||||||
|                         "example": 1669824037, |  | ||||||
|                         "name": "timestamp", |                         "name": "timestamp", | ||||||
|                         "in": "formData" |                         "in": "formData" | ||||||
|                     }, |                     }, | ||||||
|                     { |                     { | ||||||
|                         "type": "string", |                         "type": "string", | ||||||
|                         "example": "Hello World", |  | ||||||
|                         "name": "title", |                         "name": "title", | ||||||
|                         "in": "formData" |                         "in": "formData" | ||||||
|                     }, |                     }, | ||||||
|                     { |                     { | ||||||
|                         "type": "string", |                         "type": "integer", | ||||||
|                         "example": "7725", |  | ||||||
|                         "name": "user_id", |                         "name": "user_id", | ||||||
|                         "in": "formData" |                         "in": "formData" | ||||||
|  |                     }, | ||||||
|  |                     { | ||||||
|  |                         "type": "string", | ||||||
|  |                         "name": "user_key", | ||||||
|  |                         "in": "formData" | ||||||
|                     } |                     } | ||||||
|                 ], |                 ], | ||||||
|                 "responses": { |                 "responses": { | ||||||
| @@ -3103,6 +2959,7 @@ | |||||||
|                 1151, |                 1151, | ||||||
|                 1152, |                 1152, | ||||||
|                 1153, |                 1153, | ||||||
|  |                 1152, | ||||||
|                 1161, |                 1161, | ||||||
|                 1171, |                 1171, | ||||||
|                 1201, |                 1201, | ||||||
| @@ -3148,6 +3005,7 @@ | |||||||
|                 "BINDFAIL_QUERY_PARAM", |                 "BINDFAIL_QUERY_PARAM", | ||||||
|                 "BINDFAIL_BODY_PARAM", |                 "BINDFAIL_BODY_PARAM", | ||||||
|                 "BINDFAIL_URI_PARAM", |                 "BINDFAIL_URI_PARAM", | ||||||
|  |                 "BINDFAIL_HEADER_PARAM", | ||||||
|                 "INVALID_BODY_PARAM", |                 "INVALID_BODY_PARAM", | ||||||
|                 "INVALID_ENUM_VALUE", |                 "INVALID_ENUM_VALUE", | ||||||
|                 "NO_TITLE", |                 "NO_TITLE", | ||||||
| @@ -3413,7 +3271,7 @@ | |||||||
|                 "messages": { |                 "messages": { | ||||||
|                     "type": "array", |                     "type": "array", | ||||||
|                     "items": { |                     "items": { | ||||||
|                         "$ref": "#/definitions/models.MessageJSON" |                         "$ref": "#/definitions/models.Message" | ||||||
|                     } |                     } | ||||||
|                 }, |                 }, | ||||||
|                 "next_page_token": { |                 "next_page_token": { | ||||||
| @@ -3430,7 +3288,7 @@ | |||||||
|                 "subscriptions": { |                 "subscriptions": { | ||||||
|                     "type": "array", |                     "type": "array", | ||||||
|                     "items": { |                     "items": { | ||||||
|                         "$ref": "#/definitions/models.SubscriptionJSON" |                         "$ref": "#/definitions/models.Subscription" | ||||||
|                     } |                     } | ||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
| @@ -3441,7 +3299,7 @@ | |||||||
|                 "channels": { |                 "channels": { | ||||||
|                     "type": "array", |                     "type": "array", | ||||||
|                     "items": { |                     "items": { | ||||||
|                         "$ref": "#/definitions/models.ChannelWithSubscriptionJSON" |                         "$ref": "#/definitions/models.ChannelWithSubscription" | ||||||
|                     } |                     } | ||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
| @@ -3452,7 +3310,7 @@ | |||||||
|                 "clients": { |                 "clients": { | ||||||
|                     "type": "array", |                     "type": "array", | ||||||
|                     "items": { |                     "items": { | ||||||
|                         "$ref": "#/definitions/models.ClientJSON" |                         "$ref": "#/definitions/models.Client" | ||||||
|                     } |                     } | ||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
| @@ -3463,7 +3321,7 @@ | |||||||
|                 "messages": { |                 "messages": { | ||||||
|                     "type": "array", |                     "type": "array", | ||||||
|                     "items": { |                     "items": { | ||||||
|                         "$ref": "#/definitions/models.MessageJSON" |                         "$ref": "#/definitions/models.Message" | ||||||
|                     } |                     } | ||||||
|                 }, |                 }, | ||||||
|                 "next_page_token": { |                 "next_page_token": { | ||||||
| @@ -3480,7 +3338,7 @@ | |||||||
|                 "keys": { |                 "keys": { | ||||||
|                     "type": "array", |                     "type": "array", | ||||||
|                     "items": { |                     "items": { | ||||||
|                         "$ref": "#/definitions/models.KeyTokenJSON" |                         "$ref": "#/definitions/models.KeyToken" | ||||||
|                     } |                     } | ||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
| @@ -3491,7 +3349,7 @@ | |||||||
|                 "subscriptions": { |                 "subscriptions": { | ||||||
|                     "type": "array", |                     "type": "array", | ||||||
|                     "items": { |                     "items": { | ||||||
|                         "$ref": "#/definitions/models.SubscriptionJSON" |                         "$ref": "#/definitions/models.Subscription" | ||||||
|                     } |                     } | ||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
| @@ -3545,46 +3403,26 @@ | |||||||
|         "handler.SendMessage.combined": { |         "handler.SendMessage.combined": { | ||||||
|             "type": "object", |             "type": "object", | ||||||
|             "properties": { |             "properties": { | ||||||
|                 "channel": { |  | ||||||
|                     "type": "string", |  | ||||||
|                     "example": "test" |  | ||||||
|                 }, |  | ||||||
|                 "content": { |                 "content": { | ||||||
|                     "type": "string", |                     "type": "string" | ||||||
|                     "example": "This is a message" |  | ||||||
|                 }, |  | ||||||
|                 "key": { |  | ||||||
|                     "type": "string", |  | ||||||
|                     "example": "P3TNH8mvv14fm" |  | ||||||
|                 }, |                 }, | ||||||
|                 "msg_id": { |                 "msg_id": { | ||||||
|                     "type": "string", |                     "type": "string" | ||||||
|                     "example": "db8b0e6a-a08c-4646" |  | ||||||
|                 }, |                 }, | ||||||
|                 "priority": { |                 "priority": { | ||||||
|                     "type": "integer", |                     "type": "integer" | ||||||
|                     "enum": [ |  | ||||||
|                         0, |  | ||||||
|                         1, |  | ||||||
|                         2 |  | ||||||
|                     ], |  | ||||||
|                     "example": 1 |  | ||||||
|                 }, |  | ||||||
|                 "sender_name": { |  | ||||||
|                     "type": "string", |  | ||||||
|                     "example": "example-server" |  | ||||||
|                 }, |                 }, | ||||||
|                 "timestamp": { |                 "timestamp": { | ||||||
|                     "type": "number", |                     "type": "number" | ||||||
|                     "example": 1669824037 |  | ||||||
|                 }, |                 }, | ||||||
|                 "title": { |                 "title": { | ||||||
|                     "type": "string", |                     "type": "string" | ||||||
|                     "example": "Hello World" |  | ||||||
|                 }, |                 }, | ||||||
|                 "user_id": { |                 "user_id": { | ||||||
|                     "type": "string", |                     "type": "integer" | ||||||
|                     "example": "7725" |                 }, | ||||||
|  |                 "user_key": { | ||||||
|  |                     "type": "string" | ||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
|         }, |         }, | ||||||
| @@ -3613,7 +3451,7 @@ | |||||||
|                     "type": "integer" |                     "type": "integer" | ||||||
|                 }, |                 }, | ||||||
|                 "scn_msg_id": { |                 "scn_msg_id": { | ||||||
|                     "type": "string" |                     "type": "integer" | ||||||
|                 }, |                 }, | ||||||
|                 "success": { |                 "success": { | ||||||
|                     "type": "boolean" |                     "type": "boolean" | ||||||
| @@ -3801,7 +3639,7 @@ | |||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
|         }, |         }, | ||||||
|         "models.ChannelPreviewJSON": { |         "models.ChannelPreview": { | ||||||
|             "type": "object", |             "type": "object", | ||||||
|             "properties": { |             "properties": { | ||||||
|                 "channel_id": { |                 "channel_id": { | ||||||
| @@ -3821,7 +3659,7 @@ | |||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
|         }, |         }, | ||||||
|         "models.ChannelWithSubscriptionJSON": { |         "models.ChannelWithSubscription": { | ||||||
|             "type": "object", |             "type": "object", | ||||||
|             "properties": { |             "properties": { | ||||||
|                 "channel_id": { |                 "channel_id": { | ||||||
| @@ -3847,7 +3685,7 @@ | |||||||
|                     "type": "string" |                     "type": "string" | ||||||
|                 }, |                 }, | ||||||
|                 "subscription": { |                 "subscription": { | ||||||
|                     "$ref": "#/definitions/models.SubscriptionJSON" |                     "$ref": "#/definitions/models.Subscription" | ||||||
|                 }, |                 }, | ||||||
|                 "timestamp_created": { |                 "timestamp_created": { | ||||||
|                     "type": "string" |                     "type": "string" | ||||||
| @@ -3857,7 +3695,7 @@ | |||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
|         }, |         }, | ||||||
|         "models.ClientJSON": { |         "models.Client": { | ||||||
|             "type": "object", |             "type": "object", | ||||||
|             "properties": { |             "properties": { | ||||||
|                 "agent_model": { |                 "agent_model": { | ||||||
| @@ -3929,7 +3767,7 @@ | |||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
|         }, |         }, | ||||||
|         "models.KeyTokenJSON": { |         "models.KeyToken": { | ||||||
|             "type": "object", |             "type": "object", | ||||||
|             "properties": { |             "properties": { | ||||||
|                 "all_channels": { |                 "all_channels": { | ||||||
| @@ -3954,69 +3792,11 @@ | |||||||
|                     "type": "string" |                     "type": "string" | ||||||
|                 }, |                 }, | ||||||
|                 "permissions": { |                 "permissions": { | ||||||
|                     "type": "string" |  | ||||||
|                 }, |  | ||||||
|                 "timestamp_created": { |  | ||||||
|                     "type": "string" |  | ||||||
|                 }, |  | ||||||
|                 "timestamp_lastused": { |  | ||||||
|                     "type": "string" |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|         }, |  | ||||||
|         "models.KeyTokenPreviewJSON": { |  | ||||||
|             "type": "object", |  | ||||||
|             "properties": { |  | ||||||
|                 "all_channels": { |  | ||||||
|                     "type": "boolean" |  | ||||||
|                 }, |  | ||||||
|                 "channels": { |  | ||||||
|                     "type": "array", |                     "type": "array", | ||||||
|                     "items": { |                     "items": { | ||||||
|                         "type": "string" |                         "$ref": "#/definitions/models.TokenPerm" | ||||||
|                     } |                     } | ||||||
|                 }, |                 }, | ||||||
|                 "keytoken_id": { |  | ||||||
|                     "type": "string" |  | ||||||
|                 }, |  | ||||||
|                 "name": { |  | ||||||
|                     "type": "string" |  | ||||||
|                 }, |  | ||||||
|                 "owner_user_id": { |  | ||||||
|                     "type": "string" |  | ||||||
|                 }, |  | ||||||
|                 "permissions": { |  | ||||||
|                     "type": "string" |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|         }, |  | ||||||
|         "models.KeyTokenWithTokenJSON": { |  | ||||||
|             "type": "object", |  | ||||||
|             "properties": { |  | ||||||
|                 "all_channels": { |  | ||||||
|                     "type": "boolean" |  | ||||||
|                 }, |  | ||||||
|                 "channels": { |  | ||||||
|                     "type": "array", |  | ||||||
|                     "items": { |  | ||||||
|                         "type": "string" |  | ||||||
|                     } |  | ||||||
|                 }, |  | ||||||
|                 "keytoken_id": { |  | ||||||
|                     "type": "string" |  | ||||||
|                 }, |  | ||||||
|                 "messages_sent": { |  | ||||||
|                     "type": "integer" |  | ||||||
|                 }, |  | ||||||
|                 "name": { |  | ||||||
|                     "type": "string" |  | ||||||
|                 }, |  | ||||||
|                 "owner_user_id": { |  | ||||||
|                     "type": "string" |  | ||||||
|                 }, |  | ||||||
|                 "permissions": { |  | ||||||
|                     "type": "string" |  | ||||||
|                 }, |  | ||||||
|                 "timestamp_created": { |                 "timestamp_created": { | ||||||
|                     "type": "string" |                     "type": "string" | ||||||
|                 }, |                 }, | ||||||
| @@ -4028,7 +3808,33 @@ | |||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
|         }, |         }, | ||||||
|         "models.MessageJSON": { |         "models.KeyTokenPreview": { | ||||||
|  |             "type": "object", | ||||||
|  |             "properties": { | ||||||
|  |                 "all_channels": { | ||||||
|  |                     "type": "boolean" | ||||||
|  |                 }, | ||||||
|  |                 "channels": { | ||||||
|  |                     "type": "array", | ||||||
|  |                     "items": { | ||||||
|  |                         "type": "string" | ||||||
|  |                     } | ||||||
|  |                 }, | ||||||
|  |                 "keytoken_id": { | ||||||
|  |                     "type": "string" | ||||||
|  |                 }, | ||||||
|  |                 "name": { | ||||||
|  |                     "type": "string" | ||||||
|  |                 }, | ||||||
|  |                 "owner_user_id": { | ||||||
|  |                     "type": "string" | ||||||
|  |                 }, | ||||||
|  |                 "permissions": { | ||||||
|  |                     "type": "string" | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |         }, | ||||||
|  |         "models.Message": { | ||||||
|             "type": "object", |             "type": "object", | ||||||
|             "properties": { |             "properties": { | ||||||
|                 "channel_id": { |                 "channel_id": { | ||||||
| @@ -4053,6 +3859,7 @@ | |||||||
|                     "type": "string" |                     "type": "string" | ||||||
|                 }, |                 }, | ||||||
|                 "sender_user_id": { |                 "sender_user_id": { | ||||||
|  |                     "description": "user that sent the message (this is also the owner of the channel that contains it)", | ||||||
|                     "type": "string" |                     "type": "string" | ||||||
|                 }, |                 }, | ||||||
|                 "timestamp": { |                 "timestamp": { | ||||||
| @@ -4072,7 +3879,7 @@ | |||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
|         }, |         }, | ||||||
|         "models.SubscriptionJSON": { |         "models.Subscription": { | ||||||
|             "type": "object", |             "type": "object", | ||||||
|             "properties": { |             "properties": { | ||||||
|                 "channel_id": { |                 "channel_id": { | ||||||
| @@ -4098,7 +3905,28 @@ | |||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
|         }, |         }, | ||||||
|         "models.UserJSON": { |         "models.TokenPerm": { | ||||||
|  |             "type": "string", | ||||||
|  |             "enum": [ | ||||||
|  |                 "A", | ||||||
|  |                 "CR", | ||||||
|  |                 "CS", | ||||||
|  |                 "UR" | ||||||
|  |             ], | ||||||
|  |             "x-enum-comments": { | ||||||
|  |                 "PermAdmin": "Edit userdata (+ includes all other permissions)", | ||||||
|  |                 "PermChannelRead": "Read messages", | ||||||
|  |                 "PermChannelSend": "Send messages", | ||||||
|  |                 "PermUserRead": "Read userdata" | ||||||
|  |             }, | ||||||
|  |             "x-enum-varnames": [ | ||||||
|  |                 "PermAdmin", | ||||||
|  |                 "PermChannelRead", | ||||||
|  |                 "PermChannelSend", | ||||||
|  |                 "PermUserRead" | ||||||
|  |             ] | ||||||
|  |         }, | ||||||
|  |         "models.User": { | ||||||
|             "type": "object", |             "type": "object", | ||||||
|             "properties": { |             "properties": { | ||||||
|                 "default_channel": { |                 "default_channel": { | ||||||
| @@ -4157,7 +3985,18 @@ | |||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
|         }, |         }, | ||||||
|         "models.UserJSONWithClientsAndKeys": { |         "models.UserPreview": { | ||||||
|  |             "type": "object", | ||||||
|  |             "properties": { | ||||||
|  |                 "user_id": { | ||||||
|  |                     "type": "string" | ||||||
|  |                 }, | ||||||
|  |                 "username": { | ||||||
|  |                     "type": "string" | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |         }, | ||||||
|  |         "models.UserWithClientsAndKeys": { | ||||||
|             "type": "object", |             "type": "object", | ||||||
|             "properties": { |             "properties": { | ||||||
|                 "admin_key": { |                 "admin_key": { | ||||||
| @@ -4166,7 +4005,7 @@ | |||||||
|                 "clients": { |                 "clients": { | ||||||
|                     "type": "array", |                     "type": "array", | ||||||
|                     "items": { |                     "items": { | ||||||
|                         "$ref": "#/definitions/models.ClientJSON" |                         "$ref": "#/definitions/models.Client" | ||||||
|                     } |                     } | ||||||
|                 }, |                 }, | ||||||
|                 "default_channel": { |                 "default_channel": { | ||||||
| @@ -4230,17 +4069,6 @@ | |||||||
|                     "type": "string" |                     "type": "string" | ||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
|         }, |  | ||||||
|         "models.UserPreviewJSON": { |  | ||||||
|             "type": "object", |  | ||||||
|             "properties": { |  | ||||||
|                 "user_id": { |  | ||||||
|                     "type": "string" |  | ||||||
|                 }, |  | ||||||
|                 "username": { |  | ||||||
|                     "type": "string" |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|         } |         } | ||||||
|     }, |     }, | ||||||
|     "tags": [ |     "tags": [ | ||||||
|   | |||||||
| @@ -14,6 +14,7 @@ definitions: | |||||||
|     - 1151 |     - 1151 | ||||||
|     - 1152 |     - 1152 | ||||||
|     - 1153 |     - 1153 | ||||||
|  |     - 1152 | ||||||
|     - 1161 |     - 1161 | ||||||
|     - 1171 |     - 1171 | ||||||
|     - 1201 |     - 1201 | ||||||
| @@ -59,6 +60,7 @@ definitions: | |||||||
|     - BINDFAIL_QUERY_PARAM |     - BINDFAIL_QUERY_PARAM | ||||||
|     - BINDFAIL_BODY_PARAM |     - BINDFAIL_BODY_PARAM | ||||||
|     - BINDFAIL_URI_PARAM |     - BINDFAIL_URI_PARAM | ||||||
|  |     - BINDFAIL_HEADER_PARAM | ||||||
|     - INVALID_BODY_PARAM |     - INVALID_BODY_PARAM | ||||||
|     - INVALID_ENUM_VALUE |     - INVALID_ENUM_VALUE | ||||||
|     - NO_TITLE |     - NO_TITLE | ||||||
| @@ -242,7 +244,7 @@ definitions: | |||||||
|     properties: |     properties: | ||||||
|       messages: |       messages: | ||||||
|         items: |         items: | ||||||
|           $ref: '#/definitions/models.MessageJSON' |           $ref: '#/definitions/models.Message' | ||||||
|         type: array |         type: array | ||||||
|       next_page_token: |       next_page_token: | ||||||
|         type: string |         type: string | ||||||
| @@ -253,28 +255,28 @@ definitions: | |||||||
|     properties: |     properties: | ||||||
|       subscriptions: |       subscriptions: | ||||||
|         items: |         items: | ||||||
|           $ref: '#/definitions/models.SubscriptionJSON' |           $ref: '#/definitions/models.Subscription' | ||||||
|         type: array |         type: array | ||||||
|     type: object |     type: object | ||||||
|   handler.ListChannels.response: |   handler.ListChannels.response: | ||||||
|     properties: |     properties: | ||||||
|       channels: |       channels: | ||||||
|         items: |         items: | ||||||
|           $ref: '#/definitions/models.ChannelWithSubscriptionJSON' |           $ref: '#/definitions/models.ChannelWithSubscription' | ||||||
|         type: array |         type: array | ||||||
|     type: object |     type: object | ||||||
|   handler.ListClients.response: |   handler.ListClients.response: | ||||||
|     properties: |     properties: | ||||||
|       clients: |       clients: | ||||||
|         items: |         items: | ||||||
|           $ref: '#/definitions/models.ClientJSON' |           $ref: '#/definitions/models.Client' | ||||||
|         type: array |         type: array | ||||||
|     type: object |     type: object | ||||||
|   handler.ListMessages.response: |   handler.ListMessages.response: | ||||||
|     properties: |     properties: | ||||||
|       messages: |       messages: | ||||||
|         items: |         items: | ||||||
|           $ref: '#/definitions/models.MessageJSON' |           $ref: '#/definitions/models.Message' | ||||||
|         type: array |         type: array | ||||||
|       next_page_token: |       next_page_token: | ||||||
|         type: string |         type: string | ||||||
| @@ -285,14 +287,14 @@ definitions: | |||||||
|     properties: |     properties: | ||||||
|       keys: |       keys: | ||||||
|         items: |         items: | ||||||
|           $ref: '#/definitions/models.KeyTokenJSON' |           $ref: '#/definitions/models.KeyToken' | ||||||
|         type: array |         type: array | ||||||
|     type: object |     type: object | ||||||
|   handler.ListUserSubscriptions.response: |   handler.ListUserSubscriptions.response: | ||||||
|     properties: |     properties: | ||||||
|       subscriptions: |       subscriptions: | ||||||
|         items: |         items: | ||||||
|           $ref: '#/definitions/models.SubscriptionJSON' |           $ref: '#/definitions/models.Subscription' | ||||||
|         type: array |         type: array | ||||||
|     type: object |     type: object | ||||||
|   handler.Register.response: |   handler.Register.response: | ||||||
| @@ -327,36 +329,19 @@ definitions: | |||||||
|     type: object |     type: object | ||||||
|   handler.SendMessage.combined: |   handler.SendMessage.combined: | ||||||
|     properties: |     properties: | ||||||
|       channel: |  | ||||||
|         example: test |  | ||||||
|         type: string |  | ||||||
|       content: |       content: | ||||||
|         example: This is a message |  | ||||||
|         type: string |  | ||||||
|       key: |  | ||||||
|         example: P3TNH8mvv14fm |  | ||||||
|         type: string |         type: string | ||||||
|       msg_id: |       msg_id: | ||||||
|         example: db8b0e6a-a08c-4646 |  | ||||||
|         type: string |         type: string | ||||||
|       priority: |       priority: | ||||||
|         enum: |  | ||||||
|         - 0 |  | ||||||
|         - 1 |  | ||||||
|         - 2 |  | ||||||
|         example: 1 |  | ||||||
|         type: integer |         type: integer | ||||||
|       sender_name: |  | ||||||
|         example: example-server |  | ||||||
|         type: string |  | ||||||
|       timestamp: |       timestamp: | ||||||
|         example: 1669824037 |  | ||||||
|         type: number |         type: number | ||||||
|       title: |       title: | ||||||
|         example: Hello World |  | ||||||
|         type: string |         type: string | ||||||
|       user_id: |       user_id: | ||||||
|         example: "7725" |         type: integer | ||||||
|  |       user_key: | ||||||
|         type: string |         type: string | ||||||
|     type: object |     type: object | ||||||
|   handler.SendMessage.response: |   handler.SendMessage.response: | ||||||
| @@ -376,7 +361,7 @@ definitions: | |||||||
|       quota_max: |       quota_max: | ||||||
|         type: integer |         type: integer | ||||||
|       scn_msg_id: |       scn_msg_id: | ||||||
|         type: string |         type: integer | ||||||
|       success: |       success: | ||||||
|         type: boolean |         type: boolean | ||||||
|       suppress_send: |       suppress_send: | ||||||
| @@ -497,7 +482,7 @@ definitions: | |||||||
|       uri: |       uri: | ||||||
|         type: string |         type: string | ||||||
|     type: object |     type: object | ||||||
|   models.ChannelPreviewJSON: |   models.ChannelPreview: | ||||||
|     properties: |     properties: | ||||||
|       channel_id: |       channel_id: | ||||||
|         type: string |         type: string | ||||||
| @@ -510,7 +495,7 @@ definitions: | |||||||
|       owner_user_id: |       owner_user_id: | ||||||
|         type: string |         type: string | ||||||
|     type: object |     type: object | ||||||
|   models.ChannelWithSubscriptionJSON: |   models.ChannelWithSubscription: | ||||||
|     properties: |     properties: | ||||||
|       channel_id: |       channel_id: | ||||||
|         type: string |         type: string | ||||||
| @@ -528,13 +513,13 @@ definitions: | |||||||
|         description: can be nil, depending on endpoint |         description: can be nil, depending on endpoint | ||||||
|         type: string |         type: string | ||||||
|       subscription: |       subscription: | ||||||
|         $ref: '#/definitions/models.SubscriptionJSON' |         $ref: '#/definitions/models.Subscription' | ||||||
|       timestamp_created: |       timestamp_created: | ||||||
|         type: string |         type: string | ||||||
|       timestamp_lastsent: |       timestamp_lastsent: | ||||||
|         type: string |         type: string | ||||||
|     type: object |     type: object | ||||||
|   models.ClientJSON: |   models.Client: | ||||||
|     properties: |     properties: | ||||||
|       agent_model: |       agent_model: | ||||||
|         type: string |         type: string | ||||||
| @@ -584,7 +569,7 @@ definitions: | |||||||
|       usr_msg_id: |       usr_msg_id: | ||||||
|         type: string |         type: string | ||||||
|     type: object |     type: object | ||||||
|   models.KeyTokenJSON: |   models.KeyToken: | ||||||
|     properties: |     properties: | ||||||
|       all_channels: |       all_channels: | ||||||
|         type: boolean |         type: boolean | ||||||
| @@ -601,47 +586,9 @@ definitions: | |||||||
|       owner_user_id: |       owner_user_id: | ||||||
|         type: string |         type: string | ||||||
|       permissions: |       permissions: | ||||||
|         type: string |  | ||||||
|       timestamp_created: |  | ||||||
|         type: string |  | ||||||
|       timestamp_lastused: |  | ||||||
|         type: string |  | ||||||
|     type: object |  | ||||||
|   models.KeyTokenPreviewJSON: |  | ||||||
|     properties: |  | ||||||
|       all_channels: |  | ||||||
|         type: boolean |  | ||||||
|       channels: |  | ||||||
|         items: |         items: | ||||||
|           type: string |           $ref: '#/definitions/models.TokenPerm' | ||||||
|         type: array |         type: array | ||||||
|       keytoken_id: |  | ||||||
|         type: string |  | ||||||
|       name: |  | ||||||
|         type: string |  | ||||||
|       owner_user_id: |  | ||||||
|         type: string |  | ||||||
|       permissions: |  | ||||||
|         type: string |  | ||||||
|     type: object |  | ||||||
|   models.KeyTokenWithTokenJSON: |  | ||||||
|     properties: |  | ||||||
|       all_channels: |  | ||||||
|         type: boolean |  | ||||||
|       channels: |  | ||||||
|         items: |  | ||||||
|           type: string |  | ||||||
|         type: array |  | ||||||
|       keytoken_id: |  | ||||||
|         type: string |  | ||||||
|       messages_sent: |  | ||||||
|         type: integer |  | ||||||
|       name: |  | ||||||
|         type: string |  | ||||||
|       owner_user_id: |  | ||||||
|         type: string |  | ||||||
|       permissions: |  | ||||||
|         type: string |  | ||||||
|       timestamp_created: |       timestamp_created: | ||||||
|         type: string |         type: string | ||||||
|       timestamp_lastused: |       timestamp_lastused: | ||||||
| @@ -649,7 +596,24 @@ definitions: | |||||||
|       token: |       token: | ||||||
|         type: string |         type: string | ||||||
|     type: object |     type: object | ||||||
|   models.MessageJSON: |   models.KeyTokenPreview: | ||||||
|  |     properties: | ||||||
|  |       all_channels: | ||||||
|  |         type: boolean | ||||||
|  |       channels: | ||||||
|  |         items: | ||||||
|  |           type: string | ||||||
|  |         type: array | ||||||
|  |       keytoken_id: | ||||||
|  |         type: string | ||||||
|  |       name: | ||||||
|  |         type: string | ||||||
|  |       owner_user_id: | ||||||
|  |         type: string | ||||||
|  |       permissions: | ||||||
|  |         type: string | ||||||
|  |     type: object | ||||||
|  |   models.Message: | ||||||
|     properties: |     properties: | ||||||
|       channel_id: |       channel_id: | ||||||
|         type: string |         type: string | ||||||
| @@ -666,6 +630,8 @@ definitions: | |||||||
|       sender_name: |       sender_name: | ||||||
|         type: string |         type: string | ||||||
|       sender_user_id: |       sender_user_id: | ||||||
|  |         description: user that sent the message (this is also the owner of the channel | ||||||
|  |           that contains it) | ||||||
|         type: string |         type: string | ||||||
|       timestamp: |       timestamp: | ||||||
|         type: string |         type: string | ||||||
| @@ -678,7 +644,7 @@ definitions: | |||||||
|       usr_message_id: |       usr_message_id: | ||||||
|         type: string |         type: string | ||||||
|     type: object |     type: object | ||||||
|   models.SubscriptionJSON: |   models.Subscription: | ||||||
|     properties: |     properties: | ||||||
|       channel_id: |       channel_id: | ||||||
|         type: string |         type: string | ||||||
| @@ -695,7 +661,24 @@ definitions: | |||||||
|       timestamp_created: |       timestamp_created: | ||||||
|         type: string |         type: string | ||||||
|     type: object |     type: object | ||||||
|   models.UserJSON: |   models.TokenPerm: | ||||||
|  |     enum: | ||||||
|  |     - A | ||||||
|  |     - CR | ||||||
|  |     - CS | ||||||
|  |     - UR | ||||||
|  |     type: string | ||||||
|  |     x-enum-comments: | ||||||
|  |       PermAdmin: Edit userdata (+ includes all other permissions) | ||||||
|  |       PermChannelRead: Read messages | ||||||
|  |       PermChannelSend: Send messages | ||||||
|  |       PermUserRead: Read userdata | ||||||
|  |     x-enum-varnames: | ||||||
|  |     - PermAdmin | ||||||
|  |     - PermChannelRead | ||||||
|  |     - PermChannelSend | ||||||
|  |     - PermUserRead | ||||||
|  |   models.User: | ||||||
|     properties: |     properties: | ||||||
|       default_channel: |       default_channel: | ||||||
|         type: string |         type: string | ||||||
| @@ -734,13 +717,20 @@ definitions: | |||||||
|       username: |       username: | ||||||
|         type: string |         type: string | ||||||
|     type: object |     type: object | ||||||
|   models.UserJSONWithClientsAndKeys: |   models.UserPreview: | ||||||
|  |     properties: | ||||||
|  |       user_id: | ||||||
|  |         type: string | ||||||
|  |       username: | ||||||
|  |         type: string | ||||||
|  |     type: object | ||||||
|  |   models.UserWithClientsAndKeys: | ||||||
|     properties: |     properties: | ||||||
|       admin_key: |       admin_key: | ||||||
|         type: string |         type: string | ||||||
|       clients: |       clients: | ||||||
|         items: |         items: | ||||||
|           $ref: '#/definitions/models.ClientJSON' |           $ref: '#/definitions/models.Client' | ||||||
|         type: array |         type: array | ||||||
|       default_channel: |       default_channel: | ||||||
|         type: string |         type: string | ||||||
| @@ -783,13 +773,6 @@ definitions: | |||||||
|       username: |       username: | ||||||
|         type: string |         type: string | ||||||
|     type: object |     type: object | ||||||
|   models.UserPreviewJSON: |  | ||||||
|     properties: |  | ||||||
|       user_id: |  | ||||||
|         type: string |  | ||||||
|       username: |  | ||||||
|         type: string |  | ||||||
|     type: object |  | ||||||
| host: simplecloudnotifier.de | host: simplecloudnotifier.de | ||||||
| info: | info: | ||||||
|   contact: {} |   contact: {} | ||||||
| @@ -802,90 +785,52 @@ paths: | |||||||
|       description: All parameter can be set via query-parameter or the json body. |       description: All parameter can be set via query-parameter or the json body. | ||||||
|         Only UserID, UserKey and Title are required |         Only UserID, UserKey and Title are required | ||||||
|       parameters: |       parameters: | ||||||
|       - example: test |       - in: query | ||||||
|         in: query |  | ||||||
|         name: channel |  | ||||||
|         type: string |  | ||||||
|       - example: This is a message |  | ||||||
|         in: query |  | ||||||
|         name: content |         name: content | ||||||
|         type: string |         type: string | ||||||
|       - example: P3TNH8mvv14fm |       - in: query | ||||||
|         in: query |  | ||||||
|         name: key |  | ||||||
|         type: string |  | ||||||
|       - example: db8b0e6a-a08c-4646 |  | ||||||
|         in: query |  | ||||||
|         name: msg_id |         name: msg_id | ||||||
|         type: string |         type: string | ||||||
|       - enum: |       - in: query | ||||||
|         - 0 |  | ||||||
|         - 1 |  | ||||||
|         - 2 |  | ||||||
|         example: 1 |  | ||||||
|         in: query |  | ||||||
|         name: priority |         name: priority | ||||||
|         type: integer |         type: integer | ||||||
|       - example: example-server |       - in: query | ||||||
|         in: query |  | ||||||
|         name: sender_name |  | ||||||
|         type: string |  | ||||||
|       - example: 1669824037 |  | ||||||
|         in: query |  | ||||||
|         name: timestamp |         name: timestamp | ||||||
|         type: number |         type: number | ||||||
|       - example: Hello World |       - in: query | ||||||
|         in: query |  | ||||||
|         name: title |         name: title | ||||||
|         type: string |         type: string | ||||||
|       - example: "7725" |       - in: query | ||||||
|         in: query |  | ||||||
|         name: user_id |         name: user_id | ||||||
|  |         type: integer | ||||||
|  |       - in: query | ||||||
|  |         name: user_key | ||||||
|         type: string |         type: string | ||||||
|       - description: ' ' |       - description: ' ' | ||||||
|         in: body |         in: body | ||||||
|         name: post_body |         name: post_body | ||||||
|         schema: |         schema: | ||||||
|           $ref: '#/definitions/handler.SendMessage.combined' |           $ref: '#/definitions/handler.SendMessage.combined' | ||||||
|       - example: test |       - in: formData | ||||||
|         in: formData |  | ||||||
|         name: channel |  | ||||||
|         type: string |  | ||||||
|       - example: This is a message |  | ||||||
|         in: formData |  | ||||||
|         name: content |         name: content | ||||||
|         type: string |         type: string | ||||||
|       - example: P3TNH8mvv14fm |       - in: formData | ||||||
|         in: formData |  | ||||||
|         name: key |  | ||||||
|         type: string |  | ||||||
|       - example: db8b0e6a-a08c-4646 |  | ||||||
|         in: formData |  | ||||||
|         name: msg_id |         name: msg_id | ||||||
|         type: string |         type: string | ||||||
|       - enum: |       - in: formData | ||||||
|         - 0 |  | ||||||
|         - 1 |  | ||||||
|         - 2 |  | ||||||
|         example: 1 |  | ||||||
|         in: formData |  | ||||||
|         name: priority |         name: priority | ||||||
|         type: integer |         type: integer | ||||||
|       - example: example-server |       - in: formData | ||||||
|         in: formData |  | ||||||
|         name: sender_name |  | ||||||
|         type: string |  | ||||||
|       - example: 1669824037 |  | ||||||
|         in: formData |  | ||||||
|         name: timestamp |         name: timestamp | ||||||
|         type: number |         type: number | ||||||
|       - example: Hello World |       - in: formData | ||||||
|         in: formData |  | ||||||
|         name: title |         name: title | ||||||
|         type: string |         type: string | ||||||
|       - example: "7725" |       - in: formData | ||||||
|         in: formData |  | ||||||
|         name: user_id |         name: user_id | ||||||
|  |         type: integer | ||||||
|  |       - in: formData | ||||||
|  |         name: user_key | ||||||
|         type: string |         type: string | ||||||
|       responses: |       responses: | ||||||
|         "200": |         "200": | ||||||
| @@ -1458,7 +1403,7 @@ paths: | |||||||
|         "200": |         "200": | ||||||
|           description: OK |           description: OK | ||||||
|           schema: |           schema: | ||||||
|             $ref: '#/definitions/models.MessageJSON' |             $ref: '#/definitions/models.Message' | ||||||
|         "400": |         "400": | ||||||
|           description: supplied values/parameters cannot be parsed / are invalid |           description: supplied values/parameters cannot be parsed / are invalid | ||||||
|           schema: |           schema: | ||||||
| @@ -1494,7 +1439,7 @@ paths: | |||||||
|         "200": |         "200": | ||||||
|           description: OK |           description: OK | ||||||
|           schema: |           schema: | ||||||
|             $ref: '#/definitions/models.MessageJSON' |             $ref: '#/definitions/models.Message' | ||||||
|         "400": |         "400": | ||||||
|           description: supplied values/parameters cannot be parsed / are invalid |           description: supplied values/parameters cannot be parsed / are invalid | ||||||
|           schema: |           schema: | ||||||
| @@ -1527,7 +1472,7 @@ paths: | |||||||
|         "200": |         "200": | ||||||
|           description: OK |           description: OK | ||||||
|           schema: |           schema: | ||||||
|             $ref: '#/definitions/models.ChannelPreviewJSON' |             $ref: '#/definitions/models.ChannelPreview' | ||||||
|         "400": |         "400": | ||||||
|           description: supplied values/parameters cannot be parsed / are invalid |           description: supplied values/parameters cannot be parsed / are invalid | ||||||
|           schema: |           schema: | ||||||
| @@ -1561,7 +1506,7 @@ paths: | |||||||
|         "200": |         "200": | ||||||
|           description: OK |           description: OK | ||||||
|           schema: |           schema: | ||||||
|             $ref: '#/definitions/models.KeyTokenPreviewJSON' |             $ref: '#/definitions/models.KeyTokenPreview' | ||||||
|         "400": |         "400": | ||||||
|           description: supplied values/parameters cannot be parsed / are invalid |           description: supplied values/parameters cannot be parsed / are invalid | ||||||
|           schema: |           schema: | ||||||
| @@ -1595,7 +1540,7 @@ paths: | |||||||
|         "200": |         "200": | ||||||
|           description: OK |           description: OK | ||||||
|           schema: |           schema: | ||||||
|             $ref: '#/definitions/models.UserPreviewJSON' |             $ref: '#/definitions/models.UserPreview' | ||||||
|         "400": |         "400": | ||||||
|           description: supplied values/parameters cannot be parsed / are invalid |           description: supplied values/parameters cannot be parsed / are invalid | ||||||
|           schema: |           schema: | ||||||
| @@ -1629,7 +1574,7 @@ paths: | |||||||
|         "200": |         "200": | ||||||
|           description: OK |           description: OK | ||||||
|           schema: |           schema: | ||||||
|             $ref: '#/definitions/models.UserJSONWithClientsAndKeys' |             $ref: '#/definitions/models.UserWithClientsAndKeys' | ||||||
|         "400": |         "400": | ||||||
|           description: supplied values/parameters cannot be parsed / are invalid |           description: supplied values/parameters cannot be parsed / are invalid | ||||||
|           schema: |           schema: | ||||||
| @@ -1654,7 +1599,7 @@ paths: | |||||||
|         "200": |         "200": | ||||||
|           description: OK |           description: OK | ||||||
|           schema: |           schema: | ||||||
|             $ref: '#/definitions/models.UserJSON' |             $ref: '#/definitions/models.User' | ||||||
|         "400": |         "400": | ||||||
|           description: supplied values/parameters cannot be parsed / are invalid |           description: supplied values/parameters cannot be parsed / are invalid | ||||||
|           schema: |           schema: | ||||||
| @@ -1697,7 +1642,7 @@ paths: | |||||||
|         "200": |         "200": | ||||||
|           description: OK |           description: OK | ||||||
|           schema: |           schema: | ||||||
|             $ref: '#/definitions/models.UserJSON' |             $ref: '#/definitions/models.User' | ||||||
|         "400": |         "400": | ||||||
|           description: supplied values/parameters cannot be parsed / are invalid |           description: supplied values/parameters cannot be parsed / are invalid | ||||||
|           schema: |           schema: | ||||||
| @@ -1780,7 +1725,7 @@ paths: | |||||||
|         "200": |         "200": | ||||||
|           description: OK |           description: OK | ||||||
|           schema: |           schema: | ||||||
|             $ref: '#/definitions/models.ChannelWithSubscriptionJSON' |             $ref: '#/definitions/models.ChannelWithSubscription' | ||||||
|         "400": |         "400": | ||||||
|           description: supplied values/parameters cannot be parsed / are invalid |           description: supplied values/parameters cannot be parsed / are invalid | ||||||
|           schema: |           schema: | ||||||
| @@ -1818,7 +1763,7 @@ paths: | |||||||
|         "200": |         "200": | ||||||
|           description: OK |           description: OK | ||||||
|           schema: |           schema: | ||||||
|             $ref: '#/definitions/models.ChannelWithSubscriptionJSON' |             $ref: '#/definitions/models.ChannelWithSubscription' | ||||||
|         "400": |         "400": | ||||||
|           description: supplied values/parameters cannot be parsed / are invalid |           description: supplied values/parameters cannot be parsed / are invalid | ||||||
|           schema: |           schema: | ||||||
| @@ -1871,7 +1816,7 @@ paths: | |||||||
|         "200": |         "200": | ||||||
|           description: OK |           description: OK | ||||||
|           schema: |           schema: | ||||||
|             $ref: '#/definitions/models.ChannelWithSubscriptionJSON' |             $ref: '#/definitions/models.ChannelWithSubscription' | ||||||
|         "400": |         "400": | ||||||
|           description: supplied values/parameters cannot be parsed / are invalid |           description: supplied values/parameters cannot be parsed / are invalid | ||||||
|           schema: |           schema: | ||||||
| @@ -2030,7 +1975,7 @@ paths: | |||||||
|         "200": |         "200": | ||||||
|           description: OK |           description: OK | ||||||
|           schema: |           schema: | ||||||
|             $ref: '#/definitions/models.ClientJSON' |             $ref: '#/definitions/models.Client' | ||||||
|         "400": |         "400": | ||||||
|           description: supplied values/parameters cannot be parsed / are invalid |           description: supplied values/parameters cannot be parsed / are invalid | ||||||
|           schema: |           schema: | ||||||
| @@ -2064,7 +2009,7 @@ paths: | |||||||
|         "200": |         "200": | ||||||
|           description: OK |           description: OK | ||||||
|           schema: |           schema: | ||||||
|             $ref: '#/definitions/models.ClientJSON' |             $ref: '#/definitions/models.Client' | ||||||
|         "400": |         "400": | ||||||
|           description: supplied values/parameters cannot be parsed / are invalid |           description: supplied values/parameters cannot be parsed / are invalid | ||||||
|           schema: |           schema: | ||||||
| @@ -2101,7 +2046,7 @@ paths: | |||||||
|         "200": |         "200": | ||||||
|           description: OK |           description: OK | ||||||
|           schema: |           schema: | ||||||
|             $ref: '#/definitions/models.ClientJSON' |             $ref: '#/definitions/models.Client' | ||||||
|         "400": |         "400": | ||||||
|           description: supplied values/parameters cannot be parsed / are invalid |           description: supplied values/parameters cannot be parsed / are invalid | ||||||
|           schema: |           schema: | ||||||
| @@ -2149,7 +2094,7 @@ paths: | |||||||
|         "200": |         "200": | ||||||
|           description: OK |           description: OK | ||||||
|           schema: |           schema: | ||||||
|             $ref: '#/definitions/models.ClientJSON' |             $ref: '#/definitions/models.Client' | ||||||
|         "400": |         "400": | ||||||
|           description: supplied values/parameters cannot be parsed / are invalid |           description: supplied values/parameters cannot be parsed / are invalid | ||||||
|           schema: |           schema: | ||||||
| @@ -2221,7 +2166,7 @@ paths: | |||||||
|         "200": |         "200": | ||||||
|           description: OK |           description: OK | ||||||
|           schema: |           schema: | ||||||
|             $ref: '#/definitions/models.KeyTokenJSON' |             $ref: '#/definitions/models.KeyToken' | ||||||
|         "400": |         "400": | ||||||
|           description: supplied values/parameters cannot be parsed / are invalid |           description: supplied values/parameters cannot be parsed / are invalid | ||||||
|           schema: |           schema: | ||||||
| @@ -2260,7 +2205,7 @@ paths: | |||||||
|         "200": |         "200": | ||||||
|           description: OK |           description: OK | ||||||
|           schema: |           schema: | ||||||
|             $ref: '#/definitions/models.KeyTokenJSON' |             $ref: '#/definitions/models.KeyToken' | ||||||
|         "400": |         "400": | ||||||
|           description: supplied values/parameters cannot be parsed / are invalid |           description: supplied values/parameters cannot be parsed / are invalid | ||||||
|           schema: |           schema: | ||||||
| @@ -2299,7 +2244,7 @@ paths: | |||||||
|         "200": |         "200": | ||||||
|           description: OK |           description: OK | ||||||
|           schema: |           schema: | ||||||
|             $ref: '#/definitions/models.KeyTokenJSON' |             $ref: '#/definitions/models.KeyToken' | ||||||
|         "400": |         "400": | ||||||
|           description: supplied values/parameters cannot be parsed / are invalid |           description: supplied values/parameters cannot be parsed / are invalid | ||||||
|           schema: |           schema: | ||||||
| @@ -2341,7 +2286,7 @@ paths: | |||||||
|         "200": |         "200": | ||||||
|           description: OK |           description: OK | ||||||
|           schema: |           schema: | ||||||
|             $ref: '#/definitions/models.KeyTokenJSON' |             $ref: '#/definitions/models.KeyToken' | ||||||
|         "400": |         "400": | ||||||
|           description: supplied values/parameters cannot be parsed / are invalid |           description: supplied values/parameters cannot be parsed / are invalid | ||||||
|           schema: |           schema: | ||||||
| @@ -2381,7 +2326,7 @@ paths: | |||||||
|         "200": |         "200": | ||||||
|           description: OK |           description: OK | ||||||
|           schema: |           schema: | ||||||
|             $ref: '#/definitions/models.KeyTokenWithTokenJSON' |             $ref: '#/definitions/models.KeyToken' | ||||||
|         "400": |         "400": | ||||||
|           description: supplied values/parameters cannot be parsed / are invalid |           description: supplied values/parameters cannot be parsed / are invalid | ||||||
|           schema: |           schema: | ||||||
| @@ -2483,7 +2428,7 @@ paths: | |||||||
|         "200": |         "200": | ||||||
|           description: OK |           description: OK | ||||||
|           schema: |           schema: | ||||||
|             $ref: '#/definitions/models.SubscriptionJSON' |             $ref: '#/definitions/models.Subscription' | ||||||
|         "400": |         "400": | ||||||
|           description: supplied values/parameters cannot be parsed / are invalid |           description: supplied values/parameters cannot be parsed / are invalid | ||||||
|           schema: |           schema: | ||||||
| @@ -2517,7 +2462,7 @@ paths: | |||||||
|         "200": |         "200": | ||||||
|           description: OK |           description: OK | ||||||
|           schema: |           schema: | ||||||
|             $ref: '#/definitions/models.SubscriptionJSON' |             $ref: '#/definitions/models.Subscription' | ||||||
|         "400": |         "400": | ||||||
|           description: supplied values/parameters cannot be parsed / are invalid |           description: supplied values/parameters cannot be parsed / are invalid | ||||||
|           schema: |           schema: | ||||||
| @@ -2554,7 +2499,7 @@ paths: | |||||||
|         "200": |         "200": | ||||||
|           description: OK |           description: OK | ||||||
|           schema: |           schema: | ||||||
|             $ref: '#/definitions/models.SubscriptionJSON' |             $ref: '#/definitions/models.Subscription' | ||||||
|         "400": |         "400": | ||||||
|           description: supplied values/parameters cannot be parsed / are invalid |           description: supplied values/parameters cannot be parsed / are invalid | ||||||
|           schema: |           schema: | ||||||
| @@ -2596,7 +2541,7 @@ paths: | |||||||
|         "200": |         "200": | ||||||
|           description: OK |           description: OK | ||||||
|           schema: |           schema: | ||||||
|             $ref: '#/definitions/models.SubscriptionJSON' |             $ref: '#/definitions/models.Subscription' | ||||||
|         "400": |         "400": | ||||||
|           description: supplied values/parameters cannot be parsed / are invalid |           description: supplied values/parameters cannot be parsed / are invalid | ||||||
|           schema: |           schema: | ||||||
| @@ -2685,90 +2630,52 @@ paths: | |||||||
|       description: All parameter can be set via query-parameter or the json body. |       description: All parameter can be set via query-parameter or the json body. | ||||||
|         Only UserID, UserKey and Title are required |         Only UserID, UserKey and Title are required | ||||||
|       parameters: |       parameters: | ||||||
|       - example: test |       - in: query | ||||||
|         in: query |  | ||||||
|         name: channel |  | ||||||
|         type: string |  | ||||||
|       - example: This is a message |  | ||||||
|         in: query |  | ||||||
|         name: content |         name: content | ||||||
|         type: string |         type: string | ||||||
|       - example: P3TNH8mvv14fm |       - in: query | ||||||
|         in: query |  | ||||||
|         name: key |  | ||||||
|         type: string |  | ||||||
|       - example: db8b0e6a-a08c-4646 |  | ||||||
|         in: query |  | ||||||
|         name: msg_id |         name: msg_id | ||||||
|         type: string |         type: string | ||||||
|       - enum: |       - in: query | ||||||
|         - 0 |  | ||||||
|         - 1 |  | ||||||
|         - 2 |  | ||||||
|         example: 1 |  | ||||||
|         in: query |  | ||||||
|         name: priority |         name: priority | ||||||
|         type: integer |         type: integer | ||||||
|       - example: example-server |       - in: query | ||||||
|         in: query |  | ||||||
|         name: sender_name |  | ||||||
|         type: string |  | ||||||
|       - example: 1669824037 |  | ||||||
|         in: query |  | ||||||
|         name: timestamp |         name: timestamp | ||||||
|         type: number |         type: number | ||||||
|       - example: Hello World |       - in: query | ||||||
|         in: query |  | ||||||
|         name: title |         name: title | ||||||
|         type: string |         type: string | ||||||
|       - example: "7725" |       - in: query | ||||||
|         in: query |  | ||||||
|         name: user_id |         name: user_id | ||||||
|  |         type: integer | ||||||
|  |       - in: query | ||||||
|  |         name: user_key | ||||||
|         type: string |         type: string | ||||||
|       - description: ' ' |       - description: ' ' | ||||||
|         in: body |         in: body | ||||||
|         name: post_body |         name: post_body | ||||||
|         schema: |         schema: | ||||||
|           $ref: '#/definitions/handler.SendMessage.combined' |           $ref: '#/definitions/handler.SendMessage.combined' | ||||||
|       - example: test |       - in: formData | ||||||
|         in: formData |  | ||||||
|         name: channel |  | ||||||
|         type: string |  | ||||||
|       - example: This is a message |  | ||||||
|         in: formData |  | ||||||
|         name: content |         name: content | ||||||
|         type: string |         type: string | ||||||
|       - example: P3TNH8mvv14fm |       - in: formData | ||||||
|         in: formData |  | ||||||
|         name: key |  | ||||||
|         type: string |  | ||||||
|       - example: db8b0e6a-a08c-4646 |  | ||||||
|         in: formData |  | ||||||
|         name: msg_id |         name: msg_id | ||||||
|         type: string |         type: string | ||||||
|       - enum: |       - in: formData | ||||||
|         - 0 |  | ||||||
|         - 1 |  | ||||||
|         - 2 |  | ||||||
|         example: 1 |  | ||||||
|         in: formData |  | ||||||
|         name: priority |         name: priority | ||||||
|         type: integer |         type: integer | ||||||
|       - example: example-server |       - in: formData | ||||||
|         in: formData |  | ||||||
|         name: sender_name |  | ||||||
|         type: string |  | ||||||
|       - example: 1669824037 |  | ||||||
|         in: formData |  | ||||||
|         name: timestamp |         name: timestamp | ||||||
|         type: number |         type: number | ||||||
|       - example: Hello World |       - in: formData | ||||||
|         in: formData |  | ||||||
|         name: title |         name: title | ||||||
|         type: string |         type: string | ||||||
|       - example: "7725" |       - in: formData | ||||||
|         in: formData |  | ||||||
|         name: user_id |         name: user_id | ||||||
|  |         type: integer | ||||||
|  |       - in: formData | ||||||
|  |         name: user_key | ||||||
|         type: string |         type: string | ||||||
|       responses: |       responses: | ||||||
|         "200": |         "200": | ||||||
| @@ -2801,85 +2708,47 @@ paths: | |||||||
|       description: All parameter can be set via query-parameter or form-data body. |       description: All parameter can be set via query-parameter or form-data body. | ||||||
|         Only UserID, UserKey and Title are required |         Only UserID, UserKey and Title are required | ||||||
|       parameters: |       parameters: | ||||||
|       - example: test |       - in: query | ||||||
|         in: query |  | ||||||
|         name: channel |  | ||||||
|         type: string |  | ||||||
|       - example: This is a message |  | ||||||
|         in: query |  | ||||||
|         name: content |         name: content | ||||||
|         type: string |         type: string | ||||||
|       - example: P3TNH8mvv14fm |       - in: query | ||||||
|         in: query |  | ||||||
|         name: key |  | ||||||
|         type: string |  | ||||||
|       - example: db8b0e6a-a08c-4646 |  | ||||||
|         in: query |  | ||||||
|         name: msg_id |         name: msg_id | ||||||
|         type: string |         type: string | ||||||
|       - enum: |       - in: query | ||||||
|         - 0 |  | ||||||
|         - 1 |  | ||||||
|         - 2 |  | ||||||
|         example: 1 |  | ||||||
|         in: query |  | ||||||
|         name: priority |         name: priority | ||||||
|         type: integer |         type: integer | ||||||
|       - example: example-server |       - in: query | ||||||
|         in: query |  | ||||||
|         name: sender_name |  | ||||||
|         type: string |  | ||||||
|       - example: 1669824037 |  | ||||||
|         in: query |  | ||||||
|         name: timestamp |         name: timestamp | ||||||
|         type: number |         type: number | ||||||
|       - example: Hello World |       - in: query | ||||||
|         in: query |  | ||||||
|         name: title |         name: title | ||||||
|         type: string |         type: string | ||||||
|       - example: "7725" |       - in: query | ||||||
|         in: query |  | ||||||
|         name: user_id |         name: user_id | ||||||
|  |         type: integer | ||||||
|  |       - in: query | ||||||
|  |         name: user_key | ||||||
|         type: string |         type: string | ||||||
|       - example: test |       - in: formData | ||||||
|         in: formData |  | ||||||
|         name: channel |  | ||||||
|         type: string |  | ||||||
|       - example: This is a message |  | ||||||
|         in: formData |  | ||||||
|         name: content |         name: content | ||||||
|         type: string |         type: string | ||||||
|       - example: P3TNH8mvv14fm |       - in: formData | ||||||
|         in: formData |  | ||||||
|         name: key |  | ||||||
|         type: string |  | ||||||
|       - example: db8b0e6a-a08c-4646 |  | ||||||
|         in: formData |  | ||||||
|         name: msg_id |         name: msg_id | ||||||
|         type: string |         type: string | ||||||
|       - enum: |       - in: formData | ||||||
|         - 0 |  | ||||||
|         - 1 |  | ||||||
|         - 2 |  | ||||||
|         example: 1 |  | ||||||
|         in: formData |  | ||||||
|         name: priority |         name: priority | ||||||
|         type: integer |         type: integer | ||||||
|       - example: example-server |       - in: formData | ||||||
|         in: formData |  | ||||||
|         name: sender_name |  | ||||||
|         type: string |  | ||||||
|       - example: 1669824037 |  | ||||||
|         in: formData |  | ||||||
|         name: timestamp |         name: timestamp | ||||||
|         type: number |         type: number | ||||||
|       - example: Hello World |       - in: formData | ||||||
|         in: formData |  | ||||||
|         name: title |         name: title | ||||||
|         type: string |         type: string | ||||||
|       - example: "7725" |       - in: formData | ||||||
|         in: formData |  | ||||||
|         name: user_id |         name: user_id | ||||||
|  |         type: integer | ||||||
|  |       - in: formData | ||||||
|  |         name: user_key | ||||||
|         type: string |         type: string | ||||||
|       responses: |       responses: | ||||||
|         "200": |         "200": | ||||||
|   | |||||||
| @@ -131,7 +131,7 @@ func TestTokenKeys(t *testing.T) { | |||||||
|  |  | ||||||
| 	msg1 := tt.RequestAuthGet[gin.H](t, data.AdminKey, baseUrl, fmt.Sprintf("/api/v2/messages/%s", msg1s["scn_msg_id"])) | 	msg1 := tt.RequestAuthGet[gin.H](t, data.AdminKey, baseUrl, fmt.Sprintf("/api/v2/messages/%s", msg1s["scn_msg_id"])) | ||||||
|  |  | ||||||
| 	tt.AssertEqual(t, "AllChannels", key7.KeytokenId, msg1["used_key_id"]) | 	tt.AssertEqual(t, "used_key_id", key7.KeytokenId, msg1["used_key_id"]) | ||||||
|  |  | ||||||
| 	tt.RequestPostShouldFail(t, baseUrl, "/", gin.H{ | 	tt.RequestPostShouldFail(t, baseUrl, "/", gin.H{ | ||||||
| 		"key":     key7.Token, | 		"key":     key7.Token, | ||||||
|   | |||||||
| @@ -124,3 +124,25 @@ func TestRequestLogSimple(t *testing.T) { | |||||||
| 	} | 	} | ||||||
|  |  | ||||||
| } | } | ||||||
|  |  | ||||||
|  | func TestRequestLogAPI(t *testing.T) { | ||||||
|  | 	ws, baseUrl, stop := tt.StartSimpleWebserver(t) | ||||||
|  | 	defer stop() | ||||||
|  |  | ||||||
|  | 	data := tt.InitDefaultData(t, ws) | ||||||
|  | 	time.Sleep(900 * time.Millisecond) | ||||||
|  |  | ||||||
|  | 	ctx := ws.NewSimpleTransactionContext(5 * time.Second) | ||||||
|  | 	defer ctx.Cancel() | ||||||
|  |  | ||||||
|  | 	rl1, _, err := ws.Database.Requests.ListRequestLogs(ctx, models.RequestLogFilter{}, nil, ct.Start()) | ||||||
|  | 	tt.TestFailIfErr(t, err) | ||||||
|  |  | ||||||
|  | 	tt.RequestAuthGet[gin.H](t, data.User[0].ReadKey, baseUrl, "/api/v2/users/"+data.User[0].UID) | ||||||
|  | 	time.Sleep(900 * time.Millisecond) | ||||||
|  |  | ||||||
|  | 	rl2, _, err := ws.Database.Requests.ListRequestLogs(ctx, models.RequestLogFilter{}, nil, ct.Start()) | ||||||
|  | 	tt.TestFailIfErr(t, err) | ||||||
|  |  | ||||||
|  | 	tt.AssertEqual(t, "requestlog.count", len(rl1)+1, len(rl2)) | ||||||
|  | } | ||||||
|   | |||||||
							
								
								
									
										290
									
								
								scnserver/test/response_test.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										290
									
								
								scnserver/test/response_test.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,290 @@ | |||||||
|  | package test | ||||||
|  |  | ||||||
|  | import ( | ||||||
|  | 	tt "blackforestbytes.com/simplecloudnotifier/test/util" | ||||||
|  | 	"fmt" | ||||||
|  | 	"github.com/gin-gonic/gin" | ||||||
|  | 	"testing" | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | func TestResponseChannel(t *testing.T) { | ||||||
|  | 	ws, baseUrl, stop := tt.StartSimpleWebserver(t) | ||||||
|  | 	defer stop() | ||||||
|  |  | ||||||
|  | 	data := tt.InitDefaultData(t, ws) | ||||||
|  |  | ||||||
|  | 	response := tt.RequestAuthGetRaw(t, data.User[0].AdminKey, baseUrl, fmt.Sprintf("/api/v2/users/%s/channels/%s", data.User[0].UID, data.User[0].Channels[0])) | ||||||
|  |  | ||||||
|  | 	tt.AssertJsonStructureMatch(t, "json[channel]", response, map[string]any{ | ||||||
|  | 		"channel_id":         "id", | ||||||
|  | 		"owner_user_id":      "id", | ||||||
|  | 		"internal_name":      "string", | ||||||
|  | 		"display_name":       "string", | ||||||
|  | 		"description_name":   "null", | ||||||
|  | 		"subscribe_key":      "string", | ||||||
|  | 		"timestamp_created":  "rfc3339", | ||||||
|  | 		"timestamp_lastsent": "rfc3339", | ||||||
|  | 		"messages_sent":      "int", | ||||||
|  | 		"subscription": map[string]any{ | ||||||
|  | 			"subscription_id":       "id", | ||||||
|  | 			"subscriber_user_id":    "id", | ||||||
|  | 			"channel_owner_user_id": "id", | ||||||
|  | 			"channel_id":            "id", | ||||||
|  | 			"channel_internal_name": "string", | ||||||
|  | 			"timestamp_created":     "rfc3339", | ||||||
|  | 			"confirmed":             "bool", | ||||||
|  | 		}, | ||||||
|  | 	}) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func TestResponseClient(t *testing.T) { | ||||||
|  | 	ws, baseUrl, stop := tt.StartSimpleWebserver(t) | ||||||
|  | 	defer stop() | ||||||
|  |  | ||||||
|  | 	data := tt.InitDefaultData(t, ws) | ||||||
|  |  | ||||||
|  | 	response := tt.RequestAuthGetRaw(t, data.User[2].AdminKey, baseUrl, fmt.Sprintf("/api/v2/users/%s/clients/%s", data.User[2].UID, data.User[2].Clients[0])) | ||||||
|  |  | ||||||
|  | 	tt.AssertJsonStructureMatch(t, "json[client]", response, map[string]any{ | ||||||
|  | 		"client_id":         "id", | ||||||
|  | 		"user_id":           "id", | ||||||
|  | 		"type":              "string", | ||||||
|  | 		"fcm_token":         "string", | ||||||
|  | 		"timestamp_created": "rfc3339", | ||||||
|  | 		"agent_model":       "string", | ||||||
|  | 		"agent_version":     "string", | ||||||
|  | 		"name":              "string|null", | ||||||
|  | 	}) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func TestResponseKeyToken1(t *testing.T) { | ||||||
|  | 	ws, baseUrl, stop := tt.StartSimpleWebserver(t) | ||||||
|  | 	defer stop() | ||||||
|  |  | ||||||
|  | 	data := tt.InitDefaultData(t, ws) | ||||||
|  |  | ||||||
|  | 	response := tt.RequestAuthGetRaw(t, data.User[0].AdminKey, baseUrl, fmt.Sprintf("/api/v2/users/%s/keys/%s", data.User[0].UID, data.User[0].Keys[0])) | ||||||
|  |  | ||||||
|  | 	tt.AssertJsonStructureMatch(t, "json[key]", response, map[string]any{ | ||||||
|  | 		"keytoken_id":        "id", | ||||||
|  | 		"name":               "string", | ||||||
|  | 		"timestamp_created":  "rfc3339", | ||||||
|  | 		"timestamp_lastused": "rfc3339|null", | ||||||
|  | 		"owner_user_id":      "id", | ||||||
|  | 		"all_channels":       "bool", | ||||||
|  | 		"channels":           []any{"string"}, | ||||||
|  | 		"permissions":        "string", | ||||||
|  | 		"messages_sent":      "int", | ||||||
|  | 	}) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func TestResponseKeyToken2(t *testing.T) { | ||||||
|  | 	ws, baseUrl, stop := tt.StartSimpleWebserver(t) | ||||||
|  | 	defer stop() | ||||||
|  |  | ||||||
|  | 	data := tt.InitSingleData(t, ws) | ||||||
|  |  | ||||||
|  | 	chan1 := tt.RequestAuthPost[gin.H](t, data.AdminKey, baseUrl, fmt.Sprintf("/api/v2/users/%s/channels", data.UID), gin.H{ | ||||||
|  | 		"name": "TestChan1asdf", | ||||||
|  | 	}) | ||||||
|  |  | ||||||
|  | 	type keyobj struct { | ||||||
|  | 		KeytokenId string `json:"keytoken_id"` | ||||||
|  | 	} | ||||||
|  | 	k0 := tt.RequestAuthPost[keyobj](t, data.AdminKey, baseUrl, fmt.Sprintf("/api/v2/users/%s/keys", data.UID), gin.H{ | ||||||
|  | 		"all_channels": false, | ||||||
|  | 		"channels":     []string{chan1["channel_id"].(string)}, | ||||||
|  | 		"name":         "TKey1", | ||||||
|  | 		"permissions":  "CS", | ||||||
|  | 	}) | ||||||
|  |  | ||||||
|  | 	response := tt.RequestAuthGetRaw(t, data.AdminKey, baseUrl, fmt.Sprintf("/api/v2/users/%s/keys/%s", data.UID, k0.KeytokenId)) | ||||||
|  |  | ||||||
|  | 	tt.AssertJsonStructureMatch(t, "json[key]", response, map[string]any{ | ||||||
|  | 		"keytoken_id":        "id", | ||||||
|  | 		"name":               "string", | ||||||
|  | 		"timestamp_created":  "rfc3339", | ||||||
|  | 		"timestamp_lastused": "rfc3339|null", | ||||||
|  | 		"owner_user_id":      "id", | ||||||
|  | 		"all_channels":       "bool", | ||||||
|  | 		"channels":           []any{"string"}, | ||||||
|  | 		"permissions":        "string", | ||||||
|  | 		"messages_sent":      "int", | ||||||
|  | 	}) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func TestResponseKeyToken3(t *testing.T) { | ||||||
|  | 	ws, baseUrl, stop := tt.StartSimpleWebserver(t) | ||||||
|  | 	defer stop() | ||||||
|  |  | ||||||
|  | 	data := tt.InitSingleData(t, ws) | ||||||
|  |  | ||||||
|  | 	response := tt.RequestAuthGetRaw(t, data.AdminKey, baseUrl, fmt.Sprintf("/api/v2/users/%s/keys/current", data.UID)) | ||||||
|  |  | ||||||
|  | 	tt.AssertJsonStructureMatch(t, "json[key]", response, map[string]any{ | ||||||
|  | 		"keytoken_id":        "id", | ||||||
|  | 		"name":               "string", | ||||||
|  | 		"timestamp_created":  "rfc3339", | ||||||
|  | 		"timestamp_lastused": "rfc3339|null", | ||||||
|  | 		"owner_user_id":      "id", | ||||||
|  | 		"all_channels":       "bool", | ||||||
|  | 		"channels":           []any{"string"}, | ||||||
|  | 		"permissions":        "string", | ||||||
|  | 		"messages_sent":      "int", | ||||||
|  | 		"token":              "string", | ||||||
|  | 	}) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func TestResponseKeyToken4(t *testing.T) { | ||||||
|  | 	ws, baseUrl, stop := tt.StartSimpleWebserver(t) | ||||||
|  | 	defer stop() | ||||||
|  |  | ||||||
|  | 	data := tt.InitSingleData(t, ws) | ||||||
|  |  | ||||||
|  | 	chan1 := tt.RequestAuthPost[gin.H](t, data.AdminKey, baseUrl, fmt.Sprintf("/api/v2/users/%s/channels", data.UID), gin.H{ | ||||||
|  | 		"name": "TestChan1asdf", | ||||||
|  | 	}) | ||||||
|  |  | ||||||
|  | 	response := tt.RequestAuthPostRaw(t, data.AdminKey, baseUrl, fmt.Sprintf("/api/v2/users/%s/keys", data.UID), gin.H{ | ||||||
|  | 		"all_channels": false, | ||||||
|  | 		"channels":     []string{chan1["channel_id"].(string)}, | ||||||
|  | 		"name":         "TKey1", | ||||||
|  | 		"permissions":  "CS", | ||||||
|  | 	}) | ||||||
|  |  | ||||||
|  | 	tt.AssertJsonStructureMatch(t, "json[key]", response, map[string]any{ | ||||||
|  | 		"keytoken_id":        "id", | ||||||
|  | 		"name":               "string", | ||||||
|  | 		"timestamp_created":  "rfc3339", | ||||||
|  | 		"timestamp_lastused": "rfc3339|null", | ||||||
|  | 		"owner_user_id":      "id", | ||||||
|  | 		"all_channels":       "bool", | ||||||
|  | 		"channels":           []any{"string"}, | ||||||
|  | 		"permissions":        "string", | ||||||
|  | 		"messages_sent":      "int", | ||||||
|  | 		"token":              "string", | ||||||
|  | 	}) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func TestResponseMessage(t *testing.T) { | ||||||
|  | 	ws, baseUrl, stop := tt.StartSimpleWebserver(t) | ||||||
|  | 	defer stop() | ||||||
|  |  | ||||||
|  | 	data := tt.InitDefaultData(t, ws) | ||||||
|  |  | ||||||
|  | 	response := tt.RequestAuthGetRaw(t, data.User[0].AdminKey, baseUrl, fmt.Sprintf("/api/v2/messages/%s", data.User[0].Messages[0])) | ||||||
|  |  | ||||||
|  | 	tt.AssertJsonStructureMatch(t, "json[message]", response, map[string]any{ | ||||||
|  | 		"message_id":            "id", | ||||||
|  | 		"sender_user_id":        "id", | ||||||
|  | 		"channel_internal_name": "string", | ||||||
|  | 		"channel_id":            "id", | ||||||
|  | 		"sender_name":           "string", | ||||||
|  | 		"sender_ip":             "string", | ||||||
|  | 		"timestamp":             "rfc3339", | ||||||
|  | 		"title":                 "string", | ||||||
|  | 		"content":               "null", | ||||||
|  | 		"priority":              "int", | ||||||
|  | 		"usr_message_id":        "null", | ||||||
|  | 		"used_key_id":           "id", | ||||||
|  | 		"trimmed":               "bool", | ||||||
|  | 	}) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func TestResponseSubscription(t *testing.T) { | ||||||
|  | 	ws, baseUrl, stop := tt.StartSimpleWebserver(t) | ||||||
|  | 	defer stop() | ||||||
|  |  | ||||||
|  | 	data := tt.InitDefaultData(t, ws) | ||||||
|  |  | ||||||
|  | 	response := tt.RequestAuthGetRaw(t, data.User[0].AdminKey, baseUrl, fmt.Sprintf("/api/v2/users/%s/subscriptions/%s", data.User[0].UID, data.User[0].Subscriptions[0])) | ||||||
|  |  | ||||||
|  | 	tt.AssertJsonStructureMatch(t, "json[subscription]", response, map[string]any{ | ||||||
|  | 		"subscription_id":       "id", | ||||||
|  | 		"subscriber_user_id":    "id", | ||||||
|  | 		"channel_owner_user_id": "id", | ||||||
|  | 		"channel_id":            "id", | ||||||
|  | 		"channel_internal_name": "string", | ||||||
|  | 		"timestamp_created":     "rfc3339", | ||||||
|  | 		"confirmed":             "bool", | ||||||
|  | 	}) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func TestResponseUser(t *testing.T) { | ||||||
|  | 	ws, baseUrl, stop := tt.StartSimpleWebserver(t) | ||||||
|  | 	defer stop() | ||||||
|  |  | ||||||
|  | 	data := tt.InitDefaultData(t, ws) | ||||||
|  |  | ||||||
|  | 	response := tt.RequestAuthGetRaw(t, data.User[0].AdminKey, baseUrl, fmt.Sprintf("/api/v2/users/%s", data.User[0].UID)) | ||||||
|  |  | ||||||
|  | 	tt.AssertJsonStructureMatch(t, "json[user]", response, map[string]any{ | ||||||
|  | 		"user_id":                        "id", | ||||||
|  | 		"username":                       "null", | ||||||
|  | 		"timestamp_created":              "rfc3339", | ||||||
|  | 		"timestamp_lastread":             "null", | ||||||
|  | 		"timestamp_lastsent":             "rfc3339", | ||||||
|  | 		"messages_sent":                  "int", | ||||||
|  | 		"quota_used":                     "int", | ||||||
|  | 		"quota_remaining":                "int", | ||||||
|  | 		"quota_max":                      "int", | ||||||
|  | 		"is_pro":                         "bool", | ||||||
|  | 		"default_channel":                "string", | ||||||
|  | 		"max_body_size":                  "int", | ||||||
|  | 		"max_title_length":               "int", | ||||||
|  | 		"default_priority":               "int", | ||||||
|  | 		"max_channel_name_length":        "int", | ||||||
|  | 		"max_channel_description_length": "int", | ||||||
|  | 		"max_sender_name_length":         "int", | ||||||
|  | 		"max_user_message_id_length":     "int", | ||||||
|  | 	}) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func TestResponseChannelPreview(t *testing.T) { | ||||||
|  | 	ws, baseUrl, stop := tt.StartSimpleWebserver(t) | ||||||
|  | 	defer stop() | ||||||
|  |  | ||||||
|  | 	data := tt.InitDefaultData(t, ws) | ||||||
|  |  | ||||||
|  | 	response := tt.RequestAuthGetRaw(t, data.User[1].AdminKey, baseUrl, fmt.Sprintf("/api/v2/preview/channels/%s", data.User[0].Channels[0])) | ||||||
|  |  | ||||||
|  | 	tt.AssertJsonStructureMatch(t, "json[channel]", response, map[string]any{ | ||||||
|  | 		"channel_id":       "id", | ||||||
|  | 		"owner_user_id":    "id", | ||||||
|  | 		"internal_name":    "string", | ||||||
|  | 		"display_name":     "string", | ||||||
|  | 		"description_name": "string|null", | ||||||
|  | 	}) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func TestResponseUserPreview(t *testing.T) { | ||||||
|  | 	ws, baseUrl, stop := tt.StartSimpleWebserver(t) | ||||||
|  | 	defer stop() | ||||||
|  |  | ||||||
|  | 	data := tt.InitDefaultData(t, ws) | ||||||
|  |  | ||||||
|  | 	response := tt.RequestAuthGetRaw(t, data.User[1].AdminKey, baseUrl, fmt.Sprintf("/api/v2/preview/users/%s", data.User[0].UID)) | ||||||
|  |  | ||||||
|  | 	tt.AssertJsonStructureMatch(t, "json[user]", response, map[string]any{ | ||||||
|  | 		"user_id":  "id", | ||||||
|  | 		"username": "string|null", | ||||||
|  | 	}) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func TestResponseKeyTokenPreview(t *testing.T) { | ||||||
|  | 	ws, baseUrl, stop := tt.StartSimpleWebserver(t) | ||||||
|  | 	defer stop() | ||||||
|  |  | ||||||
|  | 	data := tt.InitDefaultData(t, ws) | ||||||
|  |  | ||||||
|  | 	response := tt.RequestAuthGetRaw(t, data.User[1].AdminKey, baseUrl, fmt.Sprintf("/api/v2/preview/keys/%s", data.User[0].Keys[0])) | ||||||
|  |  | ||||||
|  | 	tt.AssertJsonStructureMatch(t, "json[key]", response, map[string]any{ | ||||||
|  | 		"keytoken_id":   "id", | ||||||
|  | 		"name":          "string", | ||||||
|  | 		"owner_user_id": "id", | ||||||
|  | 		"all_channels":  "bool", | ||||||
|  | 		"channels":      []any{"id"}, | ||||||
|  | 		"permissions":   "string", | ||||||
|  | 	}) | ||||||
|  | } | ||||||
| @@ -7,6 +7,7 @@ import ( | |||||||
| 	tt "blackforestbytes.com/simplecloudnotifier/test/util" | 	tt "blackforestbytes.com/simplecloudnotifier/test/util" | ||||||
| 	"fmt" | 	"fmt" | ||||||
| 	"github.com/gin-gonic/gin" | 	"github.com/gin-gonic/gin" | ||||||
|  | 	"math/rand/v2" | ||||||
| 	"net/url" | 	"net/url" | ||||||
| 	"strings" | 	"strings" | ||||||
| 	"testing" | 	"testing" | ||||||
| @@ -836,7 +837,7 @@ func TestSendWithTimestamp(t *testing.T) { | |||||||
|  |  | ||||||
| 	tt.AssertEqual(t, "messageCount", 1, len(pusher.Data)) | 	tt.AssertEqual(t, "messageCount", 1, len(pusher.Data)) | ||||||
| 	tt.AssertStrRepEqual(t, "msg.title", "TTT", pusher.Last().Message.Title) | 	tt.AssertStrRepEqual(t, "msg.title", "TTT", pusher.Last().Message.Title) | ||||||
| 	tt.AssertStrRepEqual(t, "msg.TimestampClient", ts, pusher.Last().Message.TimestampClient.Unix()) | 	tt.AssertStrRepEqual(t, "msg.TimestampClient", ts, pusher.Last().Message.TimestampClient.Time().Unix()) | ||||||
| 	tt.AssertStrRepEqual(t, "msg.Timestamp", ts, pusher.Last().Message.Timestamp().Unix()) | 	tt.AssertStrRepEqual(t, "msg.Timestamp", ts, pusher.Last().Message.Timestamp().Unix()) | ||||||
| 	tt.AssertNotStrRepEqual(t, "msg.ts", pusher.Last().Message.TimestampClient, pusher.Last().Message.TimestampReal) | 	tt.AssertNotStrRepEqual(t, "msg.ts", pusher.Last().Message.TimestampClient, pusher.Last().Message.TimestampReal) | ||||||
| 	tt.AssertStrRepEqual(t, "msg.scn_msg_id", msg1["scn_msg_id"], pusher.Last().Message.MessageID) | 	tt.AssertStrRepEqual(t, "msg.scn_msg_id", msg1["scn_msg_id"], pusher.Last().Message.MessageID) | ||||||
| @@ -1341,8 +1342,14 @@ func TestSendParallel(t *testing.T) { | |||||||
|  |  | ||||||
| 	uid := r0["user_id"].(string) | 	uid := r0["user_id"].(string) | ||||||
| 	sendtok := r0["send_key"].(string) | 	sendtok := r0["send_key"].(string) | ||||||
|  | 	admintok := r0["admin_key"].(string) | ||||||
|  |  | ||||||
| 	count := 128 | 	count := 512 | ||||||
|  |  | ||||||
|  | 	chanNames := make([]string, 0) | ||||||
|  | 	for i := 0; i < count/50; i++ { | ||||||
|  | 		chanNames = append(chanNames, tt.ShortLipsum0(1)) | ||||||
|  | 	} | ||||||
|  |  | ||||||
| 	sem := make(chan tt.Void, count) // semaphore pattern | 	sem := make(chan tt.Void, count) // semaphore pattern | ||||||
| 	for i := 0; i < count; i++ { | 	for i := 0; i < count; i++ { | ||||||
| @@ -1350,11 +1357,31 @@ func TestSendParallel(t *testing.T) { | |||||||
| 			defer func() { | 			defer func() { | ||||||
| 				sem <- tt.Void{} | 				sem <- tt.Void{} | ||||||
| 			}() | 			}() | ||||||
|  |  | ||||||
|  | 			if rand.Int()%2 == 0 { | ||||||
| 				tt.RequestPost[gin.H](t, baseUrl, "/", gin.H{ | 				tt.RequestPost[gin.H](t, baseUrl, "/", gin.H{ | ||||||
| 					"key":     sendtok, | 					"key":     sendtok, | ||||||
| 					"user_id": uid, | 					"user_id": uid, | ||||||
| 					"title":   tt.ShortLipsum0(2), | 					"title":   tt.ShortLipsum0(2), | ||||||
| 				}) | 				}) | ||||||
|  | 			} else { | ||||||
|  | 				tt.RequestPost[gin.H](t, baseUrl, "/", gin.H{ | ||||||
|  | 					"key":     sendtok, | ||||||
|  | 					"user_id": uid, | ||||||
|  | 					"title":   tt.ShortLipsum0(2), | ||||||
|  | 					"channel": chanNames[rand.IntN(len(chanNames))], | ||||||
|  | 				}) | ||||||
|  | 			} | ||||||
|  |  | ||||||
|  | 			tt.RequestGet[tt.Void](t, baseUrl, fmt.Sprintf("/api/ping")) | ||||||
|  |  | ||||||
|  | 			tt.RequestAuthGet[gin.H](t, admintok, baseUrl, fmt.Sprintf("/api/v2/messages")) | ||||||
|  |  | ||||||
|  | 			tt.RequestAuthGet[gin.H](t, admintok, baseUrl, "/api/v2/users/"+uid) | ||||||
|  |  | ||||||
|  | 			tt.RequestAuthGet[gin.H](t, admintok, baseUrl, "/api/v2/users/"+uid+"/channels") | ||||||
|  |  | ||||||
|  | 			tt.RequestAuthGet[gin.H](t, admintok, baseUrl, "/api/v2/users/"+uid+"/clients") | ||||||
| 		}() | 		}() | ||||||
| 	} | 	} | ||||||
| 	// wait for goroutines to finish | 	// wait for goroutines to finish | ||||||
|   | |||||||
| @@ -5,6 +5,7 @@ import ( | |||||||
| 	"fmt" | 	"fmt" | ||||||
| 	"github.com/gin-gonic/gin" | 	"github.com/gin-gonic/gin" | ||||||
| 	"github.com/rs/zerolog/log" | 	"github.com/rs/zerolog/log" | ||||||
|  | 	"gogs.mikescher.com/BlackForestBytes/goext/langext" | ||||||
| 	"gogs.mikescher.com/BlackForestBytes/goext/timeext" | 	"gogs.mikescher.com/BlackForestBytes/goext/timeext" | ||||||
| 	"gopkg.in/loremipsum.v1" | 	"gopkg.in/loremipsum.v1" | ||||||
| 	"testing" | 	"testing" | ||||||
| @@ -63,6 +64,11 @@ type Userdat struct { | |||||||
| 	SendKey       string | 	SendKey       string | ||||||
| 	AdminKey      string | 	AdminKey      string | ||||||
| 	ReadKey       string | 	ReadKey       string | ||||||
|  | 	Clients       []string | ||||||
|  | 	Channels      []string | ||||||
|  | 	Messages      []string | ||||||
|  | 	Keys          []string | ||||||
|  | 	Subscriptions []string | ||||||
| } | } | ||||||
|  |  | ||||||
| const PX = -1 | const PX = -1 | ||||||
| @@ -367,7 +373,8 @@ func InitDefaultData(t *testing.T, ws *logic.Application) DefData { | |||||||
| 		body["agent_version"] = cex.AgentVersion | 		body["agent_version"] = cex.AgentVersion | ||||||
| 		body["client_type"] = cex.ClientType | 		body["client_type"] = cex.ClientType | ||||||
| 		body["fcm_token"] = cex.FCMTok | 		body["fcm_token"] = cex.FCMTok | ||||||
| 		RequestAuthPost[gin.H](t, users[cex.User].AdminKey, baseUrl, fmt.Sprintf("/api/v2/users/%s/clients", users[cex.User].UID), body) | 		r0 := RequestAuthPost[gin.H](t, users[cex.User].AdminKey, baseUrl, fmt.Sprintf("/api/v2/users/%s/clients", users[cex.User].UID), body) | ||||||
|  | 		users[cex.User].Clients = append(users[cex.User].Clients, r0["client_id"].(string)) | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	// Create Messages | 	// Create Messages | ||||||
| @@ -398,7 +405,8 @@ func InitDefaultData(t *testing.T, ws *logic.Application) DefData { | |||||||
| 			body["timestamp"] = (time.Now().Add(mex.TSOffset)).Unix() | 			body["timestamp"] = (time.Now().Add(mex.TSOffset)).Unix() | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		RequestPost[gin.H](t, baseUrl, "/", body) | 		r0 := RequestPost[gin.H](t, baseUrl, "/", body) | ||||||
|  | 		users[mex.User].Messages = append(users[mex.User].Messages, r0["scn_msg_id"].(string)) | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	// create manual channels | 	// create manual channels | ||||||
| @@ -407,6 +415,45 @@ func InitDefaultData(t *testing.T, ws *logic.Application) DefData { | |||||||
| 		RequestAuthPost[Void](t, users[9].AdminKey, baseUrl, fmt.Sprintf("/api/v2/users/%s/channels", users[9].UID), gin.H{"name": "manual@chan"}) | 		RequestAuthPost[Void](t, users[9].AdminKey, baseUrl, fmt.Sprintf("/api/v2/users/%s/channels", users[9].UID), gin.H{"name": "manual@chan"}) | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
|  | 	// list channels | ||||||
|  |  | ||||||
|  | 	for i, usr := range users { | ||||||
|  | 		type schan struct { | ||||||
|  | 			ID string `json:"channel_id"` | ||||||
|  | 		} | ||||||
|  | 		type chanlist struct { | ||||||
|  | 			Channels []schan `json:"channels"` | ||||||
|  | 		} | ||||||
|  | 		r0 := RequestAuthGet[chanlist](t, usr.AdminKey, baseUrl, fmt.Sprintf("/api/v2/users/%s/channels?selector=%s", usr.UID, "owned")) | ||||||
|  | 		users[i].Channels = langext.ArrMap(r0.Channels, func(v schan) string { return v.ID }) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// list keys | ||||||
|  |  | ||||||
|  | 	for i, usr := range users { | ||||||
|  | 		type skey struct { | ||||||
|  | 			ID string `json:"keytoken_id"` | ||||||
|  | 		} | ||||||
|  | 		type keylist struct { | ||||||
|  | 			Keys []skey `json:"keys"` | ||||||
|  | 		} | ||||||
|  | 		r0 := RequestAuthGet[keylist](t, usr.AdminKey, baseUrl, fmt.Sprintf("/api/v2/users/%s/keys", usr.UID)) | ||||||
|  | 		users[i].Keys = langext.ArrMap(r0.Keys, func(v skey) string { return v.ID }) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// list subscriptions | ||||||
|  |  | ||||||
|  | 	for i, usr := range users { | ||||||
|  | 		type ssub struct { | ||||||
|  | 			ID string `json:"subscription_id"` | ||||||
|  | 		} | ||||||
|  | 		type sublist struct { | ||||||
|  | 			Subs []ssub `json:"subscriptions"` | ||||||
|  | 		} | ||||||
|  | 		r0 := RequestAuthGet[sublist](t, usr.AdminKey, baseUrl, fmt.Sprintf("/api/v2/users/%s/subscriptions?direction=%s&confirmation=%s", usr.UID, "outgoing", "confirmed")) | ||||||
|  | 		users[i].Subscriptions = langext.ArrMap(r0.Subs, func(v ssub) string { return v.ID }) | ||||||
|  | 	} | ||||||
|  |  | ||||||
| 	// Sub/Unsub for Users 12+13 | 	// Sub/Unsub for Users 12+13 | ||||||
|  |  | ||||||
| 	{ | 	{ | ||||||
| @@ -463,18 +510,20 @@ func InitSingleData(t *testing.T, ws *logic.Application) SingleData { | |||||||
|  |  | ||||||
| 	success = true | 	success = true | ||||||
|  |  | ||||||
| 	return SingleData{ | 	sd := SingleData{ | ||||||
| 		UID:      r0.UserId, | 		UID:      r0.UserId, | ||||||
| 		AdminKey: r0.AdminKey, | 		AdminKey: r0.AdminKey, | ||||||
| 		SendKey:  r0.SendKey, | 		SendKey:  r0.SendKey, | ||||||
| 		ReadKey:  r0.ReadKey, | 		ReadKey:  r0.ReadKey, | ||||||
| 		ClientID: r0.Clients[0].ClientId, | 		ClientID: r0.Clients[0].ClientId, | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
|  | 	return sd | ||||||
| } | } | ||||||
|  |  | ||||||
| func doSubscribe(t *testing.T, baseUrl string, user Userdat, chanOwner Userdat, chanInternalName string) { | func doSubscribe(t *testing.T, baseUrl string, user Userdat, chanOwner Userdat, chanInternalName string) { | ||||||
|  |  | ||||||
| 	if user == chanOwner { | 	if user.UID == chanOwner.UID { | ||||||
|  |  | ||||||
| 		RequestAuthPost[Void](t, user.AdminKey, baseUrl, fmt.Sprintf("/api/v2/users/%s/channels", user.UID), gin.H{ | 		RequestAuthPost[Void](t, user.AdminKey, baseUrl, fmt.Sprintf("/api/v2/users/%s/channels", user.UID), gin.H{ | ||||||
| 			"channel_owner_user_id": chanOwner.UID, | 			"channel_owner_user_id": chanOwner.UID, | ||||||
|   | |||||||
| @@ -1,7 +1,6 @@ | |||||||
| package util | package util | ||||||
|  |  | ||||||
| import ( | import ( | ||||||
| 	"blackforestbytes.com/simplecloudnotifier/api/ginext" |  | ||||||
| 	"fmt" | 	"fmt" | ||||||
| 	"github.com/gin-gonic/gin" | 	"github.com/gin-gonic/gin" | ||||||
| 	"github.com/rs/zerolog/log" | 	"github.com/rs/zerolog/log" | ||||||
| @@ -13,7 +12,6 @@ func SetBufLogger() { | |||||||
| 	buflogger = &BufferWriter{cw: createConsoleWriter()} | 	buflogger = &BufferWriter{cw: createConsoleWriter()} | ||||||
| 	log.Logger = createLogger(buflogger) | 	log.Logger = createLogger(buflogger) | ||||||
| 	gin.SetMode(gin.ReleaseMode) | 	gin.SetMode(gin.ReleaseMode) | ||||||
| 	ginext.SuppressGinLogs = true |  | ||||||
| } | } | ||||||
|  |  | ||||||
| func ClearBufLogger(dump bool) { | func ClearBufLogger(dump bool) { | ||||||
| @@ -24,7 +22,6 @@ func ClearBufLogger(dump bool) { | |||||||
| 	log.Logger = createLogger(createConsoleWriter()) | 	log.Logger = createLogger(createConsoleWriter()) | ||||||
| 	buflogger = nil | 	buflogger = nil | ||||||
| 	gin.SetMode(gin.TestMode) | 	gin.SetMode(gin.TestMode) | ||||||
| 	ginext.SuppressGinLogs = false |  | ||||||
| 	if !dump { | 	if !dump { | ||||||
| 		log.Info().Msgf("Suppressed %d logmessages / printf-statements", size) | 		log.Info().Msgf("Suppressed %d logmessages / printf-statements", size) | ||||||
| 	} | 	} | ||||||
|   | |||||||
| @@ -26,10 +26,18 @@ func RequestAuthGet[TResult any](t *testing.T, akey string, baseURL string, urlS | |||||||
| 	return RequestAny[TResult](t, akey, "GET", baseURL, urlSuffix, nil, true) | 	return RequestAny[TResult](t, akey, "GET", baseURL, urlSuffix, nil, true) | ||||||
| } | } | ||||||
|  |  | ||||||
|  | func RequestAuthGetRaw(t *testing.T, akey string, baseURL string, urlSuffix string) string { | ||||||
|  | 	return RequestAny[string](t, akey, "GET", baseURL, urlSuffix, nil, false) | ||||||
|  | } | ||||||
|  |  | ||||||
| func RequestPost[TResult any](t *testing.T, baseURL string, urlSuffix string, body any) TResult { | func RequestPost[TResult any](t *testing.T, baseURL string, urlSuffix string, body any) TResult { | ||||||
| 	return RequestAny[TResult](t, "", "POST", baseURL, urlSuffix, body, true) | 	return RequestAny[TResult](t, "", "POST", baseURL, urlSuffix, body, true) | ||||||
| } | } | ||||||
|  |  | ||||||
|  | func RequestAuthPostRaw(t *testing.T, akey string, baseURL string, urlSuffix string, body any) string { | ||||||
|  | 	return RequestAny[string](t, akey, "POST", baseURL, urlSuffix, body, false) | ||||||
|  | } | ||||||
|  |  | ||||||
| func RequestAuthPost[TResult any](t *testing.T, akey string, baseURL string, urlSuffix string, body any) TResult { | func RequestAuthPost[TResult any](t *testing.T, akey string, baseURL string, urlSuffix string, body any) TResult { | ||||||
| 	return RequestAny[TResult](t, akey, "POST", baseURL, urlSuffix, body, true) | 	return RequestAny[TResult](t, akey, "POST", baseURL, urlSuffix, body, true) | ||||||
| } | } | ||||||
| @@ -166,14 +174,22 @@ func RequestAny[TResult any](t *testing.T, akey string, method string, baseURL s | |||||||
| 		TestFailFmt(t, "Statuscode != 200 (actual = %d)", resp.StatusCode) | 		TestFailFmt(t, "Statuscode != 200 (actual = %d)", resp.StatusCode) | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	var data TResult |  | ||||||
| 	if deserialize { | 	if deserialize { | ||||||
|  | 		var data TResult | ||||||
| 		if err := json.Unmarshal(respBodyBin, &data); err != nil { | 		if err := json.Unmarshal(respBodyBin, &data); err != nil { | ||||||
| 			TestFailErr(t, err) | 			TestFailErr(t, err) | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 			return data | 			return data | ||||||
|  | 		} | ||||||
|  | 		return data | ||||||
|  | 	} else { | ||||||
|  | 		if _, ok := (any(*new(TResult))).([]byte); ok { | ||||||
|  | 			return any(respBodyBin).(TResult) | ||||||
|  | 		} else if _, ok := (any(*new(TResult))).(string); ok { | ||||||
|  | 			return any(string(respBodyBin)).(TResult) | ||||||
|  | 		} else { | ||||||
|  | 			return *new(TResult) | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
| } | } | ||||||
|  |  | ||||||
| func RequestAuthAnyShouldFail(t *testing.T, akey string, method string, baseURL string, urlSuffix string, body any, expectedStatusCode int, errcode apierr.APIError) { | func RequestAuthAnyShouldFail(t *testing.T, akey string, method string, baseURL string, urlSuffix string, body any, expectedStatusCode int, errcode apierr.APIError) { | ||||||
|   | |||||||
							
								
								
									
										176
									
								
								scnserver/test/util/structure.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										176
									
								
								scnserver/test/util/structure.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,176 @@ | |||||||
|  | package util | ||||||
|  |  | ||||||
|  | import ( | ||||||
|  | 	"encoding/json" | ||||||
|  | 	"fmt" | ||||||
|  | 	"gogs.mikescher.com/BlackForestBytes/goext/langext" | ||||||
|  | 	"reflect" | ||||||
|  | 	"testing" | ||||||
|  | 	"time" | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | func AssertJsonStructureMatch(t *testing.T, key string, jsonData string, expected map[string]any) { | ||||||
|  |  | ||||||
|  | 	realData := make(map[string]any) | ||||||
|  |  | ||||||
|  | 	err := json.Unmarshal([]byte(jsonData), &realData) | ||||||
|  | 	if err != nil { | ||||||
|  | 		t.Errorf("Failed to decode json of [%s]: %s", key, err.Error()) | ||||||
|  | 		return | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	assertjsonStructureMatchMapObject(t, expected, realData, key) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func assertJsonStructureMatch(t *testing.T, schema any, realValue any, keyPath string) { | ||||||
|  |  | ||||||
|  | 	if strschema, ok := schema.(string); ok { | ||||||
|  |  | ||||||
|  | 		assertjsonStructureMatchSingleValue(t, strschema, realValue, keyPath) | ||||||
|  |  | ||||||
|  | 	} else if mapschema, ok := schema.(map[string]any); ok { | ||||||
|  |  | ||||||
|  | 		if reflect.ValueOf(realValue).Kind() != reflect.Map { | ||||||
|  | 			t.Errorf("Key < %s > is not a object (its actually %T: '%v')", keyPath, realValue, realValue) | ||||||
|  | 			return | ||||||
|  | 		} | ||||||
|  | 		if _, ok := realValue.(map[string]any); !ok { | ||||||
|  | 			t.Errorf("Key < %s > is not a object[recursive] (its actually %T: '%v')", keyPath, realValue, realValue) | ||||||
|  | 			return | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		assertjsonStructureMatchMapObject(t, mapschema, realValue.(map[string]any), keyPath) | ||||||
|  |  | ||||||
|  | 	} else if arrschema, ok := schema.([]any); ok && len(arrschema) == 1 { | ||||||
|  |  | ||||||
|  | 		if _, ok := realValue.([]any); !ok { | ||||||
|  | 			t.Errorf("Key < %s > is not a array[recursive] (its actually %T: '%v')", keyPath, realValue, realValue) | ||||||
|  | 			return | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		assertjsonStructureMatchArray(t, arrschema, realValue.([]any), keyPath) | ||||||
|  |  | ||||||
|  | 	} else { | ||||||
|  | 		t.Errorf("Unknown schema type '%s' for key < %s >", schema, keyPath) | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func assertjsonStructureMatchSingleValue(t *testing.T, strschema string, realValue any, keyPath string) { | ||||||
|  | 	switch strschema { | ||||||
|  | 	case "id": | ||||||
|  | 		if _, ok := realValue.(string); !ok { | ||||||
|  | 			t.Errorf("Key < %s > is not a string<id> (its actually %T: '%v')", keyPath, realValue, realValue) | ||||||
|  | 			return | ||||||
|  | 		} | ||||||
|  | 		if len(realValue.(string)) != 24 { //TODO validate checksum? | ||||||
|  | 			t.Errorf("Key < %s > is not a valid entity-id date (its '%v')", keyPath, realValue) | ||||||
|  | 			return | ||||||
|  | 		} | ||||||
|  | 	case "string": | ||||||
|  | 		if _, ok := realValue.(string); !ok { | ||||||
|  | 			t.Errorf("Key < %s > is not a string (its actually %T: '%v')", keyPath, realValue, realValue) | ||||||
|  | 			return | ||||||
|  | 		} | ||||||
|  | 	case "null": | ||||||
|  | 		if !langext.IsNil(realValue) { | ||||||
|  | 			t.Errorf("Key < %s > is not a NULL (its actually %T: '%v')", keyPath, realValue, realValue) | ||||||
|  | 			return | ||||||
|  | 		} | ||||||
|  | 	case "string|null": | ||||||
|  | 		if langext.IsNil(realValue) { | ||||||
|  | 			return // OK | ||||||
|  | 		} else if _, ok := realValue.(string); !ok { | ||||||
|  | 			return // OK | ||||||
|  | 		} else { | ||||||
|  | 			t.Errorf("Key < %s > is not a string|null (its actually %T: '%v')", keyPath, realValue, realValue) | ||||||
|  | 			return | ||||||
|  | 		} | ||||||
|  | 	case "rfc3339": | ||||||
|  | 		if _, ok := realValue.(string); !ok { | ||||||
|  | 			t.Errorf("Key < %s > is not a string<rfc3339> (its actually %T: '%v')", keyPath, realValue, realValue) | ||||||
|  | 			return | ||||||
|  | 		} | ||||||
|  | 		if _, err := time.Parse(time.RFC3339, realValue.(string)); err != nil { | ||||||
|  | 			t.Errorf("Key < %s > is not a valid rfc3339 date (its '%v')", keyPath, realValue) | ||||||
|  | 			return | ||||||
|  | 		} | ||||||
|  | 	case "rfc3339|null": | ||||||
|  | 		if langext.IsNil(realValue) { | ||||||
|  | 			return // OK | ||||||
|  | 		} | ||||||
|  | 		if _, ok := realValue.(string); !ok { | ||||||
|  | 			t.Errorf("Key < %s > is not a string<rfc3339> (its actually %T: '%v')", keyPath, realValue, realValue) | ||||||
|  | 			return | ||||||
|  | 		} | ||||||
|  | 		if _, err := time.Parse(time.RFC3339, realValue.(string)); err != nil { | ||||||
|  | 			t.Errorf("Key < %s > is not a valid rfc3339 date (its '%v')", keyPath, realValue) | ||||||
|  | 			return | ||||||
|  | 		} | ||||||
|  | 	case "int": | ||||||
|  | 		if _, ok := realValue.(float64); !ok { | ||||||
|  | 			t.Errorf("Key < %s > is not a int (its actually %T: '%v')", keyPath, realValue, realValue) | ||||||
|  | 			return | ||||||
|  | 		} | ||||||
|  | 		if realValue.(float64) != float64(int(realValue.(float64))) { | ||||||
|  | 			t.Errorf("Key < %s > is not a int (its actually %T: '%v')", keyPath, realValue, realValue) | ||||||
|  | 			return | ||||||
|  | 		} | ||||||
|  | 	case "float": | ||||||
|  | 		if _, ok := realValue.(float64); !ok { | ||||||
|  | 			t.Errorf("Key < %s > is not a int (its actually %T: '%v')", keyPath, realValue, realValue) | ||||||
|  | 			return | ||||||
|  | 		} | ||||||
|  | 	case "bool": | ||||||
|  | 		if _, ok := realValue.(bool); !ok { | ||||||
|  | 			t.Errorf("Key < %s > is not a int (its actually %T: '%v')", keyPath, realValue, realValue) | ||||||
|  | 			return | ||||||
|  | 		} | ||||||
|  | 	default: | ||||||
|  | 		t.Errorf("Unknown schema type '%s' for key < %s >", strschema, keyPath) | ||||||
|  | 		return | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func assertjsonStructureMatchMapObject(t *testing.T, mapschema map[string]any, realValue map[string]any, keyPath string) { | ||||||
|  |  | ||||||
|  | 	for k := range mapschema { | ||||||
|  | 		if _, ok := realValue[k]; !ok { | ||||||
|  | 			t.Errorf("Missing Key: < %s >", keyPath+"."+k) | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	for k := range realValue { | ||||||
|  | 		if _, ok := mapschema[k]; !ok { | ||||||
|  | 			t.Errorf("Additional key: < %s >", keyPath+"."+k) | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	for k, v := range realValue { | ||||||
|  |  | ||||||
|  | 		kpath := keyPath + "." + k | ||||||
|  |  | ||||||
|  | 		schema, ok := mapschema[k] | ||||||
|  |  | ||||||
|  | 		if !ok { | ||||||
|  | 			t.Errorf("Key < %s > is missing in response", kpath) | ||||||
|  | 			continue | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		assertJsonStructureMatch(t, schema, v, kpath) | ||||||
|  |  | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func assertjsonStructureMatchArray(t *testing.T, arrschema []any, realValue []any, keyPath string) { | ||||||
|  |  | ||||||
|  | 	if len(arrschema) != 1 { | ||||||
|  | 		t.Errorf("Array schema must have exactly one element, but got %d", len(arrschema)) | ||||||
|  | 		return | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	for i, realArrVal := range realValue { | ||||||
|  | 		assertJsonStructureMatch(t, arrschema[0], realArrVal, fmt.Sprintf("%s[%d]", keyPath, i)) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | } | ||||||
| @@ -3,11 +3,11 @@ package util | |||||||
| import ( | import ( | ||||||
| 	scn "blackforestbytes.com/simplecloudnotifier" | 	scn "blackforestbytes.com/simplecloudnotifier" | ||||||
| 	"blackforestbytes.com/simplecloudnotifier/api" | 	"blackforestbytes.com/simplecloudnotifier/api" | ||||||
| 	"blackforestbytes.com/simplecloudnotifier/api/ginext" |  | ||||||
| 	"blackforestbytes.com/simplecloudnotifier/google" | 	"blackforestbytes.com/simplecloudnotifier/google" | ||||||
| 	"blackforestbytes.com/simplecloudnotifier/jobs" | 	"blackforestbytes.com/simplecloudnotifier/jobs" | ||||||
| 	"blackforestbytes.com/simplecloudnotifier/logic" | 	"blackforestbytes.com/simplecloudnotifier/logic" | ||||||
| 	"blackforestbytes.com/simplecloudnotifier/push" | 	"blackforestbytes.com/simplecloudnotifier/push" | ||||||
|  | 	"gogs.mikescher.com/BlackForestBytes/goext/ginext" | ||||||
| 	"gogs.mikescher.com/BlackForestBytes/goext/langext" | 	"gogs.mikescher.com/BlackForestBytes/goext/langext" | ||||||
| 	"os" | 	"os" | ||||||
| 	"path/filepath" | 	"path/filepath" | ||||||
| @@ -88,7 +88,13 @@ func StartSimpleWebserver(t *testing.T) (*logic.Application, string, func()) { | |||||||
| 		TestFailErr(t, err) | 		TestFailErr(t, err) | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	ginengine := ginext.NewEngine(scn.Conf) | 	ginengine := ginext.NewEngine(ginext.Options{ | ||||||
|  | 		AllowCors:             &scn.Conf.Cors, | ||||||
|  | 		GinDebug:              &scn.Conf.GinDebug, | ||||||
|  | 		BufferBody:            langext.PTrue, | ||||||
|  | 		Timeout:               langext.Ptr(time.Duration(int64(scn.Conf.RequestTimeout) * int64(scn.Conf.RequestMaxRetry))), | ||||||
|  | 		BuildRequestBindError: logic.BuildGinRequestError, | ||||||
|  | 	}) | ||||||
|  |  | ||||||
| 	router := api.NewRouter(app) | 	router := api.NewRouter(app) | ||||||
|  |  | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user