save
This commit is contained in:
		
							parent
							
								
									9b15ac9fd3
								
							
						
					
					
						commit
						27df1a73b5
					
				
					 28 changed files with 1695 additions and 247 deletions
				
			
		|  | @ -11,9 +11,3 @@ SMTP_FROM="noreply@example.com" | ||||||
| # !! Choose your LN_SERVICE carefully!! | # !! Choose your LN_SERVICE carefully!! | ||||||
| # NOTE: The LN_SERVICE must support LUD-21 Payment Verification | # NOTE: The LN_SERVICE must support LUD-21 Payment Verification | ||||||
| LN_ADDRESS="your@node.win" | LN_ADDRESS="your@node.win" | ||||||
| # maps.co for GeoCoder (postal address verification) |  | ||||||
| # Get your free API key: https://geocode.maps.co |  | ||||||
| GEOCODE_MAPS_CO_API_KEY="your-api-key-from-geocode.maps.co" |  | ||||||
| # Plausible for privacy-respecting page analytics |  | ||||||
| # https://github.com/plausible/analytics |  | ||||||
| PLAUSIBLE_HOST="https://plausible.io/" |  | ||||||
|  | @ -10,6 +10,7 @@ | ||||||
|     "phpmailer/phpmailer": "^6.9.2", |     "phpmailer/phpmailer": "^6.9.2", | ||||||
|     "vlucas/phpdotenv": "^5.6", |     "vlucas/phpdotenv": "^5.6", | ||||||
|     "web-auth/webauthn-lib": "^5.0", |     "web-auth/webauthn-lib": "^5.0", | ||||||
|     "twig/twig": "^3.0" |     "twig/twig": "^3.0", | ||||||
|  |     "swentel/nostr-php": "^1.5" | ||||||
|   } |   } | ||||||
| } | } | ||||||
|  |  | ||||||
							
								
								
									
										825
									
								
								composer.lock
									
										
									
										generated
									
									
									
								
							
							
						
						
									
										825
									
								
								composer.lock
									
										
									
										generated
									
									
									
								
							|  | @ -4,8 +4,54 @@ | ||||||
|         "Read more about it at https://getcomposer.org/doc/01-basic-usage.md#installing-dependencies", |         "Read more about it at https://getcomposer.org/doc/01-basic-usage.md#installing-dependencies", | ||||||
|         "This file is @generated automatically" |         "This file is @generated automatically" | ||||||
|     ], |     ], | ||||||
|     "content-hash": "ea128e458544f87060ad53880489fcc2", |     "content-hash": "408b4a1daa73232eabf14c566a3e5d8d", | ||||||
|     "packages": [ |     "packages": [ | ||||||
|  |         { | ||||||
|  |             "name": "bitwasp/bech32", | ||||||
|  |             "version": "v0.0.1", | ||||||
|  |             "source": { | ||||||
|  |                 "type": "git", | ||||||
|  |                 "url": "https://github.com/Bit-Wasp/bech32.git", | ||||||
|  |                 "reference": "e1ea58c848a4ec59d81b697b3dfe9cc99968d0e7" | ||||||
|  |             }, | ||||||
|  |             "dist": { | ||||||
|  |                 "type": "zip", | ||||||
|  |                 "url": "https://api.github.com/repos/Bit-Wasp/bech32/zipball/e1ea58c848a4ec59d81b697b3dfe9cc99968d0e7", | ||||||
|  |                 "reference": "e1ea58c848a4ec59d81b697b3dfe9cc99968d0e7", | ||||||
|  |                 "shasum": "" | ||||||
|  |             }, | ||||||
|  |             "require-dev": { | ||||||
|  |                 "phpunit/phpunit": "^5.4.0", | ||||||
|  |                 "squizlabs/php_codesniffer": "^2.0.0" | ||||||
|  |             }, | ||||||
|  |             "type": "library", | ||||||
|  |             "autoload": { | ||||||
|  |                 "files": [ | ||||||
|  |                     "src/bech32.php" | ||||||
|  |                 ], | ||||||
|  |                 "psr-4": { | ||||||
|  |                     "BitWasp\\Bech32\\": "src/" | ||||||
|  |                 } | ||||||
|  |             }, | ||||||
|  |             "notification-url": "https://packagist.org/downloads/", | ||||||
|  |             "license": [ | ||||||
|  |                 "Unlicense" | ||||||
|  |             ], | ||||||
|  |             "authors": [ | ||||||
|  |                 { | ||||||
|  |                     "name": "Thomas Kerin", | ||||||
|  |                     "homepage": "https://thomaskerin.io", | ||||||
|  |                     "role": "Author" | ||||||
|  |                 } | ||||||
|  |             ], | ||||||
|  |             "description": "Pure (no dependencies) implementation of bech32", | ||||||
|  |             "homepage": "https://github.com/bit-wasp/bech32", | ||||||
|  |             "support": { | ||||||
|  |                 "issues": "https://github.com/Bit-Wasp/bech32/issues", | ||||||
|  |                 "source": "https://github.com/Bit-Wasp/bech32/tree/more-tests" | ||||||
|  |             }, | ||||||
|  |             "time": "2018-02-05T22:23:47+00:00" | ||||||
|  |         }, | ||||||
|         { |         { | ||||||
|             "name": "brick/math", |             "name": "brick/math", | ||||||
|             "version": "0.12.1", |             "version": "0.12.1", | ||||||
|  | @ -618,6 +664,239 @@ | ||||||
|             }, |             }, | ||||||
|             "time": "2024-10-13T11:29:49+00:00" |             "time": "2024-10-13T11:29:49+00:00" | ||||||
|         }, |         }, | ||||||
|  |         { | ||||||
|  |             "name": "phrity/net-stream", | ||||||
|  |             "version": "2.1.2", | ||||||
|  |             "source": { | ||||||
|  |                 "type": "git", | ||||||
|  |                 "url": "https://github.com/sirn-se/phrity-net-stream.git", | ||||||
|  |                 "reference": "e6ace997168bebcce814c95cd5c78c78663ae49a" | ||||||
|  |             }, | ||||||
|  |             "dist": { | ||||||
|  |                 "type": "zip", | ||||||
|  |                 "url": "https://api.github.com/repos/sirn-se/phrity-net-stream/zipball/e6ace997168bebcce814c95cd5c78c78663ae49a", | ||||||
|  |                 "reference": "e6ace997168bebcce814c95cd5c78c78663ae49a", | ||||||
|  |                 "shasum": "" | ||||||
|  |             }, | ||||||
|  |             "require": { | ||||||
|  |                 "php": "^8.0", | ||||||
|  |                 "phrity/util-errorhandler": "^1.1", | ||||||
|  |                 "psr/http-factory": "^1.0", | ||||||
|  |                 "psr/http-message": "^1.1 | ^2.0" | ||||||
|  |             }, | ||||||
|  |             "require-dev": { | ||||||
|  |                 "php-coveralls/php-coveralls": "^2.0", | ||||||
|  |                 "phpunit/phpunit": "^9.0 | ^10.0 | ^11.0", | ||||||
|  |                 "phrity/net-uri": "^2.0", | ||||||
|  |                 "squizlabs/php_codesniffer": "^3.5" | ||||||
|  |             }, | ||||||
|  |             "type": "library", | ||||||
|  |             "autoload": { | ||||||
|  |                 "psr-4": { | ||||||
|  |                     "Phrity\\Net\\": "src/" | ||||||
|  |                 } | ||||||
|  |             }, | ||||||
|  |             "notification-url": "https://packagist.org/downloads/", | ||||||
|  |             "license": [ | ||||||
|  |                 "MIT" | ||||||
|  |             ], | ||||||
|  |             "authors": [ | ||||||
|  |                 { | ||||||
|  |                     "name": "Sören Jensen", | ||||||
|  |                     "email": "sirn@sirn.se", | ||||||
|  |                     "homepage": "https://phrity.sirn.se" | ||||||
|  |                 } | ||||||
|  |             ], | ||||||
|  |             "description": "Socket stream classes implementing PSR-7 Stream and PSR-17 StreamFactory", | ||||||
|  |             "homepage": "https://phrity.sirn.se/net-stream", | ||||||
|  |             "keywords": [ | ||||||
|  |                 "Socket", | ||||||
|  |                 "client", | ||||||
|  |                 "psr-17", | ||||||
|  |                 "psr-7", | ||||||
|  |                 "server", | ||||||
|  |                 "stream", | ||||||
|  |                 "stream factory" | ||||||
|  |             ], | ||||||
|  |             "support": { | ||||||
|  |                 "issues": "https://github.com/sirn-se/phrity-net-stream/issues", | ||||||
|  |                 "source": "https://github.com/sirn-se/phrity-net-stream/tree/2.1.2" | ||||||
|  |             }, | ||||||
|  |             "time": "2025-01-09T08:07:34+00:00" | ||||||
|  |         }, | ||||||
|  |         { | ||||||
|  |             "name": "phrity/net-uri", | ||||||
|  |             "version": "2.1.0", | ||||||
|  |             "source": { | ||||||
|  |                 "type": "git", | ||||||
|  |                 "url": "https://github.com/sirn-se/phrity-net-uri.git", | ||||||
|  |                 "reference": "841190135af4fab18135226aaaf99ec5791377ac" | ||||||
|  |             }, | ||||||
|  |             "dist": { | ||||||
|  |                 "type": "zip", | ||||||
|  |                 "url": "https://api.github.com/repos/sirn-se/phrity-net-uri/zipball/841190135af4fab18135226aaaf99ec5791377ac", | ||||||
|  |                 "reference": "841190135af4fab18135226aaaf99ec5791377ac", | ||||||
|  |                 "shasum": "" | ||||||
|  |             }, | ||||||
|  |             "require": { | ||||||
|  |                 "ext-mbstring": "*", | ||||||
|  |                 "php": "^8.0", | ||||||
|  |                 "psr/http-factory": "^1.0", | ||||||
|  |                 "psr/http-message": "^1.1 | ^2.0" | ||||||
|  |             }, | ||||||
|  |             "require-dev": { | ||||||
|  |                 "php-coveralls/php-coveralls": "^2.0", | ||||||
|  |                 "phpunit/phpunit": "^9.0 | ^10.0 | ^11.0", | ||||||
|  |                 "phrity/util-errorhandler": "^1.1", | ||||||
|  |                 "squizlabs/php_codesniffer": "^3.5" | ||||||
|  |             }, | ||||||
|  |             "suggest": { | ||||||
|  |                 "ext-intl": "Enables IDN conversion for non-ASCII domains" | ||||||
|  |             }, | ||||||
|  |             "type": "library", | ||||||
|  |             "autoload": { | ||||||
|  |                 "psr-4": { | ||||||
|  |                     "Phrity\\Net\\": "src/" | ||||||
|  |                 } | ||||||
|  |             }, | ||||||
|  |             "notification-url": "https://packagist.org/downloads/", | ||||||
|  |             "license": [ | ||||||
|  |                 "MIT" | ||||||
|  |             ], | ||||||
|  |             "authors": [ | ||||||
|  |                 { | ||||||
|  |                     "name": "Sören Jensen", | ||||||
|  |                     "email": "sirn@sirn.se", | ||||||
|  |                     "homepage": "https://phrity.sirn.se" | ||||||
|  |                 } | ||||||
|  |             ], | ||||||
|  |             "description": "PSR-7 Uri and PSR-17 UriFactory implementation", | ||||||
|  |             "homepage": "https://phrity.sirn.se/net-uri", | ||||||
|  |             "keywords": [ | ||||||
|  |                 "psr-17", | ||||||
|  |                 "psr-7", | ||||||
|  |                 "uri", | ||||||
|  |                 "uri factory" | ||||||
|  |             ], | ||||||
|  |             "support": { | ||||||
|  |                 "issues": "https://github.com/sirn-se/phrity-net-uri/issues", | ||||||
|  |                 "source": "https://github.com/sirn-se/phrity-net-uri/tree/2.1.0" | ||||||
|  |             }, | ||||||
|  |             "time": "2024-07-08T06:14:09+00:00" | ||||||
|  |         }, | ||||||
|  |         { | ||||||
|  |             "name": "phrity/util-errorhandler", | ||||||
|  |             "version": "1.1.1", | ||||||
|  |             "source": { | ||||||
|  |                 "type": "git", | ||||||
|  |                 "url": "https://github.com/sirn-se/phrity-util-errorhandler.git", | ||||||
|  |                 "reference": "483228156e06673963902b1cc1e6bd9541ab4d5e" | ||||||
|  |             }, | ||||||
|  |             "dist": { | ||||||
|  |                 "type": "zip", | ||||||
|  |                 "url": "https://api.github.com/repos/sirn-se/phrity-util-errorhandler/zipball/483228156e06673963902b1cc1e6bd9541ab4d5e", | ||||||
|  |                 "reference": "483228156e06673963902b1cc1e6bd9541ab4d5e", | ||||||
|  |                 "shasum": "" | ||||||
|  |             }, | ||||||
|  |             "require": { | ||||||
|  |                 "php": "^7.4 | ^8.0" | ||||||
|  |             }, | ||||||
|  |             "require-dev": { | ||||||
|  |                 "php-coveralls/php-coveralls": "^2.0", | ||||||
|  |                 "phpunit/phpunit": "^9.0 | ^10.0 | ^11.0", | ||||||
|  |                 "squizlabs/php_codesniffer": "^3.5" | ||||||
|  |             }, | ||||||
|  |             "type": "library", | ||||||
|  |             "autoload": { | ||||||
|  |                 "psr-4": { | ||||||
|  |                     "Phrity\\Util\\": "src/" | ||||||
|  |                 } | ||||||
|  |             }, | ||||||
|  |             "notification-url": "https://packagist.org/downloads/", | ||||||
|  |             "license": [ | ||||||
|  |                 "MIT" | ||||||
|  |             ], | ||||||
|  |             "authors": [ | ||||||
|  |                 { | ||||||
|  |                     "name": "Sören Jensen", | ||||||
|  |                     "email": "sirn@sirn.se", | ||||||
|  |                     "homepage": "https://phrity.sirn.se" | ||||||
|  |                 } | ||||||
|  |             ], | ||||||
|  |             "description": "Inline error handler; catch and resolve errors for code block.", | ||||||
|  |             "homepage": "https://phrity.sirn.se/util-errorhandler", | ||||||
|  |             "keywords": [ | ||||||
|  |                 "error", | ||||||
|  |                 "warning" | ||||||
|  |             ], | ||||||
|  |             "support": { | ||||||
|  |                 "issues": "https://github.com/sirn-se/phrity-util-errorhandler/issues", | ||||||
|  |                 "source": "https://github.com/sirn-se/phrity-util-errorhandler/tree/1.1.1" | ||||||
|  |             }, | ||||||
|  |             "time": "2024-09-12T06:49:16+00:00" | ||||||
|  |         }, | ||||||
|  |         { | ||||||
|  |             "name": "phrity/websocket", | ||||||
|  |             "version": "3.2.2", | ||||||
|  |             "source": { | ||||||
|  |                 "type": "git", | ||||||
|  |                 "url": "https://github.com/sirn-se/websocket-php.git", | ||||||
|  |                 "reference": "6dceb42be8bfd500806a5508721ae14cf0d4d737" | ||||||
|  |             }, | ||||||
|  |             "dist": { | ||||||
|  |                 "type": "zip", | ||||||
|  |                 "url": "https://api.github.com/repos/sirn-se/websocket-php/zipball/6dceb42be8bfd500806a5508721ae14cf0d4d737", | ||||||
|  |                 "reference": "6dceb42be8bfd500806a5508721ae14cf0d4d737", | ||||||
|  |                 "shasum": "" | ||||||
|  |             }, | ||||||
|  |             "require": { | ||||||
|  |                 "php": "^8.1", | ||||||
|  |                 "phrity/net-stream": "^2.1", | ||||||
|  |                 "phrity/net-uri": "^2.1", | ||||||
|  |                 "psr/http-message": "^1.1 | ^2.0", | ||||||
|  |                 "psr/log": "^1.0 | ^2.0 | ^3.0" | ||||||
|  |             }, | ||||||
|  |             "require-dev": { | ||||||
|  |                 "php-coveralls/php-coveralls": "^2.0", | ||||||
|  |                 "phpstan/phpstan": "^2.0", | ||||||
|  |                 "phpunit/phpunit": "^10.0 | ^11.0", | ||||||
|  |                 "phrity/net-mock": "^2.1", | ||||||
|  |                 "phrity/util-errorhandler": "^1.1", | ||||||
|  |                 "squizlabs/php_codesniffer": "^3.5" | ||||||
|  |             }, | ||||||
|  |             "type": "library", | ||||||
|  |             "autoload": { | ||||||
|  |                 "psr-4": { | ||||||
|  |                     "WebSocket\\": "src/" | ||||||
|  |                 } | ||||||
|  |             }, | ||||||
|  |             "notification-url": "https://packagist.org/downloads/", | ||||||
|  |             "license": [ | ||||||
|  |                 "ISC" | ||||||
|  |             ], | ||||||
|  |             "authors": [ | ||||||
|  |                 { | ||||||
|  |                     "name": "Fredrik Liljegren" | ||||||
|  |                 }, | ||||||
|  |                 { | ||||||
|  |                     "name": "Sören Jensen", | ||||||
|  |                     "email": "sirn@sirn.se", | ||||||
|  |                     "homepage": "https://phrity.sirn.se" | ||||||
|  |                 } | ||||||
|  |             ], | ||||||
|  |             "description": "WebSocket client and server", | ||||||
|  |             "homepage": "https://phrity.sirn.se/websocket", | ||||||
|  |             "keywords": [ | ||||||
|  |                 "client", | ||||||
|  |                 "server", | ||||||
|  |                 "websocket" | ||||||
|  |             ], | ||||||
|  |             "support": { | ||||||
|  |                 "issues": "https://github.com/sirn-se/websocket-php/issues", | ||||||
|  |                 "source": "https://github.com/sirn-se/websocket-php/tree/3.2.2" | ||||||
|  |             }, | ||||||
|  |             "time": "2025-01-10T09:41:26+00:00" | ||||||
|  |         }, | ||||||
|         { |         { | ||||||
|             "name": "psr/clock", |             "name": "psr/clock", | ||||||
|             "version": "1.0.0", |             "version": "1.0.0", | ||||||
|  | @ -769,6 +1048,114 @@ | ||||||
|             }, |             }, | ||||||
|             "time": "2019-01-08T18:20:26+00:00" |             "time": "2019-01-08T18:20:26+00:00" | ||||||
|         }, |         }, | ||||||
|  |         { | ||||||
|  |             "name": "psr/http-factory", | ||||||
|  |             "version": "1.1.0", | ||||||
|  |             "source": { | ||||||
|  |                 "type": "git", | ||||||
|  |                 "url": "https://github.com/php-fig/http-factory.git", | ||||||
|  |                 "reference": "2b4765fddfe3b508ac62f829e852b1501d3f6e8a" | ||||||
|  |             }, | ||||||
|  |             "dist": { | ||||||
|  |                 "type": "zip", | ||||||
|  |                 "url": "https://api.github.com/repos/php-fig/http-factory/zipball/2b4765fddfe3b508ac62f829e852b1501d3f6e8a", | ||||||
|  |                 "reference": "2b4765fddfe3b508ac62f829e852b1501d3f6e8a", | ||||||
|  |                 "shasum": "" | ||||||
|  |             }, | ||||||
|  |             "require": { | ||||||
|  |                 "php": ">=7.1", | ||||||
|  |                 "psr/http-message": "^1.0 || ^2.0" | ||||||
|  |             }, | ||||||
|  |             "type": "library", | ||||||
|  |             "extra": { | ||||||
|  |                 "branch-alias": { | ||||||
|  |                     "dev-master": "1.0.x-dev" | ||||||
|  |                 } | ||||||
|  |             }, | ||||||
|  |             "autoload": { | ||||||
|  |                 "psr-4": { | ||||||
|  |                     "Psr\\Http\\Message\\": "src/" | ||||||
|  |                 } | ||||||
|  |             }, | ||||||
|  |             "notification-url": "https://packagist.org/downloads/", | ||||||
|  |             "license": [ | ||||||
|  |                 "MIT" | ||||||
|  |             ], | ||||||
|  |             "authors": [ | ||||||
|  |                 { | ||||||
|  |                     "name": "PHP-FIG", | ||||||
|  |                     "homepage": "https://www.php-fig.org/" | ||||||
|  |                 } | ||||||
|  |             ], | ||||||
|  |             "description": "PSR-17: Common interfaces for PSR-7 HTTP message factories", | ||||||
|  |             "keywords": [ | ||||||
|  |                 "factory", | ||||||
|  |                 "http", | ||||||
|  |                 "message", | ||||||
|  |                 "psr", | ||||||
|  |                 "psr-17", | ||||||
|  |                 "psr-7", | ||||||
|  |                 "request", | ||||||
|  |                 "response" | ||||||
|  |             ], | ||||||
|  |             "support": { | ||||||
|  |                 "source": "https://github.com/php-fig/http-factory" | ||||||
|  |             }, | ||||||
|  |             "time": "2024-04-15T12:06:14+00:00" | ||||||
|  |         }, | ||||||
|  |         { | ||||||
|  |             "name": "psr/http-message", | ||||||
|  |             "version": "2.0", | ||||||
|  |             "source": { | ||||||
|  |                 "type": "git", | ||||||
|  |                 "url": "https://github.com/php-fig/http-message.git", | ||||||
|  |                 "reference": "402d35bcb92c70c026d1a6a9883f06b2ead23d71" | ||||||
|  |             }, | ||||||
|  |             "dist": { | ||||||
|  |                 "type": "zip", | ||||||
|  |                 "url": "https://api.github.com/repos/php-fig/http-message/zipball/402d35bcb92c70c026d1a6a9883f06b2ead23d71", | ||||||
|  |                 "reference": "402d35bcb92c70c026d1a6a9883f06b2ead23d71", | ||||||
|  |                 "shasum": "" | ||||||
|  |             }, | ||||||
|  |             "require": { | ||||||
|  |                 "php": "^7.2 || ^8.0" | ||||||
|  |             }, | ||||||
|  |             "type": "library", | ||||||
|  |             "extra": { | ||||||
|  |                 "branch-alias": { | ||||||
|  |                     "dev-master": "2.0.x-dev" | ||||||
|  |                 } | ||||||
|  |             }, | ||||||
|  |             "autoload": { | ||||||
|  |                 "psr-4": { | ||||||
|  |                     "Psr\\Http\\Message\\": "src/" | ||||||
|  |                 } | ||||||
|  |             }, | ||||||
|  |             "notification-url": "https://packagist.org/downloads/", | ||||||
|  |             "license": [ | ||||||
|  |                 "MIT" | ||||||
|  |             ], | ||||||
|  |             "authors": [ | ||||||
|  |                 { | ||||||
|  |                     "name": "PHP-FIG", | ||||||
|  |                     "homepage": "https://www.php-fig.org/" | ||||||
|  |                 } | ||||||
|  |             ], | ||||||
|  |             "description": "Common interface for HTTP messages", | ||||||
|  |             "homepage": "https://github.com/php-fig/http-message", | ||||||
|  |             "keywords": [ | ||||||
|  |                 "http", | ||||||
|  |                 "http-message", | ||||||
|  |                 "psr", | ||||||
|  |                 "psr-7", | ||||||
|  |                 "request", | ||||||
|  |                 "response" | ||||||
|  |             ], | ||||||
|  |             "support": { | ||||||
|  |                 "source": "https://github.com/php-fig/http-message/tree/2.0" | ||||||
|  |             }, | ||||||
|  |             "time": "2023-04-04T09:54:51+00:00" | ||||||
|  |         }, | ||||||
|         { |         { | ||||||
|             "name": "psr/log", |             "name": "psr/log", | ||||||
|             "version": "3.0.2", |             "version": "3.0.2", | ||||||
|  | @ -819,6 +1206,153 @@ | ||||||
|             }, |             }, | ||||||
|             "time": "2024-09-11T13:17:53+00:00" |             "time": "2024-09-11T13:17:53+00:00" | ||||||
|         }, |         }, | ||||||
|  |         { | ||||||
|  |             "name": "simplito/bigint-wrapper-php", | ||||||
|  |             "version": "1.0.0", | ||||||
|  |             "source": { | ||||||
|  |                 "type": "git", | ||||||
|  |                 "url": "https://github.com/simplito/bigint-wrapper-php.git", | ||||||
|  |                 "reference": "cf21ec76d33f103add487b3eadbd9f5033a25930" | ||||||
|  |             }, | ||||||
|  |             "dist": { | ||||||
|  |                 "type": "zip", | ||||||
|  |                 "url": "https://api.github.com/repos/simplito/bigint-wrapper-php/zipball/cf21ec76d33f103add487b3eadbd9f5033a25930", | ||||||
|  |                 "reference": "cf21ec76d33f103add487b3eadbd9f5033a25930", | ||||||
|  |                 "shasum": "" | ||||||
|  |             }, | ||||||
|  |             "type": "library", | ||||||
|  |             "autoload": { | ||||||
|  |                 "psr-4": { | ||||||
|  |                     "BI\\": "lib/" | ||||||
|  |                 } | ||||||
|  |             }, | ||||||
|  |             "notification-url": "https://packagist.org/downloads/", | ||||||
|  |             "license": [ | ||||||
|  |                 "MIT" | ||||||
|  |             ], | ||||||
|  |             "authors": [ | ||||||
|  |                 { | ||||||
|  |                     "name": "Simplito Team", | ||||||
|  |                     "email": "s.smyczynski@simplito.com", | ||||||
|  |                     "homepage": "https://simplito.com" | ||||||
|  |                 } | ||||||
|  |             ], | ||||||
|  |             "description": "Common interface for php_gmp and php_bcmath modules", | ||||||
|  |             "support": { | ||||||
|  |                 "issues": "https://github.com/simplito/bigint-wrapper-php/issues", | ||||||
|  |                 "source": "https://github.com/simplito/bigint-wrapper-php/tree/1.0.0" | ||||||
|  |             }, | ||||||
|  |             "time": "2018-02-27T12:38:08+00:00" | ||||||
|  |         }, | ||||||
|  |         { | ||||||
|  |             "name": "simplito/bn-php", | ||||||
|  |             "version": "1.1.4", | ||||||
|  |             "source": { | ||||||
|  |                 "type": "git", | ||||||
|  |                 "url": "https://github.com/simplito/bn-php.git", | ||||||
|  |                 "reference": "83446756a81720eacc2ffb87ff97958431451fd6" | ||||||
|  |             }, | ||||||
|  |             "dist": { | ||||||
|  |                 "type": "zip", | ||||||
|  |                 "url": "https://api.github.com/repos/simplito/bn-php/zipball/83446756a81720eacc2ffb87ff97958431451fd6", | ||||||
|  |                 "reference": "83446756a81720eacc2ffb87ff97958431451fd6", | ||||||
|  |                 "shasum": "" | ||||||
|  |             }, | ||||||
|  |             "require": { | ||||||
|  |                 "simplito/bigint-wrapper-php": "~1.0.0" | ||||||
|  |             }, | ||||||
|  |             "require-dev": { | ||||||
|  |                 "phpunit/phpunit": "*" | ||||||
|  |             }, | ||||||
|  |             "type": "library", | ||||||
|  |             "autoload": { | ||||||
|  |                 "psr-4": { | ||||||
|  |                     "BN\\": "lib/" | ||||||
|  |                 } | ||||||
|  |             }, | ||||||
|  |             "notification-url": "https://packagist.org/downloads/", | ||||||
|  |             "license": [ | ||||||
|  |                 "MIT" | ||||||
|  |             ], | ||||||
|  |             "authors": [ | ||||||
|  |                 { | ||||||
|  |                     "name": "Simplito Team", | ||||||
|  |                     "email": "s.smyczynski@simplito.com", | ||||||
|  |                     "homepage": "https://simplito.com" | ||||||
|  |                 } | ||||||
|  |             ], | ||||||
|  |             "description": "Big number implementation compatible with bn.js", | ||||||
|  |             "support": { | ||||||
|  |                 "issues": "https://github.com/simplito/bn-php/issues", | ||||||
|  |                 "source": "https://github.com/simplito/bn-php/tree/1.1.4" | ||||||
|  |             }, | ||||||
|  |             "time": "2024-01-10T16:16:59+00:00" | ||||||
|  |         }, | ||||||
|  |         { | ||||||
|  |             "name": "simplito/elliptic-php", | ||||||
|  |             "version": "1.0.12", | ||||||
|  |             "source": { | ||||||
|  |                 "type": "git", | ||||||
|  |                 "url": "https://github.com/simplito/elliptic-php.git", | ||||||
|  |                 "reference": "be321666781be2be2c89c79c43ffcac834bc8868" | ||||||
|  |             }, | ||||||
|  |             "dist": { | ||||||
|  |                 "type": "zip", | ||||||
|  |                 "url": "https://api.github.com/repos/simplito/elliptic-php/zipball/be321666781be2be2c89c79c43ffcac834bc8868", | ||||||
|  |                 "reference": "be321666781be2be2c89c79c43ffcac834bc8868", | ||||||
|  |                 "shasum": "" | ||||||
|  |             }, | ||||||
|  |             "require": { | ||||||
|  |                 "ext-gmp": "*", | ||||||
|  |                 "simplito/bn-php": "~1.1.0" | ||||||
|  |             }, | ||||||
|  |             "require-dev": { | ||||||
|  |                 "phpbench/phpbench": "@dev", | ||||||
|  |                 "phpunit/phpunit": "*" | ||||||
|  |             }, | ||||||
|  |             "type": "library", | ||||||
|  |             "autoload": { | ||||||
|  |                 "psr-4": { | ||||||
|  |                     "Elliptic\\": "lib/" | ||||||
|  |                 } | ||||||
|  |             }, | ||||||
|  |             "notification-url": "https://packagist.org/downloads/", | ||||||
|  |             "license": [ | ||||||
|  |                 "MIT" | ||||||
|  |             ], | ||||||
|  |             "authors": [ | ||||||
|  |                 { | ||||||
|  |                     "name": "Simplito Team", | ||||||
|  |                     "email": "s.smyczynski@simplito.com", | ||||||
|  |                     "homepage": "https://simplito.com" | ||||||
|  |                 } | ||||||
|  |             ], | ||||||
|  |             "description": "Fast elliptic curve cryptography", | ||||||
|  |             "homepage": "https://github.com/simplito/elliptic-php", | ||||||
|  |             "keywords": [ | ||||||
|  |                 "Curve25519", | ||||||
|  |                 "ECDSA", | ||||||
|  |                 "Ed25519", | ||||||
|  |                 "EdDSA", | ||||||
|  |                 "cryptography", | ||||||
|  |                 "curve", | ||||||
|  |                 "curve25519-weier", | ||||||
|  |                 "ecc", | ||||||
|  |                 "ecdh", | ||||||
|  |                 "elliptic", | ||||||
|  |                 "nistp192", | ||||||
|  |                 "nistp224", | ||||||
|  |                 "nistp256", | ||||||
|  |                 "nistp384", | ||||||
|  |                 "nistp521", | ||||||
|  |                 "secp256k1" | ||||||
|  |             ], | ||||||
|  |             "support": { | ||||||
|  |                 "issues": "https://github.com/simplito/elliptic-php/issues", | ||||||
|  |                 "source": "https://github.com/simplito/elliptic-php/tree/1.0.12" | ||||||
|  |             }, | ||||||
|  |             "time": "2024-01-09T14:57:04+00:00" | ||||||
|  |         }, | ||||||
|         { |         { | ||||||
|             "name": "spomky-labs/cbor-php", |             "name": "spomky-labs/cbor-php", | ||||||
|             "version": "3.1.0", |             "version": "3.1.0", | ||||||
|  | @ -1011,6 +1545,73 @@ | ||||||
|             ], |             ], | ||||||
|             "time": "2025-01-03T09:35:48+00:00" |             "time": "2025-01-03T09:35:48+00:00" | ||||||
|         }, |         }, | ||||||
|  |         { | ||||||
|  |             "name": "swentel/nostr-php", | ||||||
|  |             "version": "1.5.3", | ||||||
|  |             "source": { | ||||||
|  |                 "type": "git", | ||||||
|  |                 "url": "https://github.com/nostrver-se/nostr-php.git", | ||||||
|  |                 "reference": "6ef1e05da3845e352593c7d119fc00e716b3321f" | ||||||
|  |             }, | ||||||
|  |             "dist": { | ||||||
|  |                 "type": "zip", | ||||||
|  |                 "url": "https://api.github.com/repos/nostrver-se/nostr-php/zipball/6ef1e05da3845e352593c7d119fc00e716b3321f", | ||||||
|  |                 "reference": "6ef1e05da3845e352593c7d119fc00e716b3321f", | ||||||
|  |                 "shasum": "" | ||||||
|  |             }, | ||||||
|  |             "require": { | ||||||
|  |                 "bitwasp/bech32": "^0.0.1", | ||||||
|  |                 "ext-gmp": "*", | ||||||
|  |                 "ext-xml": "*", | ||||||
|  |                 "php": ">=8.1 <8.5", | ||||||
|  |                 "phrity/websocket": "^3.0", | ||||||
|  |                 "simplito/elliptic-php": "^1.0", | ||||||
|  |                 "uma/phpecc": "^0.2.0" | ||||||
|  |             }, | ||||||
|  |             "require-dev": { | ||||||
|  |                 "friendsofphp/php-cs-fixer": "^3.51", | ||||||
|  |                 "phpunit/phpunit": "^10.5" | ||||||
|  |             }, | ||||||
|  |             "bin": [ | ||||||
|  |                 "bin/nostr-php" | ||||||
|  |             ], | ||||||
|  |             "type": "library", | ||||||
|  |             "autoload": { | ||||||
|  |                 "psr-4": { | ||||||
|  |                     "swentel\\nostr\\": "src/" | ||||||
|  |                 } | ||||||
|  |             }, | ||||||
|  |             "notification-url": "https://packagist.org/downloads/", | ||||||
|  |             "license": [ | ||||||
|  |                 "MIT" | ||||||
|  |             ], | ||||||
|  |             "authors": [ | ||||||
|  |                 { | ||||||
|  |                     "name": "Sebastian Hagens", | ||||||
|  |                     "email": "info@sebastix.nl", | ||||||
|  |                     "homepage": "https://sebastix.nl", | ||||||
|  |                     "role": "Developer & maintainer" | ||||||
|  |                 }, | ||||||
|  |                 { | ||||||
|  |                     "name": "Kristof De Jaeger", | ||||||
|  |                     "homepage": "https://realize.be", | ||||||
|  |                     "role": "Original author" | ||||||
|  |                 } | ||||||
|  |             ], | ||||||
|  |             "description": "Nostr helper library for PHP", | ||||||
|  |             "homepage": "https://nostr-php.dev", | ||||||
|  |             "keywords": [ | ||||||
|  |                 "library", | ||||||
|  |                 "nostr" | ||||||
|  |             ], | ||||||
|  |             "support": { | ||||||
|  |                 "chat": "https://t.me/nostr_php", | ||||||
|  |                 "issue": "https://github.com/swentel/nostr-php/issues", | ||||||
|  |                 "issues": "https://github.com/nostrver-se/nostr-php/issues", | ||||||
|  |                 "source": "https://github.com/nostrver-se/nostr-php/tree/push" | ||||||
|  |             }, | ||||||
|  |             "time": "2025-01-04T21:46:58+00:00" | ||||||
|  |         }, | ||||||
|         { |         { | ||||||
|             "name": "symfony/clock", |             "name": "symfony/clock", | ||||||
|             "version": "v7.2.0", |             "version": "v7.2.0", | ||||||
|  | @ -1783,16 +2384,16 @@ | ||||||
|         }, |         }, | ||||||
|         { |         { | ||||||
|             "name": "symfony/property-access", |             "name": "symfony/property-access", | ||||||
|             "version": "v7.2.0", |             "version": "v7.2.3", | ||||||
|             "source": { |             "source": { | ||||||
|                 "type": "git", |                 "type": "git", | ||||||
|                 "url": "https://github.com/symfony/property-access.git", |                 "url": "https://github.com/symfony/property-access.git", | ||||||
|                 "reference": "3ae42efba01e45aaedecf5c93c8d6a3ab3a82276" |                 "reference": "b28732e315d81fbec787f838034de7d6c9b2b902" | ||||||
|             }, |             }, | ||||||
|             "dist": { |             "dist": { | ||||||
|                 "type": "zip", |                 "type": "zip", | ||||||
|                 "url": "https://api.github.com/repos/symfony/property-access/zipball/3ae42efba01e45aaedecf5c93c8d6a3ab3a82276", |                 "url": "https://api.github.com/repos/symfony/property-access/zipball/b28732e315d81fbec787f838034de7d6c9b2b902", | ||||||
|                 "reference": "3ae42efba01e45aaedecf5c93c8d6a3ab3a82276", |                 "reference": "b28732e315d81fbec787f838034de7d6c9b2b902", | ||||||
|                 "shasum": "" |                 "shasum": "" | ||||||
|             }, |             }, | ||||||
|             "require": { |             "require": { | ||||||
|  | @ -1839,7 +2440,7 @@ | ||||||
|                 "reflection" |                 "reflection" | ||||||
|             ], |             ], | ||||||
|             "support": { |             "support": { | ||||||
|                 "source": "https://github.com/symfony/property-access/tree/v7.2.0" |                 "source": "https://github.com/symfony/property-access/tree/v7.2.3" | ||||||
|             }, |             }, | ||||||
|             "funding": [ |             "funding": [ | ||||||
|                 { |                 { | ||||||
|  | @ -1855,20 +2456,20 @@ | ||||||
|                     "type": "tidelift" |                     "type": "tidelift" | ||||||
|                 } |                 } | ||||||
|             ], |             ], | ||||||
|             "time": "2024-09-26T12:28:35+00:00" |             "time": "2025-01-17T10:56:55+00:00" | ||||||
|         }, |         }, | ||||||
|         { |         { | ||||||
|             "name": "symfony/property-info", |             "name": "symfony/property-info", | ||||||
|             "version": "v7.2.2", |             "version": "v7.2.3", | ||||||
|             "source": { |             "source": { | ||||||
|                 "type": "git", |                 "type": "git", | ||||||
|                 "url": "https://github.com/symfony/property-info.git", |                 "url": "https://github.com/symfony/property-info.git", | ||||||
|                 "reference": "1dfeb0dac7a99f7b3be42db9ccc299c5a6483fcf" |                 "reference": "dedb118fd588a92f226b390250b384d25f4192fe" | ||||||
|             }, |             }, | ||||||
|             "dist": { |             "dist": { | ||||||
|                 "type": "zip", |                 "type": "zip", | ||||||
|                 "url": "https://api.github.com/repos/symfony/property-info/zipball/1dfeb0dac7a99f7b3be42db9ccc299c5a6483fcf", |                 "url": "https://api.github.com/repos/symfony/property-info/zipball/dedb118fd588a92f226b390250b384d25f4192fe", | ||||||
|                 "reference": "1dfeb0dac7a99f7b3be42db9ccc299c5a6483fcf", |                 "reference": "dedb118fd588a92f226b390250b384d25f4192fe", | ||||||
|                 "shasum": "" |                 "shasum": "" | ||||||
|             }, |             }, | ||||||
|             "require": { |             "require": { | ||||||
|  | @ -1879,7 +2480,9 @@ | ||||||
|             "conflict": { |             "conflict": { | ||||||
|                 "phpdocumentor/reflection-docblock": "<5.2", |                 "phpdocumentor/reflection-docblock": "<5.2", | ||||||
|                 "phpdocumentor/type-resolver": "<1.5.1", |                 "phpdocumentor/type-resolver": "<1.5.1", | ||||||
|                 "symfony/dependency-injection": "<6.4" |                 "symfony/cache": "<6.4", | ||||||
|  |                 "symfony/dependency-injection": "<6.4", | ||||||
|  |                 "symfony/serializer": "<6.4" | ||||||
|             }, |             }, | ||||||
|             "require-dev": { |             "require-dev": { | ||||||
|                 "phpdocumentor/reflection-docblock": "^5.2", |                 "phpdocumentor/reflection-docblock": "^5.2", | ||||||
|  | @ -1922,7 +2525,7 @@ | ||||||
|                 "validator" |                 "validator" | ||||||
|             ], |             ], | ||||||
|             "support": { |             "support": { | ||||||
|                 "source": "https://github.com/symfony/property-info/tree/v7.2.2" |                 "source": "https://github.com/symfony/property-info/tree/v7.2.3" | ||||||
|             }, |             }, | ||||||
|             "funding": [ |             "funding": [ | ||||||
|                 { |                 { | ||||||
|  | @ -1938,20 +2541,20 @@ | ||||||
|                     "type": "tidelift" |                     "type": "tidelift" | ||||||
|                 } |                 } | ||||||
|             ], |             ], | ||||||
|             "time": "2024-12-31T11:04:50+00:00" |             "time": "2025-01-27T11:08:17+00:00" | ||||||
|         }, |         }, | ||||||
|         { |         { | ||||||
|             "name": "symfony/serializer", |             "name": "symfony/serializer", | ||||||
|             "version": "v7.2.0", |             "version": "v7.2.3", | ||||||
|             "source": { |             "source": { | ||||||
|                 "type": "git", |                 "type": "git", | ||||||
|                 "url": "https://github.com/symfony/serializer.git", |                 "url": "https://github.com/symfony/serializer.git", | ||||||
|                 "reference": "3f5ed9f5e6c02e3853109190ba38408f5e1d2dd0" |                 "reference": "320f30beb419ce4f96363ada5e225c41f1ef08ab" | ||||||
|             }, |             }, | ||||||
|             "dist": { |             "dist": { | ||||||
|                 "type": "zip", |                 "type": "zip", | ||||||
|                 "url": "https://api.github.com/repos/symfony/serializer/zipball/3f5ed9f5e6c02e3853109190ba38408f5e1d2dd0", |                 "url": "https://api.github.com/repos/symfony/serializer/zipball/320f30beb419ce4f96363ada5e225c41f1ef08ab", | ||||||
|                 "reference": "3f5ed9f5e6c02e3853109190ba38408f5e1d2dd0", |                 "reference": "320f30beb419ce4f96363ada5e225c41f1ef08ab", | ||||||
|                 "shasum": "" |                 "shasum": "" | ||||||
|             }, |             }, | ||||||
|             "require": { |             "require": { | ||||||
|  | @ -2020,7 +2623,7 @@ | ||||||
|             "description": "Handles serializing and deserializing data structures, including object graphs, into array structures or other formats like XML and JSON.", |             "description": "Handles serializing and deserializing data structures, including object graphs, into array structures or other formats like XML and JSON.", | ||||||
|             "homepage": "https://symfony.com", |             "homepage": "https://symfony.com", | ||||||
|             "support": { |             "support": { | ||||||
|                 "source": "https://github.com/symfony/serializer/tree/v7.2.0" |                 "source": "https://github.com/symfony/serializer/tree/v7.2.3" | ||||||
|             }, |             }, | ||||||
|             "funding": [ |             "funding": [ | ||||||
|                 { |                 { | ||||||
|  | @ -2036,7 +2639,7 @@ | ||||||
|                     "type": "tidelift" |                     "type": "tidelift" | ||||||
|                 } |                 } | ||||||
|             ], |             ], | ||||||
|             "time": "2024-11-25T15:21:05+00:00" |             "time": "2025-01-29T07:13:55+00:00" | ||||||
|         }, |         }, | ||||||
|         { |         { | ||||||
|             "name": "symfony/string", |             "name": "symfony/string", | ||||||
|  | @ -2276,16 +2879,16 @@ | ||||||
|         }, |         }, | ||||||
|         { |         { | ||||||
|             "name": "twig/twig", |             "name": "twig/twig", | ||||||
|             "version": "v3.18.0", |             "version": "v3.19.0", | ||||||
|             "source": { |             "source": { | ||||||
|                 "type": "git", |                 "type": "git", | ||||||
|                 "url": "https://github.com/twigphp/Twig.git", |                 "url": "https://github.com/twigphp/Twig.git", | ||||||
|                 "reference": "acffa88cc2b40dbe42eaf3a5025d6c0d4600cc50" |                 "reference": "d4f8c2b86374f08efc859323dbcd95c590f7124e" | ||||||
|             }, |             }, | ||||||
|             "dist": { |             "dist": { | ||||||
|                 "type": "zip", |                 "type": "zip", | ||||||
|                 "url": "https://api.github.com/repos/twigphp/Twig/zipball/acffa88cc2b40dbe42eaf3a5025d6c0d4600cc50", |                 "url": "https://api.github.com/repos/twigphp/Twig/zipball/d4f8c2b86374f08efc859323dbcd95c590f7124e", | ||||||
|                 "reference": "acffa88cc2b40dbe42eaf3a5025d6c0d4600cc50", |                 "reference": "d4f8c2b86374f08efc859323dbcd95c590f7124e", | ||||||
|                 "shasum": "" |                 "shasum": "" | ||||||
|             }, |             }, | ||||||
|             "require": { |             "require": { | ||||||
|  | @ -2340,7 +2943,7 @@ | ||||||
|             ], |             ], | ||||||
|             "support": { |             "support": { | ||||||
|                 "issues": "https://github.com/twigphp/Twig/issues", |                 "issues": "https://github.com/twigphp/Twig/issues", | ||||||
|                 "source": "https://github.com/twigphp/Twig/tree/v3.18.0" |                 "source": "https://github.com/twigphp/Twig/tree/v3.19.0" | ||||||
|             }, |             }, | ||||||
|             "funding": [ |             "funding": [ | ||||||
|                 { |                 { | ||||||
|  | @ -2352,7 +2955,165 @@ | ||||||
|                     "type": "tidelift" |                     "type": "tidelift" | ||||||
|                 } |                 } | ||||||
|             ], |             ], | ||||||
|             "time": "2024-12-29T10:51:50+00:00" |             "time": "2025-01-29T07:06:14+00:00" | ||||||
|  |         }, | ||||||
|  |         { | ||||||
|  |             "name": "uma/phpasn1", | ||||||
|  |             "version": "v2.5.1", | ||||||
|  |             "source": { | ||||||
|  |                 "type": "git", | ||||||
|  |                 "url": "https://github.com/1ma/PHPASN1.git", | ||||||
|  |                 "reference": "dd805d3157ddc90515ee13562a9bb241c68f4f88" | ||||||
|  |             }, | ||||||
|  |             "dist": { | ||||||
|  |                 "type": "zip", | ||||||
|  |                 "url": "https://api.github.com/repos/1ma/PHPASN1/zipball/dd805d3157ddc90515ee13562a9bb241c68f4f88", | ||||||
|  |                 "reference": "dd805d3157ddc90515ee13562a9bb241c68f4f88", | ||||||
|  |                 "shasum": "" | ||||||
|  |             }, | ||||||
|  |             "require": { | ||||||
|  |                 "php": "^8.1" | ||||||
|  |             }, | ||||||
|  |             "require-dev": { | ||||||
|  |                 "php-coveralls/php-coveralls": "^2.0", | ||||||
|  |                 "phpunit/phpunit": "^9.6" | ||||||
|  |             }, | ||||||
|  |             "suggest": { | ||||||
|  |                 "ext-bcmath": "BCmath is the fallback extension for big integer calculations", | ||||||
|  |                 "ext-curl": "For loading OID information from the web if they have not bee defined statically", | ||||||
|  |                 "ext-gmp": "GMP is the preferred extension for big integer calculations", | ||||||
|  |                 "phpseclib/bcmath_compat": "BCmath polyfill for servers where neither GMP nor BCmath is available" | ||||||
|  |             }, | ||||||
|  |             "type": "library", | ||||||
|  |             "extra": { | ||||||
|  |                 "branch-alias": { | ||||||
|  |                     "dev-master": "2.0.x-dev" | ||||||
|  |                 } | ||||||
|  |             }, | ||||||
|  |             "autoload": { | ||||||
|  |                 "psr-4": { | ||||||
|  |                     "FG\\": "lib/" | ||||||
|  |                 } | ||||||
|  |             }, | ||||||
|  |             "notification-url": "https://packagist.org/downloads/", | ||||||
|  |             "license": [ | ||||||
|  |                 "MIT" | ||||||
|  |             ], | ||||||
|  |             "authors": [ | ||||||
|  |                 { | ||||||
|  |                     "name": "Friedrich Große", | ||||||
|  |                     "email": "friedrich.grosse@gmail.com", | ||||||
|  |                     "homepage": "https://github.com/FGrosse", | ||||||
|  |                     "role": "Author" | ||||||
|  |                 }, | ||||||
|  |                 { | ||||||
|  |                     "name": "All contributors", | ||||||
|  |                     "homepage": "https://github.com/FGrosse/PHPASN1/contributors" | ||||||
|  |                 } | ||||||
|  |             ], | ||||||
|  |             "description": "A PHP Framework that allows you to encode and decode arbitrary ASN.1 structures using the ITU-T X.690 Encoding Rules.", | ||||||
|  |             "homepage": "https://github.com/FGrosse/PHPASN1", | ||||||
|  |             "keywords": [ | ||||||
|  |                 "DER", | ||||||
|  |                 "asn.1", | ||||||
|  |                 "asn1", | ||||||
|  |                 "ber", | ||||||
|  |                 "binary", | ||||||
|  |                 "decoding", | ||||||
|  |                 "encoding", | ||||||
|  |                 "x.509", | ||||||
|  |                 "x.690", | ||||||
|  |                 "x509", | ||||||
|  |                 "x690" | ||||||
|  |             ], | ||||||
|  |             "support": { | ||||||
|  |                 "source": "https://github.com/1ma/PHPASN1/tree/v2.5.1" | ||||||
|  |             }, | ||||||
|  |             "abandoned": "genkgo/php-asn1", | ||||||
|  |             "time": "2024-11-29T17:34:36+00:00" | ||||||
|  |         }, | ||||||
|  |         { | ||||||
|  |             "name": "uma/phpecc", | ||||||
|  |             "version": "v0.2.1", | ||||||
|  |             "source": { | ||||||
|  |                 "type": "git", | ||||||
|  |                 "url": "https://github.com/1ma/phpecc.git", | ||||||
|  |                 "reference": "e269be5eaef099fb46831037d097c647a6da2f69" | ||||||
|  |             }, | ||||||
|  |             "dist": { | ||||||
|  |                 "type": "zip", | ||||||
|  |                 "url": "https://api.github.com/repos/1ma/phpecc/zipball/e269be5eaef099fb46831037d097c647a6da2f69", | ||||||
|  |                 "reference": "e269be5eaef099fb46831037d097c647a6da2f69", | ||||||
|  |                 "shasum": "" | ||||||
|  |             }, | ||||||
|  |             "require": { | ||||||
|  |                 "ext-gmp": "*", | ||||||
|  |                 "php": "^8.0", | ||||||
|  |                 "uma/phpasn1": "^2.5.1" | ||||||
|  |             }, | ||||||
|  |             "require-dev": { | ||||||
|  |                 "phpunit/phpunit": "^9.0", | ||||||
|  |                 "squizlabs/php_codesniffer": "^2.0", | ||||||
|  |                 "symfony/yaml": "^6.4 || ^7.0" | ||||||
|  |             }, | ||||||
|  |             "type": "library", | ||||||
|  |             "autoload": { | ||||||
|  |                 "psr-4": { | ||||||
|  |                     "Mdanter\\Ecc\\": "src/" | ||||||
|  |                 } | ||||||
|  |             }, | ||||||
|  |             "notification-url": "https://packagist.org/downloads/", | ||||||
|  |             "license": [ | ||||||
|  |                 "MIT" | ||||||
|  |             ], | ||||||
|  |             "authors": [ | ||||||
|  |                 { | ||||||
|  |                     "name": "Matyas Danter", | ||||||
|  |                     "homepage": "http://matejdanter.com/", | ||||||
|  |                     "role": "Author" | ||||||
|  |                 }, | ||||||
|  |                 { | ||||||
|  |                     "name": "Thibaud Fabre", | ||||||
|  |                     "email": "thibaud@aztech.io", | ||||||
|  |                     "homepage": "http://aztech.io", | ||||||
|  |                     "role": "Maintainer" | ||||||
|  |                 }, | ||||||
|  |                 { | ||||||
|  |                     "name": "Thomas Kerin", | ||||||
|  |                     "email": "afk11@users.noreply.github.com", | ||||||
|  |                     "role": "Maintainer" | ||||||
|  |                 }, | ||||||
|  |                 { | ||||||
|  |                     "name": "Ology Newswire, Inc", | ||||||
|  |                     "email": "protocol@vpsqr.com", | ||||||
|  |                     "homepage": "https://vpsqr.com/", | ||||||
|  |                     "role": "Maintainer" | ||||||
|  |                 } | ||||||
|  |             ], | ||||||
|  |             "description": "Temporary fork of public-square/phpecc", | ||||||
|  |             "homepage": "https://github.com/1ma/phpecc", | ||||||
|  |             "keywords": [ | ||||||
|  |                 "Diffie", | ||||||
|  |                 "ECDSA", | ||||||
|  |                 "Hellman", | ||||||
|  |                 "curve", | ||||||
|  |                 "ecdh", | ||||||
|  |                 "elliptic", | ||||||
|  |                 "nistp192", | ||||||
|  |                 "nistp224", | ||||||
|  |                 "nistp256", | ||||||
|  |                 "nistp384", | ||||||
|  |                 "nistp521", | ||||||
|  |                 "phpecc", | ||||||
|  |                 "schnorr", | ||||||
|  |                 "secp256k1", | ||||||
|  |                 "secp256r1" | ||||||
|  |             ], | ||||||
|  |             "support": { | ||||||
|  |                 "source": "https://github.com/1ma/phpecc/tree/v0.2.1" | ||||||
|  |             }, | ||||||
|  |             "abandoned": "paragonie/ecc", | ||||||
|  |             "time": "2025-01-21T00:38:50+00:00" | ||||||
|         }, |         }, | ||||||
|         { |         { | ||||||
|             "name": "vlucas/phpdotenv", |             "name": "vlucas/phpdotenv", | ||||||
|  | @ -2522,16 +3283,16 @@ | ||||||
|         }, |         }, | ||||||
|         { |         { | ||||||
|             "name": "web-auth/webauthn-lib", |             "name": "web-auth/webauthn-lib", | ||||||
|             "version": "5.0.1", |             "version": "5.1.1", | ||||||
|             "source": { |             "source": { | ||||||
|                 "type": "git", |                 "type": "git", | ||||||
|                 "url": "https://github.com/web-auth/webauthn-lib.git", |                 "url": "https://github.com/web-auth/webauthn-lib.git", | ||||||
|                 "reference": "2cc8262b885cf01eee3c4c10ca3985bdd2614c97" |                 "reference": "6b95b2b3902d943796c3c2bac2dd14af9d031fc2" | ||||||
|             }, |             }, | ||||||
|             "dist": { |             "dist": { | ||||||
|                 "type": "zip", |                 "type": "zip", | ||||||
|                 "url": "https://api.github.com/repos/web-auth/webauthn-lib/zipball/2cc8262b885cf01eee3c4c10ca3985bdd2614c97", |                 "url": "https://api.github.com/repos/web-auth/webauthn-lib/zipball/6b95b2b3902d943796c3c2bac2dd14af9d031fc2", | ||||||
|                 "reference": "2cc8262b885cf01eee3c4c10ca3985bdd2614c97", |                 "reference": "6b95b2b3902d943796c3c2bac2dd14af9d031fc2", | ||||||
|                 "shasum": "" |                 "shasum": "" | ||||||
|             }, |             }, | ||||||
|             "require": { |             "require": { | ||||||
|  | @ -2592,7 +3353,7 @@ | ||||||
|                 "webauthn" |                 "webauthn" | ||||||
|             ], |             ], | ||||||
|             "support": { |             "support": { | ||||||
|                 "source": "https://github.com/web-auth/webauthn-lib/tree/5.0.1" |                 "source": "https://github.com/web-auth/webauthn-lib/tree/5.1.1" | ||||||
|             }, |             }, | ||||||
|             "funding": [ |             "funding": [ | ||||||
|                 { |                 { | ||||||
|  | @ -2604,7 +3365,7 @@ | ||||||
|                     "type": "patreon" |                     "type": "patreon" | ||||||
|                 } |                 } | ||||||
|             ], |             ], | ||||||
|             "time": "2024-07-20T05:24:59+00:00" |             "time": "2025-01-03T23:01:20+00:00" | ||||||
|         }, |         }, | ||||||
|         { |         { | ||||||
|             "name": "webmozart/assert", |             "name": "webmozart/assert", | ||||||
|  |  | ||||||
|  | @ -21,19 +21,31 @@ Dotenv\Dotenv::createImmutable(__DIR__ . '/../')->load(); | ||||||
| // Start the session
 | // Start the session
 | ||||||
| app::init_db(); | app::init_db(); | ||||||
| use app\models\addresses; | use app\models\addresses; | ||||||
|  | use app\models\cart_items; | ||||||
| use app\models\carts; | use app\models\carts; | ||||||
| use app\models\magic_links; | use app\models\magic_links; | ||||||
|  | use app\models\order_items; | ||||||
| use app\models\orders; | use app\models\orders; | ||||||
| use app\models\products; | use app\models\products; | ||||||
|  | use app\models\quote_items; | ||||||
|  | use app\models\quotes; | ||||||
|  | use app\models\subscriptions; | ||||||
|  | use app\models\transactions; | ||||||
| use app\models\user_addresses; | use app\models\user_addresses; | ||||||
| use app\models\users; | use app\models\users; | ||||||
| 
 | 
 | ||||||
| if (!app::$db->query("SELECT name FROM sqlite_master WHERE type='table' AND name='users'")->fetch()) { | if (!app::$db->query("SELECT name FROM sqlite_master WHERE type='table' AND name='users'")->fetch()) { | ||||||
|     addresses::init(); |     addresses::init(); | ||||||
|  |     cart_items::init(); | ||||||
|     carts::init(); |     carts::init(); | ||||||
|     magic_links::init(); |     magic_links::init(); | ||||||
|  |     order_items::init(); | ||||||
|     orders::init(); |     orders::init(); | ||||||
|     products::init(); |     products::init(); | ||||||
|  |     quote_items::init(); | ||||||
|  |     quotes::init(); | ||||||
|  |     subscriptions::init(); | ||||||
|  |     transactions::init(); | ||||||
|     user_addresses::init(); |     user_addresses::init(); | ||||||
|     users::init(); |     users::init(); | ||||||
| } | } | ||||||
|  | @ -140,6 +152,7 @@ $controller = match ($route) { | ||||||
|     '/account' => account::index($defaults), |     '/account' => account::index($defaults), | ||||||
|     '/account/profile' => account::profile(), |     '/account/profile' => account::profile(), | ||||||
|     '/account/login' => account::login($defaults), |     '/account/login' => account::login($defaults), | ||||||
|  |     '/account/email' => account::email(), | ||||||
|     '/account/logout' => account::logout(), |     '/account/logout' => account::logout(), | ||||||
|     '/magic-link' => magic_link::index(), |     '/magic-link' => magic_link::index(), | ||||||
|     '/account/returns' => account::returns($defaults), |     '/account/returns' => account::returns($defaults), | ||||||
|  |  | ||||||
|  | @ -1,5 +1,6 @@ | ||||||
| <?php | <?php | ||||||
| namespace app; | namespace app; | ||||||
|  | // for email
 | ||||||
| use PHPMailer\PHPMailer\PHPMailer; | use PHPMailer\PHPMailer\PHPMailer; | ||||||
| use PHPMailer\PHPMailer\SMTP; | use PHPMailer\PHPMailer\SMTP; | ||||||
| use PHPMailer\PHPMailer\Exception; | use PHPMailer\PHPMailer\Exception; | ||||||
|  | @ -7,6 +8,7 @@ use PHPMailer\PHPMailer\Exception; | ||||||
| class app | class app | ||||||
| { | { | ||||||
|     public static $db; |     public static $db; | ||||||
|  | 
 | ||||||
|     public static function init_db() |     public static function init_db() | ||||||
|     { |     { | ||||||
|         try { |         try { | ||||||
|  | @ -16,6 +18,7 @@ class app | ||||||
|             die("Database error: " . $e->getMessage()); |             die("Database error: " . $e->getMessage()); | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  | 
 | ||||||
|     public static function send_mail($to, $from, $from_name, $subject, $message, $HTML_message) |     public static function send_mail($to, $from, $from_name, $subject, $message, $HTML_message) | ||||||
|     { |     { | ||||||
|         $mail = new PHPMailer(exceptions: true); |         $mail = new PHPMailer(exceptions: true); | ||||||
|  | @ -40,6 +43,7 @@ class app | ||||||
|         $mail->send(); |         $mail->send(); | ||||||
|         ob_end_clean(); |         ob_end_clean(); | ||||||
|     } |     } | ||||||
|  |      | ||||||
|     public static function sendJson($data, $status = 200) |     public static function sendJson($data, $status = 200) | ||||||
|     { |     { | ||||||
|         http_response_code($status); |         http_response_code($status); | ||||||
|  |  | ||||||
|  | @ -4,6 +4,7 @@ namespace app\controllers; | ||||||
| use app\models\addresses; | use app\models\addresses; | ||||||
| use app\models\users; | use app\models\users; | ||||||
| use app\models\user_addresses; | use app\models\user_addresses; | ||||||
|  | use app\models\magic_links; | ||||||
| 
 | 
 | ||||||
| class account | class account | ||||||
| { | { | ||||||
|  | @ -12,8 +13,8 @@ class account | ||||||
|         if (!isset($_SESSION['user_id'])) { |         if (!isset($_SESSION['user_id'])) { | ||||||
|             header('Location: /account/login'); |             header('Location: /account/login'); | ||||||
|         } |         } | ||||||
|         $email = $_SESSION['user_email']; |         $user_id = $_SESSION['user_id']; | ||||||
|         $user = users::getByEmail($email); |         $user = users::getById($user_id); | ||||||
|         $default_shipping = null;  |         $default_shipping = null;  | ||||||
|         $default_billing = null;  |         $default_billing = null;  | ||||||
|         $ship_addrs = []; |         $ship_addrs = []; | ||||||
|  | @ -60,8 +61,8 @@ class account | ||||||
|             $bill_id = addresses::add( |             $bill_id = addresses::add( | ||||||
|                 $bill['name'], |                 $bill['name'], | ||||||
|                 $bill['company'], |                 $bill['company'], | ||||||
|                 $bill['street'], |                 $bill['addressLine1'], | ||||||
|                 $bill['boxapt'], |                 $bill['addressLine2'], | ||||||
|                 $bill['city'], |                 $bill['city'], | ||||||
|                 $bill['state'], |                 $bill['state'], | ||||||
|                 $bill['zip'], |                 $bill['zip'], | ||||||
|  | @ -76,8 +77,8 @@ class account | ||||||
|             $_SESSION['success'] = "Billing address saved!"; |             $_SESSION['success'] = "Billing address saved!"; | ||||||
|             header('Location: /account/billing'); |             header('Location: /account/billing'); | ||||||
|         } |         } | ||||||
|         $email = $_SESSION['user_email']; |         $user_id = $_SESSION['user_id']; | ||||||
|         $user = users::getByEmail($email); |         $user = users::getById($user_id); | ||||||
|         $default_billing = null;  |         $default_billing = null;  | ||||||
|         $bill_addrs = []; |         $bill_addrs = []; | ||||||
|         $bill_addresses = user_addresses::getBillingByUserId($_SESSION['user_id']); |         $bill_addresses = user_addresses::getBillingByUserId($_SESSION['user_id']); | ||||||
|  | @ -115,8 +116,42 @@ class account | ||||||
|             header('Location: /account'); |             header('Location: /account'); | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  | 
 | ||||||
|  |     public static function email() | ||||||
|  |     { | ||||||
|  |         $user_id = $_SESSION['user_id'] ?? null; | ||||||
|  |         if (empty($user_id)){ | ||||||
|  |             header('Location: /account/login'); | ||||||
|  |         } | ||||||
|  |         if ($_SERVER['REQUEST_METHOD'] == 'POST') {   | ||||||
|  |             $email = $_POST['email'] ?? null; | ||||||
|  |             if (empty($email)) { | ||||||
|  |                 $_SESSION['error'] = "Enter your email to get a login link"; | ||||||
|  |                 header('Location: /account'); | ||||||
|  |                 exit; | ||||||
|  |             } else { | ||||||
|  |                 $token = magic_links::add($email, $user_id); | ||||||
|  |                 users::updateReplaceEmailTokenById($user_id, $token); | ||||||
|  |                 header('Location: /account'); | ||||||
|  |                 exit; | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|     public static function login($defaults) |     public static function login($defaults) | ||||||
|     { |     { | ||||||
|  |         if ($_SERVER['REQUEST_METHOD'] == 'POST') { | ||||||
|  |             $email = $_POST['email'] ?? false; | ||||||
|  |             if (empty($email)) { | ||||||
|  |                 $_SESSION['error'] = "Enter your email to get a login link"; | ||||||
|  |                 header('Location: /account/login'); | ||||||
|  |                 exit; | ||||||
|  |             } else { | ||||||
|  |                 $token = magic_links::add($email, null); | ||||||
|  |                 header('Location: /account/login'); | ||||||
|  |                 exit; | ||||||
|  |             } | ||||||
|  |         } | ||||||
|         if (isset($_SESSION['user_id'])) { |         if (isset($_SESSION['user_id'])) { | ||||||
|             header('Location: /account'); |             header('Location: /account'); | ||||||
|         } |         } | ||||||
|  | @ -131,12 +166,14 @@ class account | ||||||
|             ] |             ] | ||||||
|         ])); |         ])); | ||||||
|     } |     } | ||||||
|  | 
 | ||||||
|     public static function logout() |     public static function logout() | ||||||
|     { |     { | ||||||
|         session_unset(); |         session_unset(); | ||||||
|         session_destroy(); |         session_destroy(); | ||||||
|         header('Location: /'); |         header('Location: /'); | ||||||
|     } |     } | ||||||
|  | 
 | ||||||
|     public static function orders($defaults) |     public static function orders($defaults) | ||||||
|     { |     { | ||||||
|         if (!isset($_SESSION['user_id'])) { |         if (!isset($_SESSION['user_id'])) { | ||||||
|  | @ -185,8 +222,8 @@ class account | ||||||
|             $ship_id = addresses::add( |             $ship_id = addresses::add( | ||||||
|                 $ship['name'], |                 $ship['name'], | ||||||
|                 $ship['company'], |                 $ship['company'], | ||||||
|                 $ship['street'], |                 $ship['addressLine1'], | ||||||
|                 $ship['boxapt'], |                 $ship['addressLine2'], | ||||||
|                 $ship['city'], |                 $ship['city'], | ||||||
|                 $ship['state'], |                 $ship['state'], | ||||||
|                 $ship['zip'], |                 $ship['zip'], | ||||||
|  | @ -201,8 +238,8 @@ class account | ||||||
|             $_SESSION['success'] = "Shipping address saved!"; |             $_SESSION['success'] = "Shipping address saved!"; | ||||||
|             header('Location: /account/shipping'); |             header('Location: /account/shipping'); | ||||||
|         } |         } | ||||||
|         $email = $_SESSION['user_email']; |         $user_id = $_SESSION['user_id']; | ||||||
|         $user = users::getByEmail($email); |         $user = users::getById($user_id); | ||||||
|         $addresses = user_addresses::getShippingByUserId($user['id']); |         $addresses = user_addresses::getShippingByUserId($user['id']); | ||||||
|         $default_shipping = null;  |         $default_shipping = null;  | ||||||
|         $ship_addrs = []; |         $ship_addrs = []; | ||||||
|  | @ -235,27 +272,30 @@ class account | ||||||
|     { |     { | ||||||
|         if ($_SERVER['REQUEST_METHOD'] == 'POST') { |         if ($_SERVER['REQUEST_METHOD'] == 'POST') { | ||||||
|             $email = $_POST['email']; |             $email = $_POST['email']; | ||||||
|  |             if (empty($email)) { | ||||||
|  |                 $_SESSION['error'] = 'Email is required.'; | ||||||
|  |             } | ||||||
|             $existingUser = users::getByEmail($email); |             $existingUser = users::getByEmail($email); | ||||||
|             if ($existingUser) { |             if ($existingUser) { | ||||||
|                 $_SESSION['error'] = 'Email already exists. Please choose a different email or log in.'; |                 $_SESSION['error'] = 'Email already exists. Please choose a different email or log in.'; | ||||||
|  |                 $_SESSION['last_post'] = $_POST; | ||||||
|                 header('Location: /account/signup'); |                 header('Location: /account/signup'); | ||||||
|                 exit; |                 exit; | ||||||
|             } |             } | ||||||
|             if (empty($email)) { |             $useShipping = $_POST['use_shipping'] ?? false; | ||||||
|                 $_SESSION['error'] = 'Email is required.'; |             $ship = addresses::validatePost("shipping"); | ||||||
|             } |             if (!isset($ship['name'])){ | ||||||
|             if (isset($_SESSION['error'])) { |                 $_SESSION['error'] = "Shipping address verification failed. Check your entry for errors."; | ||||||
|  |                 $_SESSION['last_post'] = $_POST; | ||||||
|                 header('Location: /account/signup'); |                 header('Location: /account/signup'); | ||||||
|             } |             } | ||||||
|             $useShipping = $_POST['use_shipping'] ?? false; |             if (!$useShipping) { | ||||||
|             if ($useShipping) { |  | ||||||
|                 $ship = addresses::validatePost("shipping"); |  | ||||||
|             } else { |  | ||||||
|                 $ship = addresses::validatePost("shipping"); |  | ||||||
|                 $bill = addresses::validatePost("billing"); |                 $bill = addresses::validatePost("billing"); | ||||||
|  |                 if (!isset($bill['name'])){ | ||||||
|  |                     $_SESSION['error'] = "Billing address verification failed. Check your entry for errors."; | ||||||
|  |                     $_SESSION['last_post'] = $_POST; | ||||||
|  |                     header('Location: /account/signup'); | ||||||
|                 } |                 } | ||||||
|             if (empty($email)) { |  | ||||||
|                 $_SESSION['error'] = 'Email is required.'; |  | ||||||
|             } |             } | ||||||
|             if (isset($_SESSION['error'])) { |             if (isset($_SESSION['error'])) { | ||||||
|                 $_SESSION['last_post'] = $_POST; |                 $_SESSION['last_post'] = $_POST; | ||||||
|  | @ -264,8 +304,8 @@ class account | ||||||
|             $ship_id = addresses::add( |             $ship_id = addresses::add( | ||||||
|                 $ship['name'], |                 $ship['name'], | ||||||
|                 $ship['company'], |                 $ship['company'], | ||||||
|                 $ship['street'], |                 $ship['addressLine1'], | ||||||
|                 $ship['boxapt'], |                 $ship['addressLine2'], | ||||||
|                 $ship['city'], |                 $ship['city'], | ||||||
|                 $ship['state'], |                 $ship['state'], | ||||||
|                 $ship['zip'], |                 $ship['zip'], | ||||||
|  | @ -278,8 +318,8 @@ class account | ||||||
|                 $bill_id = addresses::add( |                 $bill_id = addresses::add( | ||||||
|                     $bill['name'], |                     $bill['name'], | ||||||
|                     $bill['company'], |                     $bill['company'], | ||||||
|                     $bill['street'], |                     $bill['addressLine1'], | ||||||
|                     $bill['boxapt'], |                     $bill['addressLine2'], | ||||||
|                     $bill['city'], |                     $bill['city'], | ||||||
|                     $bill['state'], |                     $bill['state'], | ||||||
|                     $bill['zip'], |                     $bill['zip'], | ||||||
|  |  | ||||||
|  | @ -8,60 +8,45 @@ class magic_link | ||||||
| { | { | ||||||
|     public static function index() |     public static function index() | ||||||
|     { |     { | ||||||
|         $email = $_GET['email'] ?? null; |  | ||||||
|         $token = $_GET['token'] ?? null; |         $token = $_GET['token'] ?? null; | ||||||
|         $signup = $_GET['signup'] ?? null; |         if (!$token) { | ||||||
| 
 |             $_SESSION['error'] = "Invalid or expired link."; | ||||||
|         if (empty($email) && empty($token)) { |  | ||||||
|             $_SESSION['error'] = "Enter your email to get a login link"; |  | ||||||
|             header('Location: /account/login'); |             header('Location: /account/login'); | ||||||
|             exit; |             exit; | ||||||
|         } |         } else { | ||||||
| 
 |             $link = magic_links::validateToken(token: $token); | ||||||
|         if ($email && empty($token) && empty($signup)) { |  | ||||||
|             $link = magic_links::add(email: $email); |  | ||||||
|             $subject = "Your Magic Sign-In Link"; |  | ||||||
|             $message = "Copy and paste the link into your browser: $link"; |  | ||||||
|             $HTML_message = "Click the link to sign in: <a href='$link'>$link</a>"; |  | ||||||
|             app::send_mail(to: $email, from: $_ENV['SMTP_FROM'], from_name: $_ENV['APP_NAME'], subject: $subject, message: $message, HTML_message: $HTML_message); |  | ||||||
|             $_SESSION['success'] = 'Link sent to your email!'; |  | ||||||
|             header('Location: /account/login'); |  | ||||||
|             exit; |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         if ($email && empty($token) && $signup == "1") { |  | ||||||
|             $link = magic_links::add(email: $email); |  | ||||||
|             $subject = "Your Magic Sign-In Link"; |  | ||||||
|             $message = "Copy and paste the link into your browser: $link"; |  | ||||||
|             $HTML_message = "Click the link to sign in: <a href='$link'>$link</a>"; |  | ||||||
|             app::send_mail(to: $email, from: $_ENV['SMTP_FROM'], from_name: $_ENV['APP_NAME'], subject: $subject, message: $message, HTML_message: $HTML_message); |  | ||||||
|             $_SESSION['success'] = 'Account created! Please check your email inbox for the verification link.'; |  | ||||||
|             header('Location: /account/login'); |  | ||||||
|             exit; |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         if ($token && empty($email)) { |  | ||||||
|             $link = magic_links::validate(token: $token); |  | ||||||
| 
 |  | ||||||
|             if (!$link) { |             if (!$link) { | ||||||
|                 $_SESSION['error'] = "Invalid or expired link."; |                 $_SESSION['error'] = "Invalid or expired link."; | ||||||
|                 header('Location: /account/login'); |                 header('Location: /account/login'); | ||||||
|  |                 exit; | ||||||
|             } |             } | ||||||
|             // handle signup vs. login 
 |             $user = $link['user_id'] ? users::getById($link['user_id']) : users::getByEmail($link['email']); | ||||||
|             $user = users::getByEmail($link['email']); |             if ($user) { // user with this email exists, log them in
 | ||||||
|             if ($user) { |  | ||||||
|                 $_SESSION['user_email'] = $link['email']; |                 $_SESSION['user_email'] = $link['email']; | ||||||
|                 $_SESSION['user_id'] = $user['id']; |                 $_SESSION['user_id'] = $user['id']; | ||||||
|                 if (!$user['verified']) { |                 if (!$user['verified']) { | ||||||
|                     users::verify($link['email']); |                     users::verify($link['email']); | ||||||
|                 } |                 } | ||||||
|                 header('Location: /account'); |                 header('Location: /account'); | ||||||
|             } else { |                 exit; | ||||||
|                 // used to pre-fill email signup field
 |             } else { // no users with this email
 | ||||||
|  |                 $user_replacing_email = users::getByReplaceEmailToken($token); | ||||||
|  |                 if ($user_replacing_email) { // user is replacing their email
 | ||||||
|  |                     $user_id = $user_replacing_email['id']; | ||||||
|  |                     users::updateEmailById($user_id, $link['email']); | ||||||
|  |                     $_SESSION['user_email'] = $link['email']; | ||||||
|  |                     $_SESSION['user_id'] = $user_id; | ||||||
|  |                     if (!$user['verified']) { | ||||||
|  |                         users::verify($link['email']); | ||||||
|  |                     } | ||||||
|  |                     header('Location: /account'); | ||||||
|  |                     exit; | ||||||
|  |                 } else { // new user signup
 | ||||||
|                     $_SESSION['user_email'] = $link['email']; |                     $_SESSION['user_email'] = $link['email']; | ||||||
|                     header('Location: /account/signup'); |                     header('Location: /account/signup'); | ||||||
|  |                     exit; | ||||||
|  |                 } | ||||||
|             } |             } | ||||||
|             exit(); |  | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -10,8 +10,8 @@ class addresses | ||||||
|             id INTEGER PRIMARY KEY AUTOINCREMENT, |             id INTEGER PRIMARY KEY AUTOINCREMENT, | ||||||
|             name TEXT NOT NULL, |             name TEXT NOT NULL, | ||||||
|             company TEXT, |             company TEXT, | ||||||
|             street TEXT NOT NULL, |             addressLine1 TEXT NOT NULL, | ||||||
|             boxapt TEXT NOT NULL, |             addressLine2 TEXT NOT NULL, | ||||||
|             city TEXT NOT NULL, |             city TEXT NOT NULL, | ||||||
|             state TEXT NOT NULL, |             state TEXT NOT NULL, | ||||||
|             zip TEXT NOT NULL, |             zip TEXT NOT NULL, | ||||||
|  | @ -25,36 +25,52 @@ class addresses | ||||||
|     { |     { | ||||||
|         $name = $_POST["{$type}_name"]; |         $name = $_POST["{$type}_name"]; | ||||||
|         $company = $_POST["{$type}_company"] ?? null; |         $company = $_POST["{$type}_company"] ?? null; | ||||||
|         $boxapt = $_POST["{$type}_boxapt"] ?? null; |         $addressLine2 = $_POST["{$type}_addressLine2"] ?? null; | ||||||
|         $street = $_POST["{$type}_street"]; |         $addressLine1 = $_POST["{$type}_addressLine1"]; | ||||||
|         $city = $_POST["{$type}_city"]; |         $city = $_POST["{$type}_city"]; | ||||||
|         $state = $_POST["{$type}_state"]; |         $state = $_POST["{$type}_state"]; | ||||||
|         $zip = $_POST["{$type}_zip"]; |         $zip = $_POST["{$type}_zip"]; | ||||||
|         $phone = $_POST["{$type}_phone"]; |         $phone = $_POST["{$type}_phone"]; | ||||||
|         // check all required fields are set
 |         // check all required fields are set
 | ||||||
|         if (empty($name) || empty($street) || empty($city) || empty($state) || empty($zip) || empty($phone)) { |         if (empty($name) || empty($addressLine1) || empty($city) || empty($state) || empty($zip) || empty($phone)) { | ||||||
|             $_SESSION['error'] = "Missing required {$type} information."; |             $_SESSION['error'] = "Missing required {$type} information."; | ||||||
|         } |         } | ||||||
|         // TODO: find a match using postal database and return that
 |         $url = "https://nominatim.openstreetmap.org/search?" . http_build_query([ | ||||||
|  |             "q" => implode(" ", array_filter([$addressLine1, $addressLine2, $city, $state, $zip])), | ||||||
|  |             "format" => "json", | ||||||
|  |             "addressdetails" => 1, | ||||||
|  |             "limit" => 1 | ||||||
|  |         ]); | ||||||
|  |         $ch = curl_init(); | ||||||
|  |         curl_setopt($ch, CURLOPT_URL, $url); | ||||||
|  |         curl_setopt($ch, CURLOPT_RETURNTRANSFER, 1); | ||||||
|  |         curl_setopt($ch, CURLOPT_USERAGENT, "AddressValidator/1.0"); | ||||||
|  |         $response = curl_exec($ch); | ||||||
|  |         curl_close($ch); | ||||||
|  |         $data = json_decode($response, true); | ||||||
|  |         if (empty($data)) { | ||||||
|  |             return ["error" => "Address not found"]; | ||||||
|  |         } | ||||||
|  |         $addressDetails = $data[0]['address']; | ||||||
|         return [ |         return [ | ||||||
|             'name'         => $name, |             'name'         => $name, | ||||||
|             'company'      => $company, |             'company'      => $company, | ||||||
|             'street' => $street, |             "addressLine1" => ($addressDetails['house_number'] ?? '') . ' ' . ($addressDetails['building'] ?? '') . ' ' . ($addressDetails['road'] ?? ''), | ||||||
|             'boxapt' => $boxapt, |             "addressLine2" => $addressLine2, | ||||||
|             'city' => $city, |             "city"         => $addressDetails['city'] ?? $addressDetails['town'] ?? $addressDetails['village'] ?? '', | ||||||
|             'state' => $state, |             "state"        => $addressDetails['state_code'] ?? ($addressDetails['state'] ?? ''), | ||||||
|             'zip' => $zip, |             "zip"          => $addressDetails['postcode'] ?? '', | ||||||
|             'phone'        => $phone |             'phone'        => $phone | ||||||
|         ]; |         ]; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     public static function add($name, $company, $street, $boxapt, $city, $state, $zip, $phone, $billing, $shipping) |     public static function add($name, $company, $addressLine1, $addressLine2, $city, $state, $zip, $phone, $billing, $shipping) | ||||||
|     { |     { | ||||||
|         $query = "INSERT INTO addresses (
 |         $query = "INSERT INTO addresses (
 | ||||||
|             name,  |             name,  | ||||||
|             company,  |             company,  | ||||||
|             street, |             addressLine1, | ||||||
|             boxapt,  |             addressLine2,  | ||||||
|             city,  |             city,  | ||||||
|             state,  |             state,  | ||||||
|             zip, |             zip, | ||||||
|  | @ -64,8 +80,8 @@ class addresses | ||||||
|         ) VALUES ( |         ) VALUES ( | ||||||
|             :name,  |             :name,  | ||||||
|             :company,  |             :company,  | ||||||
|             :street, |             :addressLine1, | ||||||
|             :boxapt, |             :addressLine2, | ||||||
|             :city,  |             :city,  | ||||||
|             :state,  |             :state,  | ||||||
|             :zip, |             :zip, | ||||||
|  | @ -76,8 +92,8 @@ class addresses | ||||||
|         $stmt = app::$db->prepare($query); |         $stmt = app::$db->prepare($query); | ||||||
|         $stmt->bindParam(':name', $name); |         $stmt->bindParam(':name', $name); | ||||||
|         $stmt->bindParam(':company', $company); |         $stmt->bindParam(':company', $company); | ||||||
|         $stmt->bindParam(':street', $street); |         $stmt->bindParam(':addressLine1', $addressLine1); | ||||||
|         $stmt->bindParam(':boxapt', $boxapt); |         $stmt->bindParam(':addressLine2', $addressLine2); | ||||||
|         $stmt->bindParam(':city', $city); |         $stmt->bindParam(':city', $city); | ||||||
|         $stmt->bindParam(':state', $state); |         $stmt->bindParam(':state', $state); | ||||||
|         $stmt->bindParam(':zip', $zip); |         $stmt->bindParam(':zip', $zip); | ||||||
|  |  | ||||||
							
								
								
									
										53
									
								
								src/models/cart_items.php
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										53
									
								
								src/models/cart_items.php
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,53 @@ | ||||||
|  | <?php | ||||||
|  | namespace app\models; | ||||||
|  | 
 | ||||||
|  | use app\app; | ||||||
|  | 
 | ||||||
|  | class cart_items | ||||||
|  | { | ||||||
|  |     public static function init() | ||||||
|  |     { | ||||||
|  |         app::$db->exec("CREATE TABLE IF NOT EXISTS cart_items (
 | ||||||
|  |             id INTEGER PRIMARY KEY AUTOINCREMENT, | ||||||
|  |             cart_id INTEGER NOT NULL, | ||||||
|  |             product_id INTEGER NOT NULL, | ||||||
|  |             quantity INTEGER NOT NULL CHECK(quantity > 0), | ||||||
|  |             added_at DATETIME DEFAULT CURRENT_TIMESTAMP, | ||||||
|  |             FOREIGN KEY (cart_id) REFERENCES carts(id) ON DELETE CASCADE, | ||||||
|  |             FOREIGN KEY (product_id) REFERENCES products(id) | ||||||
|  |         );");
 | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     public static function addItem(int $cartId, int $productId, int $quantity) | ||||||
|  |     { | ||||||
|  |         $stmt = app::$db->prepare("INSERT INTO cart_items (cart_id, product_id, quantity) 
 | ||||||
|  |                                    VALUES (:cart_id, :product_id, :quantity)");
 | ||||||
|  |         $stmt->execute([ | ||||||
|  |             'cart_id' => $cartId, | ||||||
|  |             'product_id' => $productId, | ||||||
|  |             'quantity' => $quantity | ||||||
|  |         ]); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     public static function updateItem(int $cartItemId, int $quantity) | ||||||
|  |     { | ||||||
|  |         $stmt = app::$db->prepare("UPDATE cart_items SET quantity = :quantity WHERE cart_item_id = :cart_item_id"); | ||||||
|  |         $stmt->execute([ | ||||||
|  |             'cart_item_id' => $cartItemId, | ||||||
|  |             'quantity' => $quantity | ||||||
|  |         ]); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     public static function removeItem(int $cartItemId) | ||||||
|  |     { | ||||||
|  |         $stmt = app::$db->prepare("DELETE FROM cart_items WHERE cart_item_id = :cart_item_id"); | ||||||
|  |         $stmt->execute(['cart_item_id' => $cartItemId]); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     public static function getCartItems(int $cartId): array | ||||||
|  |     { | ||||||
|  |         $stmt = app::$db->prepare("SELECT * FROM cart_items WHERE cart_id = :cart_id"); | ||||||
|  |         $stmt->execute(['cart_id' => $cartId]); | ||||||
|  |         return $stmt->fetchAll(); | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | @ -2,20 +2,42 @@ | ||||||
| namespace app\models; | namespace app\models; | ||||||
| 
 | 
 | ||||||
| use app\app; | use app\app; | ||||||
|  | 
 | ||||||
| class carts | class carts | ||||||
| { | { | ||||||
|     public static function init() |     public static function init() | ||||||
|     { |     { | ||||||
|         app::$db->exec("CREATE TABLE IF NOT EXISTS order_items (
 |         app::$db->exec("CREATE TABLE IF NOT EXISTS carts (
 | ||||||
|             order_item_id INTEGER PRIMARY KEY AUTOINCREMENT,  |             id INTEGER PRIMARY KEY AUTOINCREMENT,  | ||||||
|             order_id INTEGER NOT NULL,                      |             user_id INTEGER NOT NULL,  | ||||||
|             product_id INTEGER NOT NULL,                     |             short_id TEXT UNIQUE NOT NULL, | ||||||
|             quantity INTEGER NOT NULL CHECK(quantity > 0),   |             created_at DATETIME DEFAULT CURRENT_TIMESTAMP, | ||||||
|             price REAL NOT NULL CHECK(price >= 0),          |             FOREIGN KEY (user_id) REFERENCES users(id) | ||||||
|             FOREIGN KEY (order_id) REFERENCES orders(order_id),  |  | ||||||
|             FOREIGN KEY (product_id) REFERENCES products(product_id)  |  | ||||||
|         );");
 |         );");
 | ||||||
|     } |     } | ||||||
|  | 
 | ||||||
|  |     public static function addCart(int $userId): string | ||||||
|  |     { | ||||||
|  |         return self::createCart($userId, 'user_cart_id'); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     public static function addSaved(int $userId): string | ||||||
|  |     { | ||||||
|  |         return self::createCart($userId, 'user_saved_for_later_id'); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private static function createCart(int $userId, string $column): string | ||||||
|  |     { | ||||||
|  |         $characters = '123456789ABCDEFGHJKLMNPQRSTUVWXYZ'; | ||||||
|  |         $shortId = ''; | ||||||
|  | 
 | ||||||
|  |         for ($i = 0; $i < 6; $i++) { | ||||||
|  |             $shortId .= $characters[random_int(0, strlen($characters) - 1)]; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         app::$db->prepare("INSERT INTO carts (user_id, short_id) VALUES (:user_id, :short_id)") | ||||||
|  |             ->execute(['user_id' => $userId, 'short_id' => $shortId]); | ||||||
|  | 
 | ||||||
|  |         return $shortId; | ||||||
|  |     } | ||||||
| } | } | ||||||
| 
 |  | ||||||
| 
 |  | ||||||
|  |  | ||||||
|  | @ -8,38 +8,84 @@ class magic_links | ||||||
|     { |     { | ||||||
|         app::$db->exec("CREATE TABLE IF NOT EXISTS magic_links (
 |         app::$db->exec("CREATE TABLE IF NOT EXISTS magic_links (
 | ||||||
|             id INTEGER PRIMARY KEY AUTOINCREMENT, |             id INTEGER PRIMARY KEY AUTOINCREMENT, | ||||||
|  |             user_id INTEGER REFERENCES users(id), | ||||||
|             email TEXT NOT NULL, |             email TEXT NOT NULL, | ||||||
|  |             code TEXT NOT NULL, | ||||||
|             token TEXT NOT NULL, |             token TEXT NOT NULL, | ||||||
|             expires_at DATETIME NOT NULL, |             expires_at DATETIME NOT NULL, | ||||||
|             used BOOLEAN DEFAULT FALSE |             used BOOLEAN DEFAULT FALSE | ||||||
|         )");
 |         )");
 | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     public static function add($email) |     public static function add($email, $user_id) | ||||||
|     { |     { | ||||||
|  |         $code = str_pad(strval(random_int(0, 999999)), 6, "0", STR_PAD_LEFT); | ||||||
|         $token = bin2hex(random_bytes(32)); |         $token = bin2hex(random_bytes(32)); | ||||||
|         $expires_at = date('Y-m-d H:i:s', time() + 60 * 15); |         $expires_at = date('Y-m-d H:i:s', time() + 60 * 15); | ||||||
|         app::$db->query("INSERT INTO magic_links (
 |         $query = "INSERT INTO magic_links (
 | ||||||
|             email, |             email, | ||||||
|  |             user_id, | ||||||
|             token, |             token, | ||||||
|  |             code, | ||||||
|             expires_at |             expires_at | ||||||
|         ) VALUES ( |         ) VALUES ( | ||||||
|             '$email',  |             :email, | ||||||
|             '$token',  |             :user_id, | ||||||
|             '$expires_at' |             :token,  | ||||||
|         )");
 |             :code, | ||||||
|         return $_ENV['APP_HOST'] . "/magic-link?token=" . urlencode($token); |             :expires_at | ||||||
|  |         )";
 | ||||||
|  |         $stmt = app::$db->prepare($query); | ||||||
|  |         $stmt->bindParam(':email', $email); | ||||||
|  |         $stmt->bindParam(':user_id', $user_id); | ||||||
|  |         $stmt->bindParam(':token', $token); | ||||||
|  |         $stmt->bindParam(':code', $code); | ||||||
|  |         $stmt->bindParam(':expires_at', $expires_at); | ||||||
|  |         $stmt->execute(); | ||||||
|  |         $link = $_ENV['APP_HOST'] . "/magic-link?token=" . urlencode($token); | ||||||
|  |         $subject = "Your Magic Sign-In Link"; | ||||||
|  |         $message = "Enter this code into the sign-in form\n$code\n or copy-paste this link into your browser to sign in:\n$link"; | ||||||
|  |         $HTML_message = "Click the link to sign in: <a href='$link'>$link</a> or enter this code:<br/>$code"; | ||||||
|  |         app::send_mail(to: $email, from: $_ENV['SMTP_FROM'], from_name: $_ENV['APP_NAME'], subject: $subject, message: $message, HTML_message: $HTML_message); | ||||||
|  |         $_SESSION['success'] = 'Link sent to your email!'; | ||||||
|  |         return $token; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     public static function validate($token) |     public static function validateToken($token) | ||||||
|     { |     { | ||||||
|         $link = app::$db->query("SELECT * FROM magic_links 
 |         $query = "SELECT * FROM magic_links 
 | ||||||
|         WHERE token = '$token'  |             WHERE token = :token  | ||||||
|             AND used = FALSE  |             AND used = FALSE  | ||||||
|         AND expires_at > datetime('now') |             AND expires_at > datetime('now')";
 | ||||||
|     ")->fetch(\PDO::FETCH_ASSOC);
 |         $stmt = app::$db->prepare($query); | ||||||
|         // void the token once validated
 |         $stmt->bindParam(':token', $token); | ||||||
|         app::$db->query("UPDATE magic_links SET used = TRUE WHERE token = '$token'"); |         $stmt->execute(); | ||||||
|  |         $link = $stmt->fetch(\PDO::FETCH_ASSOC); | ||||||
|  | 
 | ||||||
|  |         $updateQuery = "UPDATE magic_links SET used = TRUE WHERE token = :token"; | ||||||
|  |         $updateStmt = app::$db->prepare($updateQuery); | ||||||
|  |         $updateStmt->bindParam(':token', $token); | ||||||
|  |         $updateStmt->execute(); | ||||||
|  | 
 | ||||||
|  |         return $link; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     public static function validateCode($code) | ||||||
|  |     { | ||||||
|  |         $query = "SELECT * FROM magic_links 
 | ||||||
|  |             WHERE code = :code  | ||||||
|  |             AND used = FALSE  | ||||||
|  |             AND expires_at > datetime('now')";
 | ||||||
|  |         $stmt = app::$db->prepare($query); | ||||||
|  |         $stmt->bindParam(':code', $code); | ||||||
|  |         $stmt->execute(); | ||||||
|  |         $link = $stmt->fetch(\PDO::FETCH_ASSOC); | ||||||
|  | 
 | ||||||
|  |         $updateQuery = "UPDATE magic_links SET used = TRUE WHERE code = :code"; | ||||||
|  |         $updateStmt = app::$db->prepare($updateQuery); | ||||||
|  |         $updateStmt->bindParam(':code', $code); | ||||||
|  |         $updateStmt->execute(); | ||||||
|  | 
 | ||||||
|         return $link; |         return $link; | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
							
								
								
									
										61
									
								
								src/models/order_items.php
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										61
									
								
								src/models/order_items.php
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,61 @@ | ||||||
|  | <?php | ||||||
|  | namespace app\models; | ||||||
|  | 
 | ||||||
|  | use app\app; | ||||||
|  | 
 | ||||||
|  | class order_items | ||||||
|  | { | ||||||
|  |     public static function init() | ||||||
|  |     { | ||||||
|  |         app::$db->exec("CREATE TABLE IF NOT EXISTS order_items (
 | ||||||
|  |             id INTEGER PRIMARY KEY AUTOINCREMENT, | ||||||
|  |             order_id INTEGER NOT NULL, | ||||||
|  |             product_id INTEGER NOT NULL, | ||||||
|  |             quantity INTEGER NOT NULL CHECK(quantity > 0), | ||||||
|  |             added_at DATETIME DEFAULT CURRENT_TIMESTAMP, | ||||||
|  |             FOREIGN KEY (order_id) REFERENCES orders(id) ON DELETE CASCADE, | ||||||
|  |             FOREIGN KEY (product_id) REFERENCES products(id) | ||||||
|  |         );");
 | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     public static function addItem(int $orderId, int $productId, int $quantity) | ||||||
|  |     { | ||||||
|  |         if ($quantity <= 0) { | ||||||
|  |             throw new \InvalidArgumentException('Quantity must be greater than zero.'); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         $stmt = app::$db->prepare("INSERT INTO order_items (order_id, product_id, quantity) 
 | ||||||
|  |                                    VALUES (:order_id, :product_id, :quantity)");
 | ||||||
|  |         $stmt->execute([ | ||||||
|  |             'order_id' => $orderId, | ||||||
|  |             'product_id' => $productId, | ||||||
|  |             'quantity' => $quantity | ||||||
|  |         ]); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     public static function updateItem(int $orderItemId, int $quantity) | ||||||
|  |     { | ||||||
|  |         if ($quantity <= 0) { | ||||||
|  |             throw new \InvalidArgumentException('Quantity must be greater than zero.'); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         $stmt = app::$db->prepare("UPDATE order_items SET quantity = :quantity WHERE order_item_id = :order_item_id"); | ||||||
|  |         $stmt->execute([ | ||||||
|  |             'order_item_id' => $orderItemId, | ||||||
|  |             'quantity' => $quantity | ||||||
|  |         ]); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     public static function removeItem(int $orderItemId) | ||||||
|  |     { | ||||||
|  |         $stmt = app::$db->prepare("DELETE FROM order_items WHERE order_item_id = :order_item_id"); | ||||||
|  |         $stmt->execute(['order_item_id' => $orderItemId]); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     public static function getOrderItems(int $orderId): array | ||||||
|  |     { | ||||||
|  |         $stmt = app::$db->prepare("SELECT * FROM order_items WHERE order_id = :order_id"); | ||||||
|  |         $stmt->execute(['order_id' => $orderId]); | ||||||
|  |         return $stmt->fetchAll(); | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | @ -2,17 +2,72 @@ | ||||||
| namespace app\models; | namespace app\models; | ||||||
| 
 | 
 | ||||||
| use app\app; | use app\app; | ||||||
|  | 
 | ||||||
| class orders | class orders | ||||||
| { | { | ||||||
|  |     const STATUSES = [ | ||||||
|  |         'SHIPPED', 'PENDING', 'HOLD', 'PARTIAL',  | ||||||
|  |         'BACKORDER', 'FAILED', 'CANCELED', 'PROCESSING' | ||||||
|  |     ]; | ||||||
|  | 
 | ||||||
|     public static function init() |     public static function init() | ||||||
|     { |     { | ||||||
|         app::$db->exec("CREATE TABLE IF NOT EXISTS orders (
 |         app::$db->exec("CREATE TABLE IF NOT EXISTS orders (
 | ||||||
|             order_id INTEGER PRIMARY KEY AUTOINCREMENT,  |             id INTEGER PRIMARY KEY AUTOINCREMENT, | ||||||
|             user_id INTEGER NOT NULL, |             user_id INTEGER NOT NULL, | ||||||
|             total_amount REAL NOT NULL CHECK(total_amount >= 0), |             value_sats INTEGER NOT NULL CHECK(value_sats >= 0), | ||||||
|             status TEXT NOT NULL CHECK(status IN ('pending', 'completed', 'cancelled')), |             value_cents INTEGER NOT NULL CHECK(value_cents >= 0), | ||||||
|             created_at DATETIME DEFAULT CURRENT_TIMESTAMP, |             created_at DATETIME DEFAULT CURRENT_TIMESTAMP, | ||||||
|             FOREIGN KEY (user_id) REFERENCES users(user_id) |             status TEXT CHECK(status IN ('" . implode("', '", self::STATUSES) . "')) NOT NULL DEFAULT 'PENDING', | ||||||
|  |             FOREIGN KEY (user_id) REFERENCES users(id) | ||||||
|         );");
 |         );");
 | ||||||
|     } |     } | ||||||
|  | 
 | ||||||
|  |     public static function createOrder(int $userId, int $valueSats, int $valueCents, string $status = 'PENDING'): int | ||||||
|  |     { | ||||||
|  |         self::validateStatus($status); | ||||||
|  | 
 | ||||||
|  |         $stmt = app::$db->prepare("INSERT INTO orders (user_id, value_sats, value_cents, status) 
 | ||||||
|  |                                    VALUES (:user_id, :value_sats, :value_cents, :status)");
 | ||||||
|  |         $stmt->execute([ | ||||||
|  |             'user_id' => $userId, | ||||||
|  |             'value_sats' => $valueSats, | ||||||
|  |             'value_cents' => $valueCents, | ||||||
|  |             'status' => $status | ||||||
|  |         ]); | ||||||
|  | 
 | ||||||
|  |         return app::$db->lastInsertId(); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     public static function updateStatus(int $orderId, string $status) | ||||||
|  |     { | ||||||
|  |         self::validateStatus($status); | ||||||
|  | 
 | ||||||
|  |         $stmt = app::$db->prepare("UPDATE orders SET status = :status WHERE order_id = :order_id"); | ||||||
|  |         $stmt->execute([ | ||||||
|  |             'order_id' => $orderId, | ||||||
|  |             'status' => $status | ||||||
|  |         ]); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     public static function getOrder(int $orderId): array | ||||||
|  |     { | ||||||
|  |         $stmt = app::$db->prepare("SELECT * FROM orders WHERE order_id = :order_id"); | ||||||
|  |         $stmt->execute(['order_id' => $orderId]); | ||||||
|  |         return $stmt->fetch() ?: []; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     public static function getUserOrders(int $userId): array | ||||||
|  |     { | ||||||
|  |         $stmt = app::$db->prepare("SELECT * FROM orders WHERE user_id = :user_id ORDER BY created_at DESC"); | ||||||
|  |         $stmt->execute(['user_id' => $userId]); | ||||||
|  |         return $stmt->fetchAll(); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private static function validateStatus(string $status) | ||||||
|  |     { | ||||||
|  |         if (!in_array($status, self::STATUSES, true)) { | ||||||
|  |             throw new \InvalidArgumentException("Invalid order status: $status"); | ||||||
|  |         } | ||||||
|  |     } | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -2,16 +2,69 @@ | ||||||
| namespace app\models; | namespace app\models; | ||||||
| 
 | 
 | ||||||
| use app\app; | use app\app; | ||||||
|  | 
 | ||||||
| class products | class products | ||||||
| { | { | ||||||
|     public static function init() |     public static function init() | ||||||
|     { |     { | ||||||
|         app::$db->exec("CREATE TABLE IF NOT EXISTS products (
 |         app::$db->exec("CREATE TABLE IF NOT EXISTS products (
 | ||||||
|             product_id INTEGER PRIMARY KEY AUTOINCREMENT, |             id INTEGER PRIMARY KEY AUTOINCREMENT, | ||||||
|             name TEXT NOT NULL,                           |             title TEXT NOT NULL, | ||||||
|             description TEXT,                            |             desc TEXT, | ||||||
|             price REAL NOT NULL CHECK(price >= 0),        |             stock_qty INTEGER NOT NULL DEFAULT 0 CHECK(stock_qty >= 0), | ||||||
|             qty INTEGER NOT NULL DEFAULT 0 CHECK(qty >= 0)  |             specs_json TEXT, | ||||||
|  |             sats_price INTEGER NOT NULL DEFAULT 0 CHECK(sats_price >= 0), | ||||||
|  |             cents_price INTEGER NOT NULL DEFAULT 0 CHECK(cents_price >= 0), | ||||||
|  |             digital BOOLEAN NOT NULL DEFAULT 0, | ||||||
|  |             subscription BOOLEAN NOT NULL DEFAULT 0, | ||||||
|  |             image_url_0 TEXT, | ||||||
|  |             image_url_1 TEXT, | ||||||
|  |             image_url_2 TEXT, | ||||||
|  |             image_url_3 TEXT, | ||||||
|  |             image_url_4 TEXT, | ||||||
|  |             image_url_5 TEXT, | ||||||
|  |             image_url_6 TEXT, | ||||||
|  |             image_url_7 TEXT, | ||||||
|  |             image_url_8 TEXT, | ||||||
|  |             image_url_9 TEXT, | ||||||
|  |             image_url_10 TEXT, | ||||||
|  |             image_url_11 TEXT | ||||||
|         )");
 |         )");
 | ||||||
|     } |     } | ||||||
|  | 
 | ||||||
|  |     public static function add($title, $desc, $stock_qty, $specs_json, $sats_price, $cents_price, $digital, $subscription, $images) | ||||||
|  |     { | ||||||
|  |         $stmt = app::$db->prepare("INSERT INTO products (
 | ||||||
|  |             title, desc, stock_qty, specs_json, sats_price, cents_price, digital, subscription, | ||||||
|  |             image_url_0, image_url_1, image_url_2, image_url_3, image_url_4, image_url_5, | ||||||
|  |             image_url_6, image_url_7, image_url_8, image_url_9, image_url_10, image_url_11 | ||||||
|  |         ) VALUES ( | ||||||
|  |             :title, :desc, :stock_qty, :specs_json, :sats_price, :cents_price, :digital, :subscription, | ||||||
|  |             :image_url_0, :image_url_1, :image_url_2, :image_url_3, :image_url_4, :image_url_5, | ||||||
|  |             :image_url_6, :image_url_7, :image_url_8, :image_url_9, :image_url_10, :image_url_11 | ||||||
|  |         )");
 | ||||||
|  | 
 | ||||||
|  |         $stmt->execute([ | ||||||
|  |             ':title' => $title, | ||||||
|  |             ':desc' => $desc, | ||||||
|  |             ':stock_qty' => $stock_qty, | ||||||
|  |             ':specs_json' => $specs_json, | ||||||
|  |             ':sats_price' => $sats_price, | ||||||
|  |             ':cents_price' => $cents_price, | ||||||
|  |             ':digital' => (int) $digital, | ||||||
|  |             ':subscription' => (int) $subscription, | ||||||
|  |             ':image_url_0' => $images[0] ?? null, | ||||||
|  |             ':image_url_1' => $images[1] ?? null, | ||||||
|  |             ':image_url_2' => $images[2] ?? null, | ||||||
|  |             ':image_url_3' => $images[3] ?? null, | ||||||
|  |             ':image_url_4' => $images[4] ?? null, | ||||||
|  |             ':image_url_5' => $images[5] ?? null, | ||||||
|  |             ':image_url_6' => $images[6] ?? null, | ||||||
|  |             ':image_url_7' => $images[7] ?? null, | ||||||
|  |             ':image_url_8' => $images[8] ?? null, | ||||||
|  |             ':image_url_9' => $images[9] ?? null, | ||||||
|  |             ':image_url_10' => $images[10] ?? null, | ||||||
|  |             ':image_url_11' => $images[11] ?? null | ||||||
|  |         ]); | ||||||
|  |     } | ||||||
| } | } | ||||||
|  |  | ||||||
							
								
								
									
										70
									
								
								src/models/quote_items.php
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										70
									
								
								src/models/quote_items.php
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,70 @@ | ||||||
|  | <?php | ||||||
|  | namespace app\models; | ||||||
|  | 
 | ||||||
|  | use app\app; | ||||||
|  | 
 | ||||||
|  | class quote_items | ||||||
|  | { | ||||||
|  |     public static function init() | ||||||
|  |     { | ||||||
|  |         app::$db->exec("CREATE TABLE IF NOT EXISTS quote_items (
 | ||||||
|  |             id INTEGER PRIMARY KEY AUTOINCREMENT, | ||||||
|  |             quote_id INTEGER NOT NULL, | ||||||
|  |             product_id INTEGER NOT NULL, | ||||||
|  |             quantity INTEGER NOT NULL CHECK(quantity > 0), | ||||||
|  |             price REAL NOT NULL CHECK(price >= 0), | ||||||
|  |             added_at DATETIME DEFAULT CURRENT_TIMESTAMP, | ||||||
|  |             FOREIGN KEY (quote_id) REFERENCES quotes(id) ON DELETE CASCADE, | ||||||
|  |             FOREIGN KEY (product_id) REFERENCES products(id) | ||||||
|  |         );");
 | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     public static function addItem(int $quoteId, int $productId, int $quantity, float $price) | ||||||
|  |     { | ||||||
|  |         if ($quantity <= 0) { | ||||||
|  |             throw new \InvalidArgumentException('Quantity must be greater than zero.'); | ||||||
|  |         } | ||||||
|  |         if ($price < 0) { | ||||||
|  |             throw new \InvalidArgumentException('Price must be non-negative.'); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         $stmt = app::$db->prepare("INSERT INTO quote_items (quote_id, product_id, quantity, price) 
 | ||||||
|  |                                    VALUES (:quote_id, :product_id, :quantity, :price)");
 | ||||||
|  |         $stmt->execute([ | ||||||
|  |             'quote_id' => $quoteId, | ||||||
|  |             'product_id' => $productId, | ||||||
|  |             'quantity' => $quantity, | ||||||
|  |             'price' => $price | ||||||
|  |         ]); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     public static function updateItem(int $quoteItemId, int $quantity, float $price) | ||||||
|  |     { | ||||||
|  |         if ($quantity <= 0) { | ||||||
|  |             throw new \InvalidArgumentException('Quantity must be greater than zero.'); | ||||||
|  |         } | ||||||
|  |         if ($price < 0) { | ||||||
|  |             throw new \InvalidArgumentException('Price must be non-negative.'); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         $stmt = app::$db->prepare("UPDATE quote_items SET quantity = :quantity, price = :price WHERE quote_item_id = :quote_item_id"); | ||||||
|  |         $stmt->execute([ | ||||||
|  |             'quote_item_id' => $quoteItemId, | ||||||
|  |             'quantity' => $quantity, | ||||||
|  |             'price' => $price | ||||||
|  |         ]); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     public static function removeItem(int $quoteItemId) | ||||||
|  |     { | ||||||
|  |         $stmt = app::$db->prepare("DELETE FROM quote_items WHERE quote_item_id = :quote_item_id"); | ||||||
|  |         $stmt->execute(['quote_item_id' => $quoteItemId]); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     public static function getQuoteItems(int $quoteId): array | ||||||
|  |     { | ||||||
|  |         $stmt = app::$db->prepare("SELECT * FROM quote_items WHERE quote_id = :quote_id"); | ||||||
|  |         $stmt->execute(['quote_id' => $quoteId]); | ||||||
|  |         return $stmt->fetchAll(); | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										69
									
								
								src/models/quotes.php
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										69
									
								
								src/models/quotes.php
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,69 @@ | ||||||
|  | <?php | ||||||
|  | namespace app\models; | ||||||
|  | 
 | ||||||
|  | use app\app; | ||||||
|  | 
 | ||||||
|  | class quotes | ||||||
|  | { | ||||||
|  |     private const STATUSES = [ | ||||||
|  |         'DRAFT', 'PUBLISHED', 'SENT', 'PURCHASED',  | ||||||
|  |         'EXPIRED', 'CANCELED' | ||||||
|  |     ]; | ||||||
|  | 
 | ||||||
|  |     public static function init() | ||||||
|  |     { | ||||||
|  |         app::$db->exec("CREATE TABLE IF NOT EXISTS quotes (
 | ||||||
|  |             id INTEGER PRIMARY KEY AUTOINCREMENT, | ||||||
|  |             user_id INTEGER NOT NULL, | ||||||
|  |             created_at DATETIME DEFAULT CURRENT_TIMESTAMP, | ||||||
|  |             status TEXT CHECK(status IN ('" . implode("', '", self::STATUSES) . "')) NOT NULL DEFAULT 'DRAFT', | ||||||
|  |             FOREIGN KEY (user_id) REFERENCES users(id) | ||||||
|  |         );");
 | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     public static function createQuote(int $userId, string $status = 'DRAFT'): int | ||||||
|  |     { | ||||||
|  |         self::validateStatus($status); | ||||||
|  | 
 | ||||||
|  |         $stmt = app::$db->prepare("INSERT INTO quotes (user_id, status) 
 | ||||||
|  |                                    VALUES (:user_id, :status)");
 | ||||||
|  |         $stmt->execute([ | ||||||
|  |             'user_id' => $userId, | ||||||
|  |             'status' => $status | ||||||
|  |         ]); | ||||||
|  | 
 | ||||||
|  |         return app::$db->lastInsertId(); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     public static function updateStatus(int $quoteId, string $status) | ||||||
|  |     { | ||||||
|  |         self::validateStatus($status); | ||||||
|  | 
 | ||||||
|  |         $stmt = app::$db->prepare("UPDATE quotes SET status = :status WHERE quote_id = :quote_id"); | ||||||
|  |         $stmt->execute([ | ||||||
|  |             'quote_id' => $quoteId, | ||||||
|  |             'status' => $status | ||||||
|  |         ]); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     public static function getQuote(int $quoteId): array | ||||||
|  |     { | ||||||
|  |         $stmt = app::$db->prepare("SELECT * FROM quotes WHERE quote_id = :quote_id"); | ||||||
|  |         $stmt->execute(['quote_id' => $quoteId]); | ||||||
|  |         return $stmt->fetch() ?: []; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     public static function getUserQuotes(int $userId): array | ||||||
|  |     { | ||||||
|  |         $stmt = app::$db->prepare("SELECT * FROM quotes WHERE user_id = :user_id ORDER BY created_at DESC"); | ||||||
|  |         $stmt->execute(['user_id' => $userId]); | ||||||
|  |         return $stmt->fetchAll(); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private static function validateStatus(string $status) | ||||||
|  |     { | ||||||
|  |         if (!in_array($status, self::STATUSES, true)) { | ||||||
|  |             throw new \InvalidArgumentException("Invalid quote status: $status"); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										101
									
								
								src/models/subscriptions.php
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										101
									
								
								src/models/subscriptions.php
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,101 @@ | ||||||
|  | <?php | ||||||
|  | namespace app\models; | ||||||
|  | 
 | ||||||
|  | use app\app; | ||||||
|  | 
 | ||||||
|  | class Subscriptions | ||||||
|  | { | ||||||
|  |     const STATES = [ | ||||||
|  |         'TRIAL', 'START', 'RENEWAL' | ||||||
|  |     ]; | ||||||
|  | 
 | ||||||
|  |     const STATUS = [ | ||||||
|  |         'COMPLETED', 'CANCELED' | ||||||
|  |     ]; | ||||||
|  | 
 | ||||||
|  |     public static function init() | ||||||
|  |     { | ||||||
|  |         app::$db->exec("CREATE TABLE IF NOT EXISTS subscriptions (
 | ||||||
|  |             subscription_id INTEGER PRIMARY KEY AUTOINCREMENT, | ||||||
|  |             user_id INTEGER NOT NULL, | ||||||
|  |             product_id INTEGER NOT NULL, | ||||||
|  |             start_date DATETIME NOT NULL, | ||||||
|  |             renews_at DATETIME NOT NULL, | ||||||
|  |             status TEXT CHECK(status IN ('" . implode("', '", self::STATUS) . "')) NOT NULL DEFAULT 'COMPLETED', | ||||||
|  |             state TEXT CHECK(state IN ('" . implode("', '", self::STATES) . "')) NOT NULL DEFAULT 'TRIAL', | ||||||
|  |             invoice_date DATETIME NOT NULL, | ||||||
|  |             FOREIGN KEY (user_id) REFERENCES users(id), | ||||||
|  |             FOREIGN KEY (product_id) REFERENCES products(id) | ||||||
|  |         );");
 | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     public static function createSubscription(int $userId, int $productId, string $state = 'TRIAL', string $status = 'COMPLETED', string $startDate, string $renewAt, string $invoiceDate): int | ||||||
|  |     { | ||||||
|  |         self::validateState($state); | ||||||
|  |         self::validateStatus($status); | ||||||
|  | 
 | ||||||
|  |         $stmt = app::$db->prepare("INSERT INTO subscriptions (user_id, product_id, state, status, start_date, renews_at, invoice_date) 
 | ||||||
|  |                                    VALUES (:user_id, :product_id, :state, :status, :start_date, :renews_at, :invoice_date)");
 | ||||||
|  |         $stmt->execute([ | ||||||
|  |             'user_id' => $userId, | ||||||
|  |             'product_id' => $productId, | ||||||
|  |             'state' => $state, | ||||||
|  |             'status' => $status, | ||||||
|  |             'start_date' => $startDate, | ||||||
|  |             'renews_at' => $renewAt, | ||||||
|  |             'invoice_date' => $invoiceDate | ||||||
|  |         ]); | ||||||
|  | 
 | ||||||
|  |         return app::$db->lastInsertId(); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     public static function updateState(int $subscriptionId, string $state) | ||||||
|  |     { | ||||||
|  |         self::validateState($state); | ||||||
|  | 
 | ||||||
|  |         $stmt = app::$db->prepare("UPDATE subscriptions SET state = :state WHERE subscription_id = :subscription_id"); | ||||||
|  |         $stmt->execute([ | ||||||
|  |             'subscription_id' => $subscriptionId, | ||||||
|  |             'state' => $state | ||||||
|  |         ]); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     public static function updateStatus(int $subscriptionId, string $status) | ||||||
|  |     { | ||||||
|  |         self::validateStatus($status); | ||||||
|  | 
 | ||||||
|  |         $stmt = app::$db->prepare("UPDATE subscriptions SET status = :status WHERE subscription_id = :subscription_id"); | ||||||
|  |         $stmt->execute([ | ||||||
|  |             'subscription_id' => $subscriptionId, | ||||||
|  |             'status' => $status | ||||||
|  |         ]); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     public static function getSubscription(int $subscriptionId): array | ||||||
|  |     { | ||||||
|  |         $stmt = app::$db->prepare("SELECT * FROM subscriptions WHERE subscription_id = :subscription_id"); | ||||||
|  |         $stmt->execute(['subscription_id' => $subscriptionId]); | ||||||
|  |         return $stmt->fetch() ?: []; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     public static function getUserSubscriptions(int $userId): array | ||||||
|  |     { | ||||||
|  |         $stmt = app::$db->prepare("SELECT * FROM subscriptions WHERE user_id = :user_id ORDER BY start_date DESC"); | ||||||
|  |         $stmt->execute(['user_id' => $userId]); | ||||||
|  |         return $stmt->fetchAll(); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private static function validateState(string $state) | ||||||
|  |     { | ||||||
|  |         if (!in_array($state, self::STATES, true)) { | ||||||
|  |             throw new \InvalidArgumentException("Invalid subscription state: $state"); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private static function validateStatus(string $status) | ||||||
|  |     { | ||||||
|  |         if (!in_array($status, self::STATUS, true)) { | ||||||
|  |             throw new \InvalidArgumentException("Invalid subscription status: $status"); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | @ -2,53 +2,89 @@ | ||||||
| namespace app\models; | namespace app\models; | ||||||
| 
 | 
 | ||||||
| use app\app; | use app\app; | ||||||
|  | 
 | ||||||
| class transactions | class transactions | ||||||
| { | { | ||||||
|  |     const TYPES = ['CREDIT', 'REWARD', 'REDEEM', 'REVOKE', 'DEPOSIT']; | ||||||
|  | 
 | ||||||
|     public static function init() |     public static function init() | ||||||
|     { |     { | ||||||
|         app::$db->exec("CREATE TABLE transactions (
 |         $typesList = "'" . implode("', '", self::TYPES) . "'"; | ||||||
|  |         app::$db->exec("CREATE TABLE IF NOT EXISTS transactions (
 | ||||||
|             id INTEGER PRIMARY KEY AUTOINCREMENT, |             id INTEGER PRIMARY KEY AUTOINCREMENT, | ||||||
|             user_id INTEGER NOT NULL, |             user_id INTEGER NOT NULL, | ||||||
|             type TEXT CHECK(transaction_type IN ('credit', 'spend', 'withdraw')) NOT NULL, |             type TEXT CHECK(type IN ($typesList)) NOT NULL, | ||||||
|             cents REAL DEFAULT 0, |             cents INTEGER DEFAULT 0, | ||||||
|             sats REAL DEFAULT 0, |             sats INTEGER DEFAULT 0, | ||||||
|             date TIMESTAMP DEFAULT CURRENT_TIMESTAMP, |             date TIMESTAMP DEFAULT CURRENT_TIMESTAMP, | ||||||
|             FOREIGN KEY (user_id) REFERENCES users(id) |             FOREIGN KEY (user_id) REFERENCES users(id) | ||||||
|         )");
 |         )");
 | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     public static function add($user_id, $transaction_type, $cents, $sats_amount) |     public static function add($user_id, $transaction_type, $cents, $sats) | ||||||
|     { |     { | ||||||
|         $query = "INSERT INTO transactions (
 |         if (!in_array($transaction_type, self::TYPES)) { | ||||||
|             user_id, |             throw new \Exception("Invalid transaction type."); | ||||||
|             type, |         } | ||||||
|             cents, |         if ($cents < 0 || $sats < 0) { | ||||||
|             sats |             throw new \Exception("Amounts must be non-negative integers."); | ||||||
|         ) VALUES ( |         } | ||||||
|             :user_id, |          | ||||||
|             :transaction_type, |         $currentBalance = self::getUserBalance($user_id); | ||||||
|             :cents, |          | ||||||
|             :sats |         if (in_array($transaction_type, ['REDEEM', 'REVOKE']) && ($currentBalance['total_cents'] < $cents || $currentBalance['total_sats'] < $sats)) { | ||||||
|         )";
 |             throw new \Exception("Insufficient funds."); | ||||||
|  |         } | ||||||
|  |          | ||||||
|  |         $query = "INSERT INTO transactions (user_id, type, cents, sats) VALUES (:user_id, :transaction_type, :cents, :sats)"; | ||||||
|         $stmt = app::$db->prepare($query); |         $stmt = app::$db->prepare($query); | ||||||
|         $stmt->bindParam(':user_id', $user_id); |         $stmt->bindParam(':user_id', $user_id); | ||||||
|         $stmt->bindParam(':transaction_type', $transaction_type); |         $stmt->bindParam(':transaction_type', $transaction_type); | ||||||
|         $stmt->bindParam(':cents', $cents); |         $stmt->bindParam(':cents', $cents); | ||||||
|         $stmt->bindParam(':sats', $sats_amount); |         $stmt->bindParam(':sats', $sats); | ||||||
|         $stmt->execute(); |         $stmt->execute(); | ||||||
|  |          | ||||||
|         return app::$db->lastInsertId(); |         return app::$db->lastInsertId(); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     public static function getUserBalance($user_id) |     public static function getUserBalance($user_id) | ||||||
|     { |     { | ||||||
|         $query = "SELECT SUM(cents) AS total_cents, 
 |         $query = "SELECT COALESCE(SUM(cents), 0) AS total_cents, COALESCE(SUM(sats), 0) AS total_sats FROM transactions WHERE user_id = :user_id"; | ||||||
|             SUM(sats) AS total_sats  |  | ||||||
|             FROM transactions  |  | ||||||
|             WHERE user_id = :user_id";
 |  | ||||||
|         $stmt = app::$db->prepare($query); |         $stmt = app::$db->prepare($query); | ||||||
|         $stmt->bindParam(':user_id', $user_id); |         $stmt->bindParam(':user_id', $user_id); | ||||||
|         $stmt->execute(); |         $stmt->execute(); | ||||||
|         $result = $stmt->fetch(); |         return $stmt->fetch(); | ||||||
|         return $result; |     } | ||||||
|  | 
 | ||||||
|  |     public static function getRecent($n) | ||||||
|  |     { | ||||||
|  |         $query = "SELECT * FROM transactions ORDER BY date DESC LIMIT :n"; | ||||||
|  |         $stmt = app::$db->prepare($query); | ||||||
|  |         $stmt->bindParam(':n', $n, \PDO::PARAM_INT); | ||||||
|  |         $stmt->execute(); | ||||||
|  |         return $stmt->fetchAll(); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     public static function getWhales($n, $currency) | ||||||
|  |     { | ||||||
|  |         if (!in_array($currency, ['cents', 'sats'])) { | ||||||
|  |             throw new \Exception("Invalid currency type."); | ||||||
|  |         } | ||||||
|  |         $query = "SELECT user_id, COALESCE(SUM($currency), 0) AS total FROM transactions GROUP BY user_id ORDER BY total DESC LIMIT :n"; | ||||||
|  |         $stmt = app::$db->prepare($query); | ||||||
|  |         $stmt->bindParam(':n', $n, \PDO::PARAM_INT); | ||||||
|  |         $stmt->execute(); | ||||||
|  |         return $stmt->fetchAll(); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     public static function liabilities($currency) | ||||||
|  |     { | ||||||
|  |         if (!in_array($currency, ['cents', 'sats'])) { | ||||||
|  |             throw new \Exception("Invalid currency type."); | ||||||
|  |         } | ||||||
|  |         $query = "SELECT COALESCE(SUM($currency), 0) AS total FROM transactions"; | ||||||
|  |         $stmt = app::$db->prepare($query); | ||||||
|  |         $stmt->execute(); | ||||||
|  |         return $stmt->fetchColumn(); | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -18,19 +18,27 @@ class user_addresses | ||||||
| 
 | 
 | ||||||
|     public static function getShippingByUserId($id) |     public static function getShippingByUserId($id) | ||||||
|     { |     { | ||||||
|         $addrs = app::$db->query("SELECT a.* FROM users u
 |         $query = "SELECT a.* FROM users u
 | ||||||
|           JOIN user_addresses ua ON u.id = ua.user_id |           JOIN user_addresses ua ON u.id = ua.user_id | ||||||
|           JOIN addresses a ON ua.address_id = a.id |           JOIN addresses a ON ua.address_id = a.id | ||||||
|           WHERE u.id = '$id' AND a.shipping = 1")->fetch(\PDO::FETCH_ASSOC);
 |           WHERE u.id = :id AND a.shipping = 1";
 | ||||||
|  |         $stmt = app::$db->prepare($query); | ||||||
|  |         $stmt->bindParam(':id', $id); | ||||||
|  |         $stmt->execute(); | ||||||
|  |         $addrs = $stmt->fetch(\PDO::FETCH_ASSOC); | ||||||
|         return [$addrs]; |         return [$addrs]; | ||||||
|     } |     } | ||||||
|      |      | ||||||
|     public static function getBillingByUserId($id) |     public static function getBillingByUserId($id) | ||||||
|     { |     { | ||||||
|         $addrs = app::$db->query("SELECT a.* FROM users u
 |         $query = "SELECT a.* FROM users u
 | ||||||
|           JOIN user_addresses ua ON u.id = ua.user_id |           JOIN user_addresses ua ON u.id = ua.user_id | ||||||
|           JOIN addresses a ON ua.address_id = a.id |           JOIN addresses a ON ua.address_id = a.id | ||||||
|           WHERE u.id = '$id' AND a.billing = 1")->fetch(\PDO::FETCH_ASSOC);
 |           WHERE u.id = :id AND a.billing = 1";
 | ||||||
|  |         $stmt = app::$db->prepare($query); | ||||||
|  |         $stmt->bindParam(':id', $id); | ||||||
|  |         $stmt->execute(); | ||||||
|  |         $addrs = $stmt->fetch(\PDO::FETCH_ASSOC); | ||||||
|         return [$addrs]; |         return [$addrs]; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -2,6 +2,8 @@ | ||||||
| namespace app\models; | namespace app\models; | ||||||
| 
 | 
 | ||||||
| use app\app; | use app\app; | ||||||
|  | use swentel\nostr\Key\Key; | ||||||
|  | 
 | ||||||
| class users | class users | ||||||
| { | { | ||||||
|     public static function init() |     public static function init() | ||||||
|  | @ -14,15 +16,44 @@ class users | ||||||
|             opt_in_promotional BOOLEAN NOT NULL, |             opt_in_promotional BOOLEAN NOT NULL, | ||||||
|             verified BOOLEAN NOT NULL, |             verified BOOLEAN NOT NULL, | ||||||
|             dark_theme BOOLEAN NOT NULL, |             dark_theme BOOLEAN NOT NULL, | ||||||
|             generated_base58 TEXT UNIQUE, |             nsec TEXT, | ||||||
|  |             npub TEXT NOT NULL, | ||||||
|             attached_lightning_address TEXT, |             attached_lightning_address TEXT, | ||||||
|  |             replace_email_token TEXT, | ||||||
|             name TEXT, |             name TEXT, | ||||||
|             company_name TEXT, |             company_name TEXT, | ||||||
|             company_type TEXT, |             company_type TEXT, | ||||||
|             company_size TEXT, |             company_size TEXT, | ||||||
|             created_at DATETIME DEFAULT CURRENT_TIMESTAMP |             created_at DATETIME DEFAULT CURRENT_TIMESTAMP | ||||||
|         )");
 |         )");
 | ||||||
|         app::$db->exec('CREATE INDEX IF NOT EXISTS idx_user_email ON users (email)'); |     } | ||||||
|  | 
 | ||||||
|  |     public static function updateReplaceEmailTokenById($user_id, $replace_token) | ||||||
|  |     { | ||||||
|  |         $query = "UPDATE users SET replace_email_token = :replace_token WHERE id = :user_id"; | ||||||
|  |         $stmt = app::$db->prepare($query); | ||||||
|  |         $stmt->bindParam(':replace_token', $replace_token); | ||||||
|  |         $stmt->bindParam(':user_id', $user_id); | ||||||
|  |         $stmt->execute(); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     public static function updateEmailById($user_id, $email) | ||||||
|  |     { | ||||||
|  |         $query = "UPDATE users SET email = :email WHERE id = :user_id"; | ||||||
|  |         $stmt = app::$db->prepare($query); | ||||||
|  |         $stmt->bindParam(':email', $email); | ||||||
|  |         $stmt->bindParam(':user_id', $user_id); | ||||||
|  |         $stmt->execute(); | ||||||
|  |         users::updateReplaceEmailTokenById($user_id, null); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     public static function getByReplaceEmailToken($token) | ||||||
|  |     { | ||||||
|  |         $query = "SELECT * FROM users WHERE replace_email_token = :token"; | ||||||
|  |         $stmt = app::$db->prepare($query); | ||||||
|  |         $stmt->bindParam(':token', $token); | ||||||
|  |         $stmt->execute(); | ||||||
|  |         return $stmt->fetch(\PDO::FETCH_ASSOC); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     public static function updateProfileById($user_id, $post) |     public static function updateProfileById($user_id, $post) | ||||||
|  | @ -44,20 +75,29 @@ class users | ||||||
| 
 | 
 | ||||||
|     public static function add($email, $ship_id, $bill_id, $opt_in_promotional, $verified, $dark_theme) |     public static function add($email, $ship_id, $bill_id, $opt_in_promotional, $verified, $dark_theme) | ||||||
|     { |     { | ||||||
|  |         $key = new Key(); | ||||||
|  |         $private_key = $key->generatePrivateKey(); | ||||||
|  |         $public_key  = $key->getPublicKey($private_key); | ||||||
|  |         $npub = $key->convertPublicKeyToBech32($public_key); | ||||||
|  |         $nsec = $key->convertPrivateKeyToBech32($private_key); | ||||||
|         $query = "INSERT INTO users (
 |         $query = "INSERT INTO users (
 | ||||||
|             email,  |             email,  | ||||||
|             shipping_address_id, |             shipping_address_id, | ||||||
|             billing_address_id, |             billing_address_id, | ||||||
|             opt_in_promotional, |             opt_in_promotional, | ||||||
|             verified, |             verified, | ||||||
|             dark_theme |             dark_theme, | ||||||
|  |             nsec, | ||||||
|  |             npub | ||||||
|         ) VALUES ( |         ) VALUES ( | ||||||
|             :email, |             :email, | ||||||
|             :shipping_address_id, |             :shipping_address_id, | ||||||
|             :billing_address_id, |             :billing_address_id, | ||||||
|             :opt_in_promotional, |             :opt_in_promotional, | ||||||
|             :verified, |             :verified, | ||||||
|             :dark_theme |             :dark_theme, | ||||||
|  |             :nsec, | ||||||
|  |             :npub | ||||||
|         )";
 |         )";
 | ||||||
|         $stmt = app::$db->prepare($query); |         $stmt = app::$db->prepare($query); | ||||||
|         $stmt->bindParam(':email', $email); |         $stmt->bindParam(':email', $email); | ||||||
|  | @ -66,18 +106,35 @@ class users | ||||||
|         $stmt->bindParam(':opt_in_promotional', $opt_in_promotional); |         $stmt->bindParam(':opt_in_promotional', $opt_in_promotional); | ||||||
|         $stmt->bindParam(':verified', $verified); |         $stmt->bindParam(':verified', $verified); | ||||||
|         $stmt->bindParam(':dark_theme', $dark_theme); |         $stmt->bindParam(':dark_theme', $dark_theme); | ||||||
|  |         $stmt->bindParam(':nsec', $nsec); | ||||||
|  |         $stmt->bindParam(':npub', $npub); | ||||||
|         $stmt->execute(); |         $stmt->execute(); | ||||||
|         return app::$db->lastInsertId(); |         return app::$db->lastInsertId(); | ||||||
| 
 |  | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     public static function verify($email) |     public static function verify($email) | ||||||
|     { |     { | ||||||
|         app::$db->exec("UPDATE users SET verified = 1 WHERE email = '$email'"); |         $query = "UPDATE users SET verified = 1 WHERE email = :email"; | ||||||
|  |         $stmt = app::$db->prepare($query); | ||||||
|  |         $stmt->bindParam(':email', $email); | ||||||
|  |         $stmt->execute(); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     public static function getById($id) | ||||||
|  |     { | ||||||
|  |         $query = "SELECT * FROM users WHERE id = :id"; | ||||||
|  |         $stmt = app::$db->prepare($query); | ||||||
|  |         $stmt->bindParam(':id', $id); | ||||||
|  |         $stmt->execute(); | ||||||
|  |         return $stmt->fetch(\PDO::FETCH_ASSOC); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     public static function getByEmail($email) |     public static function getByEmail($email) | ||||||
|     { |     { | ||||||
|         return app::$db->query("SELECT * FROM users WHERE email = '$email'")->fetch(\PDO::FETCH_ASSOC); |         $query = "SELECT * FROM users WHERE email = :email"; | ||||||
|  |         $stmt = app::$db->prepare($query); | ||||||
|  |         $stmt->bindParam(':email', $email); | ||||||
|  |         $stmt->execute(); | ||||||
|  |         return $stmt->fetch(\PDO::FETCH_ASSOC); | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -1,32 +1,3 @@ | ||||||
| @tailwind base; | @tailwind base; | ||||||
| @tailwind components; | @tailwind components; | ||||||
| @tailwind utilities; | @tailwind utilities; | ||||||
| /* #Mega Menu Styles |  | ||||||
|   –––––––––––––––––––––––––––––––––––––––––––––––––– */ |  | ||||||
| .mega-menu { |  | ||||||
|   opacity: 0; |  | ||||||
|   visibility: hidden; |  | ||||||
|   z-index: -900; |  | ||||||
|   left: 0; |  | ||||||
|   top: 38px; |  | ||||||
|   position: absolute; |  | ||||||
|   text-align: left; |  | ||||||
|   width: 100%; |  | ||||||
|   transition: all 0.15s linear 0s; |  | ||||||
| } |  | ||||||
| /* #hoverable Class Styles */ |  | ||||||
| .hoverable { |  | ||||||
|   position: static; |  | ||||||
| } |  | ||||||
| .hoverable > a:after { |  | ||||||
|   content: "\25BC"; |  | ||||||
|   font-size: 10px; |  | ||||||
|   padding-left: 6px; |  | ||||||
|   position: relative; |  | ||||||
|   top: -1px; |  | ||||||
| } |  | ||||||
| .hoverable:hover .mega-menu { |  | ||||||
|   opacity: 1; |  | ||||||
|   visibility: visible; |  | ||||||
|   z-index: 900; |  | ||||||
| } |  | ||||||
|  |  | ||||||
|  | @ -17,8 +17,8 @@ | ||||||
|             <div class='flex flex-col'> |             <div class='flex flex-col'> | ||||||
|                 <span>{{ default_billing.name }}</span> |                 <span>{{ default_billing.name }}</span> | ||||||
|                 <span>{{ default_billing.company }}</span> |                 <span>{{ default_billing.company }}</span> | ||||||
|                 <span>{{ default_billing.street }}</span> |                 <span>{{ default_billing.addressLine1 }}</span> | ||||||
|                 <span>{{ default_billing.boxapt }}</span> |                 <span>{{ default_billing.addressLine2 }}</span> | ||||||
|                 <span>{{ default_billing.city }}, {{ default_billing.state }} {{ default_billing.zip }}</span> |                 <span>{{ default_billing.city }}, {{ default_billing.state }} {{ default_billing.zip }}</span> | ||||||
|                 <span>{{ default_billing.phone }}</span> |                 <span>{{ default_billing.phone }}</span> | ||||||
|             </div> |             </div> | ||||||
|  |  | ||||||
|  | @ -40,8 +40,8 @@ | ||||||
|       <div class="flex flex-col gap-1"> |       <div class="flex flex-col gap-1"> | ||||||
|          <h4 class="font-semibold">{{ default_shipping.name }}</h4> |          <h4 class="font-semibold">{{ default_shipping.name }}</h4> | ||||||
|          <h4 class="font-semibold">{{ default_shipping.company }}</h4> |          <h4 class="font-semibold">{{ default_shipping.company }}</h4> | ||||||
|          <h4 class="font-semibold">{{ default_shipping.street }}</h4> |          <h4 class="font-semibold">{{ default_shipping.addressLine1 }}</h4> | ||||||
|          <h4 class="font-semibold">{{ default_billing.boxapt }}</h4> |          <h4 class="font-semibold">{{ default_billing.addressLine2 }}</h4> | ||||||
|          <h4 class="font-semibold">{{ default_shipping.city }}, {{ default_shipping.state }}, {{ default_shipping.zip }}</h4> |          <h4 class="font-semibold">{{ default_shipping.city }}, {{ default_shipping.state }}, {{ default_shipping.zip }}</h4> | ||||||
|          <h4 class="font-semibold">{{ default_shipping.phone }}</h4> |          <h4 class="font-semibold">{{ default_shipping.phone }}</h4> | ||||||
|       </div> |       </div> | ||||||
|  | @ -55,8 +55,8 @@ | ||||||
|       <div class="flex flex-col gap-1"> |       <div class="flex flex-col gap-1"> | ||||||
|          <h4 class="font-semibold">{{ default_billing.name }}</h4> |          <h4 class="font-semibold">{{ default_billing.name }}</h4> | ||||||
|          <h4 class="font-semibold">{{ default_billing.company }}</h4> |          <h4 class="font-semibold">{{ default_billing.company }}</h4> | ||||||
|          <h4 class="font-semibold">{{ default_billing.street }}</h4> |          <h4 class="font-semibold">{{ default_billing.addressLine1 }}</h4> | ||||||
|          <h4 class="font-semibold">{{ default_billing.boxapt }}</h4> |          <h4 class="font-semibold">{{ default_billing.addressLine2 }}</h4> | ||||||
|          <h4 class="font-semibold">{{ default_billing.city }}, {{ default_billing.state }}, {{ default_billing.zip }}</h4> |          <h4 class="font-semibold">{{ default_billing.city }}, {{ default_billing.state }}, {{ default_billing.zip }}</h4> | ||||||
|          <h4 class="font-semibold">{{ default_billing.phone }}</h4> |          <h4 class="font-semibold">{{ default_billing.phone }}</h4> | ||||||
|       </div> |       </div> | ||||||
|  |  | ||||||
|  | @ -5,7 +5,7 @@ | ||||||
|             {% include 'lib/rule.twig' %} |             {% include 'lib/rule.twig' %} | ||||||
|         </div> |         </div> | ||||||
|         {% include 'lib/alert.twig' %} |         {% include 'lib/alert.twig' %} | ||||||
|         <form action="/magic-link" method="get" class="flex flex-col gap-4"> |         <form action="/account/login" method="post" class="flex flex-col gap-4"> | ||||||
|             {% include 'lib/input.twig' with {  |             {% include 'lib/input.twig' with {  | ||||||
|                 type: 'email',  |                 type: 'email',  | ||||||
|                 name: 'email',  |                 name: 'email',  | ||||||
|  |  | ||||||
|  | @ -8,8 +8,8 @@ | ||||||
|         <div class='flex flex-col'> |         <div class='flex flex-col'> | ||||||
|             <span>{{ default_shipping.name }}</span> |             <span>{{ default_shipping.name }}</span> | ||||||
|             <span>{{ default_shipping.company }}</span> |             <span>{{ default_shipping.company }}</span> | ||||||
|             <span>{{ default_shipping.street }}</span> |             <span>{{ default_shipping.addressLine1 }}</span> | ||||||
|             <span>{{ default_shipping.boxapt }}</span> |             <span>{{ default_shipping.addressLine2 }}</span> | ||||||
|             <span>{{ default_shipping.city }}, {{ default_shipping.state }} {{ default_shipping.zip }}</span> |             <span>{{ default_shipping.city }}, {{ default_shipping.state }} {{ default_shipping.zip }}</span> | ||||||
|             <span>{{ default_shipping.phone }}</span> |             <span>{{ default_shipping.phone }}</span> | ||||||
|         </div> |         </div> | ||||||
|  |  | ||||||
|  | @ -34,9 +34,9 @@ | ||||||
|             {% include 'lib/form/address.twig' with { |             {% include 'lib/form/address.twig' with { | ||||||
|                 action: 'shipping', |                 action: 'shipping', | ||||||
|                 name: session.last_post.shipping_name, |                 name: session.last_post.shipping_name, | ||||||
|                 street: session.last_post.shipping_street, |                 addressLine1: session.last_post.shipping_addressLine1, | ||||||
|                 company: session.last_post.shipping_company, |                 company: session.last_post.shipping_company, | ||||||
|                 boxapt: session.last_post.shipping_boxapt, |                 addressLine2: session.last_post.shipping_addressLine2, | ||||||
|                 city: session.last_post.shipping_city, |                 city: session.last_post.shipping_city, | ||||||
|                 state: session.last_post.shipping_state, |                 state: session.last_post.shipping_state, | ||||||
|                 zip: session.last_post.shipping_zip, |                 zip: session.last_post.shipping_zip, | ||||||
|  | @ -51,16 +51,17 @@ | ||||||
|                 </h4> |                 </h4> | ||||||
|                 {% include 'lib/toggle.twig' with {  |                 {% include 'lib/toggle.twig' with {  | ||||||
|                     label: 'Same as shipping',  |                     label: 'Same as shipping',  | ||||||
|                     name: 'use_shipping'  |                     name: 'use_shipping', | ||||||
|  |                     on: true  | ||||||
|                 } %} |                 } %} | ||||||
|             </div> |             </div> | ||||||
|             <div  id="billing-address" style="display: none;"> |             <div  id="billing-address" style="display: none;"> | ||||||
|                 {% include 'lib/form/address.twig' with { |                 {% include 'lib/form/address.twig' with { | ||||||
|                     action: 'billing', |                     action: 'billing', | ||||||
|                     name: session.last_post.billing_name, |                     name: session.last_post.billing_name, | ||||||
|                     street: session.last_post.billing_street, |                     addressLine1: session.last_post.billing_addressLine1, | ||||||
|                     company: session.last_post.billing_company, |                     company: session.last_post.billing_company, | ||||||
|                     boxapt: session.last_post.billing_boxapt, |                     addressLine2: session.last_post.billing_addressLine2, | ||||||
|                     city: session.last_post.billing_city, |                     city: session.last_post.billing_city, | ||||||
|                     state: session.last_post.billing_state, |                     state: session.last_post.billing_state, | ||||||
|                     zip: session.last_post.billing_zip, |                     zip: session.last_post.billing_zip, | ||||||
|  |  | ||||||
|  | @ -1,3 +1,34 @@ | ||||||
|  | <style> | ||||||
|  |     /* #Mega Menu Styles | ||||||
|  |   –––––––––––––––––––––––––––––––––––––––––––––––––– */ | ||||||
|  |     .mega-menu { | ||||||
|  |         opacity: 0; | ||||||
|  |         visibility: hidden; | ||||||
|  |         z-index: -900; | ||||||
|  |         left: 0; | ||||||
|  |         top: 38px; | ||||||
|  |         position: absolute; | ||||||
|  |         text-align: left; | ||||||
|  |         width: 100%; | ||||||
|  |         transition: all 0.15s linear 0s; | ||||||
|  |     } | ||||||
|  |     /* #hoverable Class Styles */ | ||||||
|  |     .hoverable { | ||||||
|  |         position: static; | ||||||
|  |     } | ||||||
|  |     .hoverable > a:after { | ||||||
|  |         content: "\25BC"; | ||||||
|  |         font-size: 10px; | ||||||
|  |         padding-left: 6px; | ||||||
|  |         position: relative; | ||||||
|  |         top: -1px; | ||||||
|  |     } | ||||||
|  |     .hoverable:hover .mega-menu { | ||||||
|  |         opacity: 1; | ||||||
|  |         visibility: visible; | ||||||
|  |         z-index: 900; | ||||||
|  |     } | ||||||
|  | </style> | ||||||
| <header class="flex flex-col items-center w-full gap-3 mb-8"> | <header class="flex flex-col items-center w-full gap-3 mb-8"> | ||||||
|     <div class="{{ colors.header.banner }} py-1 text-sm flex w-full justify-center"> |     <div class="{{ colors.header.banner }} py-1 text-sm flex w-full justify-center"> | ||||||
|         <div class="w-[97%] lg:w-[90%] xl:w-4/5 flex justify-between"> |         <div class="w-[97%] lg:w-[90%] xl:w-4/5 flex justify-between"> | ||||||
|  |  | ||||||
|  | @ -14,16 +14,16 @@ | ||||||
|     } %} |     } %} | ||||||
|     {% include 'lib/input.twig' with {  |     {% include 'lib/input.twig' with {  | ||||||
|         type: 'text',  |         type: 'text',  | ||||||
|         name: action ~ '_street',  |         name: action ~ '_addressLine1',  | ||||||
|         label: 'Street', |         label: 'Address Line 1', | ||||||
|         value: street |         value: addressLine1 | ||||||
|     } %} |     } %} | ||||||
|     {% include 'lib/input.twig' with {  |     {% include 'lib/input.twig' with {  | ||||||
|         type: 'text',  |         type: 'text',  | ||||||
|         name: action ~ '_boxapt',  |         name: action ~ '_addressLine2',  | ||||||
|         label: 'PO Box/Apt#', |         label: 'Address Line 2', | ||||||
|         optional: true, |         optional: true, | ||||||
|         value: boxapt |         value: addressLine2 | ||||||
|     } %} |     } %} | ||||||
|     <div class="flex gap-4"> |     <div class="flex gap-4"> | ||||||
|         {% include 'lib/input.twig' with {  |         {% include 'lib/input.twig' with {  | ||||||
|  |  | ||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue
	
	 count-null
						count-null