From cf14306c2b3f82a81f8d56669a71633b4d4b5fce Mon Sep 17 00:00:00 2001 From: marvin-borner@live.com Date: Mon, 16 Apr 2018 21:09:05 +0200 Subject: Main merge to user management system - files are now at /main/public/ --- main/.github/CONTRIBUTING.md | 76 + main/.github/ISSUE_TEMPLATE.md | 2 + main/.travis.yml | 36 + main/CHANGELOG.md | 553 ++ main/LICENSE.md | 7 + main/README.md | 124 + main/STYLE-GUIDE.md | 44 + main/app/.env.example | 13 + main/app/.htaccess | 15 + main/app/cache/.gitkeep | 0 main/app/defines.php | 52 + main/app/logs/.gitkeep | 0 main/app/sessions/.gitkeep | 0 main/app/sprinkles.example.json | 10 + main/app/sprinkles/ConfigManager/CHANGELOG.md | 27 + main/app/sprinkles/ConfigManager/LICENSE | 21 + main/app/sprinkles/ConfigManager/README.md | 56 + .../app/sprinkles/ConfigManager/asset-bundles.json | 16 + .../ConfigManager/assets/js/ConfigManager.js | 92 + main/app/sprinkles/ConfigManager/composer.json | 26 + .../ConfigManager/locale/en_US/AdminLTE.php | 27 + .../ConfigManager/locale/en_US/ConfigManager.php | 27 + .../ConfigManager/locale/fr_FR/AdminLTE.php | 27 + .../ConfigManager/locale/fr_FR/ConfigManager.php | 27 + .../ConfigManager/routes/ConfigManager.php | 16 + .../ConfigManager/schema/config/AdminLTE.json | 32 + .../ConfigManager/schema/config/site.json | 40 + .../sprinkles/ConfigManager/src/ConfigManager.php | 40 + .../src/Controller/ConfigManagerController.php | 173 + .../src/Database/Migrations/v100/SettingsTable.php | 48 + .../Migrations/v101/SettingsPermissions.php | 70 + .../ConfigManager/src/Database/Models/Config.php | 70 + .../src/ServicesProvider/ServicesProvider.php | 31 + .../ConfigManager/src/Util/ConfigManager.php | 243 + .../templates/pages/ConfigManager.html.twig | 64 + main/app/sprinkles/FormGenerator/.gitignore | 3 + main/app/sprinkles/FormGenerator/CHANGELOG.md | 78 + main/app/sprinkles/FormGenerator/LICENSE | 21 + main/app/sprinkles/FormGenerator/README.md | 352 ++ .../app/sprinkles/FormGenerator/asset-bundles.json | 17 + .../assets/js/widget-formGenerator.js | 348 ++ main/app/sprinkles/FormGenerator/bower.json | 30 + main/app/sprinkles/FormGenerator/composer.json | 25 + .../FormGenerator/locale/en_US/FormGenerator.php | 14 + .../FormGenerator/locale/fr_FR/FormGenerator.php | 14 + .../FormGenerator/routes/FormGenerator.php | 12 + .../src/Controller/FormGeneratorController.php | 26 + .../sprinkles/FormGenerator/src/Element/Alert.php | 33 + .../FormGenerator/src/Element/BaseInput.php | 111 + .../FormGenerator/src/Element/Checkbox.php | 38 + .../sprinkles/FormGenerator/src/Element/Hidden.php | 32 + .../FormGenerator/src/Element/InputInterface.php | 19 + .../sprinkles/FormGenerator/src/Element/Select.php | 41 + .../sprinkles/FormGenerator/src/Element/Text.php | 37 + .../FormGenerator/src/Element/Textarea.php | 38 + main/app/sprinkles/FormGenerator/src/Form.php | 188 + .../FormGenerator/FormGenerator.html.twig | 33 + .../templates/FormGenerator/confirm.html.twig | 28 + .../templates/FormGenerator/macros/alert.html.twig | 5 + .../FormGenerator/macros/checkbox.html.twig | 5 + .../FormGenerator/macros/hidden.html.twig | 3 + .../FormGenerator/macros/select.html.twig | 5 + .../templates/FormGenerator/macros/text.html.twig | 10 + .../FormGenerator/macros/textarea.html.twig | 5 + .../templates/FormGenerator/modal-large.html.twig | 3 + .../templates/FormGenerator/modal.html.twig | 62 + .../templates/FormGenerator/typehead.html.twig | 15 + .../FormGenerator/tests/Unit/FormGeneratorTest.php | 408 ++ .../FormGenerator/tests/Unit/data/bad.json | 12 + .../FormGenerator/tests/Unit/data/good.json | 67 + main/app/sprinkles/account/asset-bundles.json | 79 + .../userfrosting/js/pages/account-settings.js | 29 + .../userfrosting/js/pages/forgot-password.js | 19 + .../assets/userfrosting/js/pages/register.js | 94 + .../userfrosting/js/pages/resend-verification.js | 19 + .../userfrosting/js/pages/set-or-reset-password.js | 19 + .../assets/userfrosting/js/pages/sign-in.js | 39 + main/app/sprinkles/account/bower.json | 28 + main/app/sprinkles/account/composer.json | 24 + main/app/sprinkles/account/config/default.php | 79 + main/app/sprinkles/account/config/production.php | 67 + .../sprinkles/account/factories/Permissions.php | 19 + main/app/sprinkles/account/factories/Roles.php | 18 + main/app/sprinkles/account/factories/Users.php | 23 + main/app/sprinkles/account/locale/ar/messages.php | 176 + main/app/sprinkles/account/locale/ar/validate.php | 18 + .../sprinkles/account/locale/de_DE/messages.php | 188 + .../sprinkles/account/locale/de_DE/validate.php | 21 + .../sprinkles/account/locale/en_US/messages.php | 183 + .../sprinkles/account/locale/en_US/validate.php | 19 + .../sprinkles/account/locale/es_ES/messages.php | 189 + .../sprinkles/account/locale/es_ES/validate.php | 19 + main/app/sprinkles/account/locale/fa/messages.php | 178 + main/app/sprinkles/account/locale/fa/validate.php | 20 + .../sprinkles/account/locale/fr_FR/messages.php | 179 + .../sprinkles/account/locale/fr_FR/validate.php | 18 + .../sprinkles/account/locale/it_IT/messages.php | 186 + .../sprinkles/account/locale/it_IT/validate.php | 21 + .../sprinkles/account/locale/pt_PT/messages.php | 166 + .../sprinkles/account/locale/pt_PT/validate.php | 18 + .../sprinkles/account/locale/ru_RU/messages.php | 183 + .../sprinkles/account/locale/ru_RU/validate.php | 19 + .../sprinkles/account/locale/th_TH/messages.php | 164 + .../sprinkles/account/locale/th_TH/validate.php | 18 + main/app/sprinkles/account/locale/tr/messages.php | 183 + main/app/sprinkles/account/locale/tr/validate.php | 19 + .../sprinkles/account/locale/zh_CN/messages.php | 177 + .../sprinkles/account/locale/zh_CN/validate.php | 19 + main/app/sprinkles/account/routes/routes.php | 59 + .../account/schema/requests/account-settings.yaml | 35 + .../account/schema/requests/account-verify.yaml | 6 + .../account/schema/requests/check-username.yaml | 17 + .../account/schema/requests/deny-password.yaml | 5 + .../account/schema/requests/forgot-password.yaml | 6 + .../sprinkles/account/schema/requests/login.yaml | 19 + .../account/schema/requests/profile-settings.yaml | 24 + .../account/schema/requests/register.yaml | 75 + .../schema/requests/resend-verification.yaml | 6 + .../account/schema/requests/set-password.yaml | 29 + main/app/sprinkles/account/src/Account.php | 20 + .../account/src/Authenticate/AuthGuard.php | 56 + .../account/src/Authenticate/Authenticator.php | 419 ++ .../Exception/AccountDisabledException.php | 21 + .../Exception/AccountInvalidException.php | 21 + .../Exception/AccountNotVerifiedException.php | 21 + .../Exception/AuthCompromisedException.php | 20 + .../Exception/AuthExpiredException.php | 21 + .../Exception/InvalidCredentialsException.php | 21 + .../sprinkles/account/src/Authenticate/Hasher.php | 108 + .../src/Authorize/AccessConditionExpression.php | 139 + .../src/Authorize/AuthorizationException.php | 23 + .../account/src/Authorize/AuthorizationManager.php | 157 + .../src/Authorize/ParserNodeFunctionEvaluator.php | 193 + .../account/src/Bakery/CreateAdminUser.php | 334 ++ .../account/src/Controller/AccountController.php | 1293 +++++ .../Exception/SpammyRequestException.php | 20 + .../Database/Migrations/v400/ActivitiesTable.php | 54 + .../src/Database/Migrations/v400/GroupsTable.php | 82 + .../Migrations/v400/PasswordResetsTable.php | 57 + .../Migrations/v400/PermissionRolesTable.php | 55 + .../Database/Migrations/v400/PermissionsTable.php | 262 + .../Database/Migrations/v400/PersistencesTable.php | 57 + .../Database/Migrations/v400/RoleUsersTable.php | 55 + .../src/Database/Migrations/v400/RolesTable.php | 78 + .../src/Database/Migrations/v400/UsersTable.php | 69 + .../Migrations/v400/VerificationsTable.php | 57 + .../account/src/Database/Models/Activity.php | 86 + .../account/src/Database/Models/Group.php | 69 + .../account/src/Database/Models/PasswordReset.php | 76 + .../account/src/Database/Models/Permission.php | 121 + .../sprinkles/account/src/Database/Models/Role.php | 105 + .../sprinkles/account/src/Database/Models/User.php | 493 ++ .../account/src/Database/Models/Verification.php | 70 + .../Handler/AuthCompromisedExceptionHandler.php | 34 + .../Error/Handler/AuthExpiredExceptionHandler.php | 50 + .../Error/Handler/ForbiddenExceptionHandler.php | 31 + .../app/sprinkles/account/src/Facades/Password.php | 28 + .../src/Log/UserActivityDatabaseHandler.php | 33 + .../account/src/Log/UserActivityProcessor.php | 45 + .../src/Repository/PasswordResetRepository.php | 34 + .../account/src/Repository/TokenRepository.php | 230 + .../src/Repository/VerificationRepository.php | 32 + .../src/ServicesProvider/ServicesProvider.php | 444 ++ .../account/src/Twig/AccountExtension.php | 65 + .../account/src/Util/HashFailedException.php | 21 + main/app/sprinkles/account/src/Util/Util.php | 39 + .../templates/forms/settings-account.html.twig | 37 + .../templates/forms/settings-profile.html.twig | 40 + .../templates/mail/password-reset.html.twig | 22 + .../templates/mail/resend-verification.html.twig | 17 + .../templates/mail/verify-account.html.twig | 21 + .../account/templates/modals/tos.html.twig | 16 + .../templates/navigation/main-nav.html.twig | 13 + .../templates/navigation/user-card.html.twig | 33 + .../templates/pages/account-settings.html.twig | 45 + .../templates/pages/error/compromised.html.twig | 11 + .../templates/pages/forgot-password.html.twig | 46 + .../account/templates/pages/register.html.twig | 105 + .../templates/pages/resend-verification.html.twig | 46 + .../templates/pages/reset-password.html.twig | 56 + .../account/templates/pages/set-password.html.twig | 55 + .../account/templates/pages/sign-in.html.twig | 84 + .../sprinkles/account/tests/Unit/FactoriesTest.php | 30 + .../sprinkles/account/tests/Unit/HasherTest.php | 71 + main/app/sprinkles/admin/asset-bundles.json | 170 + .../assets/userfrosting/css/tablesorter-custom.css | 30 + .../assets/userfrosting/js/pages/activities.js | 16 + .../assets/userfrosting/js/pages/dashboard.js | 49 + .../admin/assets/userfrosting/js/pages/group.js | 24 + .../admin/assets/userfrosting/js/pages/groups.js | 24 + .../assets/userfrosting/js/pages/permission.js | 20 + .../assets/userfrosting/js/pages/permissions.js | 16 + .../admin/assets/userfrosting/js/pages/role.js | 23 + .../admin/assets/userfrosting/js/pages/roles.js | 24 + .../admin/assets/userfrosting/js/pages/user.js | 25 + .../admin/assets/userfrosting/js/pages/users.js | 24 + .../admin/assets/userfrosting/js/widgets/groups.js | 111 + .../admin/assets/userfrosting/js/widgets/roles.js | 148 + .../admin/assets/userfrosting/js/widgets/users.js | 277 + main/app/sprinkles/admin/composer.json | 22 + main/app/sprinkles/admin/locale/ar/messages.php | 135 + main/app/sprinkles/admin/locale/de_DE/messages.php | 161 + main/app/sprinkles/admin/locale/en_US/messages.php | 160 + main/app/sprinkles/admin/locale/es_ES/messages.php | 164 + main/app/sprinkles/admin/locale/fa/messages.php | 158 + main/app/sprinkles/admin/locale/fr_FR/messages.php | 147 + main/app/sprinkles/admin/locale/it_IT/messages.php | 160 + main/app/sprinkles/admin/locale/pt_PT/messages.php | 139 + main/app/sprinkles/admin/locale/ru_RU/messages.php | 160 + main/app/sprinkles/admin/locale/th_TH/messages.php | 134 + main/app/sprinkles/admin/locale/tr/messages.php | 160 + main/app/sprinkles/admin/locale/zh_CN/messages.php | 161 + main/app/sprinkles/admin/routes/activities.php | 19 + main/app/sprinkles/admin/routes/admin.php | 23 + main/app/sprinkles/admin/routes/groups.php | 39 + main/app/sprinkles/admin/routes/permissions.php | 25 + main/app/sprinkles/admin/routes/roles.php | 45 + main/app/sprinkles/admin/routes/users.php | 51 + .../admin/schema/requests/group/create.yaml | 35 + .../admin/schema/requests/group/edit-info.yaml | 27 + .../admin/schema/requests/group/get-by-slug.yaml | 6 + .../admin/schema/requests/role/create.yaml | 26 + .../admin/schema/requests/role/edit-field.yaml | 24 + .../admin/schema/requests/role/edit-info.yaml | 20 + .../admin/schema/requests/role/get-by-slug.yaml | 6 + .../admin/schema/requests/user/create.yaml | 72 + .../admin/schema/requests/user/edit-field.yaml | 60 + .../admin/schema/requests/user/edit-info.yaml | 36 + .../admin/schema/requests/user/edit-password.yaml | 30 + .../schema/requests/user/get-by-username.yaml | 6 + main/app/sprinkles/admin/src/Admin.php | 20 + .../admin/src/Controller/ActivityController.php | 85 + .../admin/src/Controller/AdminController.php | 150 + .../admin/src/Controller/GroupController.php | 725 +++ .../admin/src/Controller/PermissionController.php | 206 + .../admin/src/Controller/RoleController.php | 930 +++ .../admin/src/Controller/UserController.php | 1261 +++++ .../src/ServicesProvider/ServicesProvider.php | 84 + .../admin/src/Sprunje/ActivitySprunje.php | 80 + .../sprinkles/admin/src/Sprunje/GroupSprunje.php | 42 + .../admin/src/Sprunje/PermissionSprunje.php | 93 + .../admin/src/Sprunje/PermissionUserSprunje.php | 48 + .../sprinkles/admin/src/Sprunje/RoleSprunje.php | 67 + .../admin/src/Sprunje/UserPermissionSprunje.php | 48 + .../sprinkles/admin/src/Sprunje/UserSprunje.php | 185 + .../admin/templates/forms/group.html.twig | 69 + .../sprinkles/admin/templates/forms/role.html.twig | 56 + .../sprinkles/admin/templates/forms/user.html.twig | 125 + .../admin/templates/mail/password-create.html.twig | 19 + .../templates/modals/confirm-clear-cache.html.twig | 17 + .../modals/confirm-delete-group.html.twig | 17 + .../templates/modals/confirm-delete-role.html.twig | 17 + .../templates/modals/confirm-delete-user.html.twig | 17 + .../admin/templates/modals/group.html.twig | 7 + .../modals/role-manage-permissions.html.twig | 94 + .../admin/templates/modals/role.html.twig | 7 + .../templates/modals/user-manage-roles.html.twig | 77 + .../templates/modals/user-set-password.html.twig | 62 + .../admin/templates/modals/user.html.twig | 7 + .../admin/templates/navigation/navbar.html.twig | 15 + .../templates/navigation/sidebar-menu.html.twig | 43 + .../templates/navigation/sidebar-user.html.twig | 10 + .../admin/templates/navigation/sidebar.html.twig | 10 + .../admin/templates/navigation/user-card.html.twig | 8 + .../templates/pages/abstract/dashboard.html.twig | 87 + .../admin/templates/pages/activities.html.twig | 46 + .../admin/templates/pages/dashboard.html.twig | 282 + .../admin/templates/pages/group.html.twig | 106 + .../admin/templates/pages/groups.html.twig | 52 + .../admin/templates/pages/permission.html.twig | 91 + .../admin/templates/pages/permissions.html.twig | 45 + .../sprinkles/admin/templates/pages/role.html.twig | 129 + .../admin/templates/pages/roles.html.twig | 50 + .../sprinkles/admin/templates/pages/user.html.twig | 195 + .../admin/templates/pages/users.html.twig | 53 + .../admin/templates/tables/activities.html.twig | 73 + .../admin/templates/tables/groups.html.twig | 69 + .../admin/templates/tables/permissions.html.twig | 66 + .../admin/templates/tables/roles.html.twig | 74 + .../admin/templates/tables/users.html.twig | 149 + .../admin/tests/Integration/SprunjeTests.php | 111 + main/app/sprinkles/core/asset-bundles.json | 113 + .../sprinkles/core/assets/SiteAssets/css/main.css | 250 + .../sprinkles/core/assets/SiteAssets/css/slick.css | 2 + .../assets/SiteAssets/icons/BurgerMenuShort.svg | 57 + .../SiteAssets/icons/ExploreGlobeOutline.svg | 210 + .../assets/SiteAssets/icons/FriendFeedOutline.svg | 82 + .../SiteAssets/icons/MessageBubbleOutline.svg | 73 + .../assets/SiteAssets/icons/UserGroupOutline.svg | 80 + .../core/assets/SiteAssets/icons/UserOutline.svg | 54 + .../sprinkles/core/assets/SiteAssets/js/chat.js | 117 + .../core/assets/SiteAssets/js/encryption.js | 3407 +++++++++++ .../core/assets/SiteAssets/js/fontawesome.js | 5 + .../sprinkles/core/assets/SiteAssets/js/jquery.js | 4 + .../core/assets/SiteAssets/js/language.js | 36 + .../sprinkles/core/assets/SiteAssets/js/linkify.js | 369 ++ .../sprinkles/core/assets/SiteAssets/js/main.js | 91 + .../core/assets/SiteAssets/js/modernizr.js | 1 + .../sprinkles/core/assets/SiteAssets/js/slick.js | 556 ++ .../core/assets/SiteAssets/languages/ExcelFile.xls | Bin 0 -> 52736 bytes .../SiteAssets/languages/json/Translations.json | 50 + .../core/assets/SiteAssets/languages/json/de.json | 14 + .../core/assets/SiteAssets/languages/json/en.json | 14 + .../core/assets/SiteAssets/languages/json/fr.json | 14 + .../core/assets/SiteAssets/languages/json/kl.json | 14 + .../php/Chatserver/bin/WebChatServer.php | 18 + .../php/Chatserver/src/ChatProcessor.php | 118 + .../core/assets/SiteAssets/php/SavePublicKey.php | 17 + .../core/assets/SiteAssets/php/composer.json | 11 + .../core/assets/SiteAssets/php/composer.lock | 943 ++++ .../core/assets/SiteAssets/php/scripts.php | 12 + .../core/assets/SiteAssets/php/stylesheet.php | 6 + .../core/assets/SiteAssets/php/vendor/autoload.php | 7 + .../php/vendor/cboden/ratchet/.gitignore | 5 + .../php/vendor/cboden/ratchet/.travis.yml | 20 + .../php/vendor/cboden/ratchet/CHANGELOG.md | 135 + .../SiteAssets/php/vendor/cboden/ratchet/LICENSE | 19 + .../SiteAssets/php/vendor/cboden/ratchet/Makefile | 42 + .../SiteAssets/php/vendor/cboden/ratchet/README.md | 83 + .../php/vendor/cboden/ratchet/composer.json | 36 + .../php/vendor/cboden/ratchet/phpunit.xml.dist | 30 + .../src/Ratchet/AbstractConnectionDecorator.php | 41 + .../php/vendor/cboden/ratchet/src/Ratchet/App.php | 145 + .../ratchet/src/Ratchet/ComponentInterface.php | 31 + .../ratchet/src/Ratchet/ConnectionInterface.php | 26 + .../src/Ratchet/Http/CloseResponseTrait.php | 22 + .../ratchet/src/Ratchet/Http/HttpRequestParser.php | 64 + .../cboden/ratchet/src/Ratchet/Http/HttpServer.php | 76 + .../src/Ratchet/Http/HttpServerInterface.php | 14 + .../src/Ratchet/Http/NoOpHttpServerController.php | 18 + .../ratchet/src/Ratchet/Http/OriginCheck.php | 65 + .../cboden/ratchet/src/Ratchet/Http/Router.php | 96 + .../src/Ratchet/MessageComponentInterface.php | 5 + .../ratchet/src/Ratchet/MessageInterface.php | 12 + .../ratchet/src/Ratchet/Server/EchoServer.php | 23 + .../ratchet/src/Ratchet/Server/FlashPolicy.php | 200 + .../ratchet/src/Ratchet/Server/IoConnection.php | 38 + .../cboden/ratchet/src/Ratchet/Server/IoServer.php | 140 + .../ratchet/src/Ratchet/Server/IpBlackList.php | 111 + .../Ratchet/Session/Serialize/HandlerInterface.php | 16 + .../Ratchet/Session/Serialize/PhpBinaryHandler.php | 33 + .../src/Ratchet/Session/Serialize/PhpHandler.php | 49 + .../src/Ratchet/Session/SessionProvider.php | 243 + .../Ratchet/Session/Storage/Proxy/VirtualProxy.php | 54 + .../Session/Storage/VirtualSessionStorage.php | 88 + .../cboden/ratchet/src/Ratchet/Wamp/Exception.php | 5 + .../ratchet/src/Ratchet/Wamp/JsonException.php | 31 + .../ratchet/src/Ratchet/Wamp/ServerProtocol.php | 161 + .../cboden/ratchet/src/Ratchet/Wamp/Topic.php | 99 + .../ratchet/src/Ratchet/Wamp/TopicManager.php | 125 + .../ratchet/src/Ratchet/Wamp/WampConnection.php | 115 + .../cboden/ratchet/src/Ratchet/Wamp/WampServer.php | 67 + .../src/Ratchet/Wamp/WampServerInterface.php | 43 + .../ratchet/src/Ratchet/WebSocket/ConnContext.php | 20 + .../Ratchet/WebSocket/MessageCallableInterface.php | 8 + .../WebSocket/MessageComponentInterface.php | 6 + .../ratchet/src/Ratchet/WebSocket/WsConnection.php | 45 + .../ratchet/src/Ratchet/WebSocket/WsServer.php | 225 + .../src/Ratchet/WebSocket/WsServerInterface.php | 14 + .../ratchet/tests/autobahn/bin/fuzzingserver.php | 36 + .../ratchet/tests/autobahn/fuzzingclient-all.json | 15 + .../tests/autobahn/fuzzingclient-profile.json | 12 + .../tests/autobahn/fuzzingclient-quick.json | 12 + .../php/vendor/cboden/ratchet/tests/bootstrap.php | 4 + .../Ratchet/AbstractMessageComponentTestCase.php | 50 + .../tests/helpers/Ratchet/Mock/Component.php | 35 + .../tests/helpers/Ratchet/Mock/Connection.php | 20 + .../helpers/Ratchet/Mock/ConnectionDecorator.php | 22 + .../tests/helpers/Ratchet/Mock/WampComponent.php | 43 + .../tests/helpers/Ratchet/NullComponent.php | 28 + .../Ratchet/Wamp/Stub/WsWampServerInterface.php | 7 + .../WebSocket/Stub/WsMessageComponentInterface.php | 7 + .../tests/unit/AbstractConnectionDecoratorTest.php | 147 + .../tests/unit/Http/HttpRequestParserTest.php | 50 + .../ratchet/tests/unit/Http/HttpServerTest.php | 64 + .../ratchet/tests/unit/Http/OriginCheckTest.php | 46 + .../cboden/ratchet/tests/unit/Http/RouterTest.php | 165 + .../ratchet/tests/unit/Server/EchoServerTest.php | 26 + .../tests/unit/Server/FlashPolicyComponentTest.php | 152 + .../ratchet/tests/unit/Server/IoConnectionTest.php | 32 + .../ratchet/tests/unit/Server/IoServerTest.php | 118 + .../tests/unit/Server/IpBlackListComponentTest.php | 125 + .../unit/Session/Serialize/PhpHandlerTest.php | 43 + .../tests/unit/Session/SessionComponentTest.php | 124 + .../Storage/VirtualSessionStoragePDOTest.php | 53 + .../ratchet/tests/unit/Wamp/ServerProtocolTest.php | 295 + .../ratchet/tests/unit/Wamp/TopicManagerTest.php | 226 + .../cboden/ratchet/tests/unit/Wamp/TopicTest.php | 164 + .../ratchet/tests/unit/Wamp/WampConnectionTest.php | 77 + .../ratchet/tests/unit/Wamp/WampServerTest.php | 49 + .../SiteAssets/php/vendor/composer/ClassLoader.php | 445 ++ .../assets/SiteAssets/php/vendor/composer/LICENSE | 21 + .../php/vendor/composer/autoload_classmap.php | 84 + .../php/vendor/composer/autoload_files.php | 15 + .../php/vendor/composer/autoload_namespaces.php | 10 + .../php/vendor/composer/autoload_psr4.php | 28 + .../php/vendor/composer/autoload_real.php | 70 + .../php/vendor/composer/autoload_static.php | 238 + .../SiteAssets/php/vendor/composer/installed.json | 1125 ++++ .../php/vendor/evenement/evenement/.gitignore | 2 + .../php/vendor/evenement/evenement/.travis.yml | 24 + .../php/vendor/evenement/evenement/CHANGELOG.md | 35 + .../php/vendor/evenement/evenement/LICENSE | 19 + .../php/vendor/evenement/evenement/README.md | 83 + .../php/vendor/evenement/evenement/composer.json | 29 + .../php/vendor/evenement/evenement/doc/00-intro.md | 28 + .../php/vendor/evenement/evenement/doc/01-api.md | 91 + .../evenement/evenement/doc/02-plugin-system.md | 155 + .../examples/benchmark-emit-no-arguments.php | 28 + .../evenement/examples/benchmark-emit-once.php | 30 + .../examples/benchmark-emit-one-argument.php | 28 + .../evenement/examples/benchmark-emit.php | 28 + .../examples/benchmark-remove-listener-once.php | 39 + .../vendor/evenement/evenement/phpunit.xml.dist | 24 + .../evenement/src/Evenement/EventEmitter.php | 17 + .../src/Evenement/EventEmitterInterface.php | 22 + .../evenement/src/Evenement/EventEmitterTrait.php | 135 + .../tests/Evenement/Tests/EventEmitterTest.php | 438 ++ .../evenement/tests/Evenement/Tests/Listener.php | 51 + .../evenement/tests/Evenement/Tests/functions.php | 17 + .../php/vendor/guzzlehttp/psr7/CHANGELOG.md | 110 + .../SiteAssets/php/vendor/guzzlehttp/psr7/LICENSE | 19 + .../php/vendor/guzzlehttp/psr7/README.md | 739 +++ .../php/vendor/guzzlehttp/psr7/composer.json | 39 + .../vendor/guzzlehttp/psr7/src/AppendStream.php | 233 + .../vendor/guzzlehttp/psr7/src/BufferStream.php | 137 + .../vendor/guzzlehttp/psr7/src/CachingStream.php | 138 + .../vendor/guzzlehttp/psr7/src/DroppingStream.php | 42 + .../php/vendor/guzzlehttp/psr7/src/FnStream.php | 149 + .../vendor/guzzlehttp/psr7/src/InflateStream.php | 52 + .../vendor/guzzlehttp/psr7/src/LazyOpenStream.php | 39 + .../php/vendor/guzzlehttp/psr7/src/LimitStream.php | 155 + .../vendor/guzzlehttp/psr7/src/MessageTrait.php | 183 + .../vendor/guzzlehttp/psr7/src/MultipartStream.php | 153 + .../vendor/guzzlehttp/psr7/src/NoSeekStream.php | 22 + .../php/vendor/guzzlehttp/psr7/src/PumpStream.php | 165 + .../php/vendor/guzzlehttp/psr7/src/Request.php | 142 + .../php/vendor/guzzlehttp/psr7/src/Response.php | 132 + .../vendor/guzzlehttp/psr7/src/ServerRequest.php | 358 ++ .../php/vendor/guzzlehttp/psr7/src/Stream.php | 257 + .../guzzlehttp/psr7/src/StreamDecoratorTrait.php | 149 + .../vendor/guzzlehttp/psr7/src/StreamWrapper.php | 121 + .../vendor/guzzlehttp/psr7/src/UploadedFile.php | 316 ++ .../php/vendor/guzzlehttp/psr7/src/Uri.php | 702 +++ .../vendor/guzzlehttp/psr7/src/UriNormalizer.php | 216 + .../php/vendor/guzzlehttp/psr7/src/UriResolver.php | 219 + .../php/vendor/guzzlehttp/psr7/src/functions.php | 828 +++ .../guzzlehttp/psr7/src/functions_include.php | 6 + .../vendor/kriswallsmith/assetic/CHANGELOG-1.0.md | 36 + .../vendor/kriswallsmith/assetic/CHANGELOG-1.1.md | 57 + .../vendor/kriswallsmith/assetic/CHANGELOG-1.2.md | 49 + .../php/vendor/kriswallsmith/assetic/Gemfile | 5 + .../php/vendor/kriswallsmith/assetic/LICENSE | 19 + .../php/vendor/kriswallsmith/assetic/README.md | 345 ++ .../php/vendor/kriswallsmith/assetic/composer.json | 56 + .../php/vendor/kriswallsmith/assetic/package.json | 19 + .../assetic/src/Assetic/Asset/AssetCache.php | 174 + .../assetic/src/Assetic/Asset/AssetCollection.php | 238 + .../src/Assetic/Asset/AssetCollectionInterface.php | 59 + .../assetic/src/Assetic/Asset/AssetInterface.php | 166 + .../assetic/src/Assetic/Asset/AssetReference.php | 164 + .../assetic/src/Assetic/Asset/BaseAsset.php | 181 + .../assetic/src/Assetic/Asset/FileAsset.php | 78 + .../assetic/src/Assetic/Asset/GlobAsset.php | 115 + .../assetic/src/Assetic/Asset/HttpAsset.php | 79 + .../Iterator/AssetCollectionFilterIterator.php | 84 + .../Asset/Iterator/AssetCollectionIterator.php | 128 + .../assetic/src/Assetic/Asset/StringAsset.php | 55 + .../assetic/src/Assetic/AssetManager.php | 89 + .../assetic/src/Assetic/AssetWriter.php | 94 + .../assetic/src/Assetic/Cache/ApcCache.php | 66 + .../assetic/src/Assetic/Cache/ArrayCache.php | 58 + .../assetic/src/Assetic/Cache/CacheInterface.php | 53 + .../assetic/src/Assetic/Cache/ConfigCache.php | 123 + .../assetic/src/Assetic/Cache/ExpiringCache.php | 60 + .../assetic/src/Assetic/Cache/FilesystemCache.php | 65 + .../assetic/src/Assetic/Exception/Exception.php | 21 + .../src/Assetic/Exception/FilterException.php | 73 + .../Assetic/Extension/Twig/AsseticExtension.php | 76 + .../Extension/Twig/AsseticFilterFunction.php | 24 + .../Extension/Twig/AsseticFilterInvoker.php | 59 + .../Assetic/Extension/Twig/AsseticFilterNode.php | 22 + .../src/Assetic/Extension/Twig/AsseticNode.php | 165 + .../Assetic/Extension/Twig/AsseticTokenParser.php | 198 + .../Assetic/Extension/Twig/TwigFormulaLoader.php | 108 + .../src/Assetic/Extension/Twig/TwigResource.php | 56 + .../src/Assetic/Extension/Twig/ValueContainer.php | 79 + .../assetic/src/Assetic/Factory/AssetFactory.php | 424 ++ .../src/Assetic/Factory/LazyAssetManager.php | 210 + .../Factory/Loader/BasePhpFormulaLoader.php | 160 + .../Assetic/Factory/Loader/CachedFormulaLoader.php | 68 + .../Factory/Loader/FormulaLoaderInterface.php | 34 + .../Factory/Loader/FunctionCallsFormulaLoader.php | 53 + .../Resource/CoalescingDirectoryResource.php | 112 + .../Assetic/Factory/Resource/DirectoryResource.php | 133 + .../src/Assetic/Factory/Resource/FileResource.php | 47 + .../Factory/Resource/IteratorResourceInterface.php | 21 + .../Assetic/Factory/Resource/ResourceInterface.php | 43 + .../Assetic/Factory/Worker/CacheBustingWorker.php | 70 + .../Assetic/Factory/Worker/EnsureFilterWorker.php | 61 + .../src/Assetic/Factory/Worker/WorkerInterface.php | 33 + .../src/Assetic/Filter/AutoprefixerFilter.php | 87 + .../assetic/src/Assetic/Filter/BaseCssFilter.php | 54 + .../assetic/src/Assetic/Filter/BaseNodeFilter.php | 44 + .../src/Assetic/Filter/BaseProcessFilter.php | 57 + .../assetic/src/Assetic/Filter/CallablesFilter.php | 62 + .../assetic/src/Assetic/Filter/CleanCssFilter.php | 343 ++ .../src/Assetic/Filter/CoffeeScriptFilter.php | 83 + .../assetic/src/Assetic/Filter/CompassFilter.php | 391 ++ .../src/Assetic/Filter/CssCacheBustingFilter.php | 65 + .../assetic/src/Assetic/Filter/CssEmbedFilter.php | 143 + .../assetic/src/Assetic/Filter/CssImportFilter.php | 108 + .../assetic/src/Assetic/Filter/CssMinFilter.php | 72 + .../src/Assetic/Filter/CssRewriteFilter.php | 102 + .../assetic/src/Assetic/Filter/DartFilter.php | 73 + .../Filter/DependencyExtractorInterface.php | 34 + .../src/Assetic/Filter/EmberPrecompileFilter.php | 87 + .../src/Assetic/Filter/FilterCollection.php | 82 + .../assetic/src/Assetic/Filter/FilterInterface.php | 36 + .../Filter/GoogleClosure/BaseCompilerFilter.php | 101 + .../Filter/GoogleClosure/CompilerApiFilter.php | 130 + .../Filter/GoogleClosure/CompilerJarFilter.php | 112 + .../assetic/src/Assetic/Filter/GssFilter.php | 142 + .../src/Assetic/Filter/HandlebarsFilter.php | 106 + .../src/Assetic/Filter/HashableInterface.php | 27 + .../assetic/src/Assetic/Filter/JSMinFilter.php | 34 + .../assetic/src/Assetic/Filter/JSMinPlusFilter.php | 34 + .../assetic/src/Assetic/Filter/JSqueezeFilter.php | 77 + .../assetic/src/Assetic/Filter/JpegoptimFilter.php | 81 + .../assetic/src/Assetic/Filter/JpegtranFilter.php | 103 + .../assetic/src/Assetic/Filter/LessFilter.php | 206 + .../assetic/src/Assetic/Filter/LessphpFilter.php | 167 + .../Assetic/Filter/MinifyCssCompressorFilter.php | 35 + .../assetic/src/Assetic/Filter/OptiPngFilter.php | 75 + .../assetic/src/Assetic/Filter/PackagerFilter.php | 65 + .../assetic/src/Assetic/Filter/PackerFilter.php | 56 + .../src/Assetic/Filter/PhpCssEmbedFilter.php | 52 + .../assetic/src/Assetic/Filter/PngoutFilter.php | 128 + .../assetic/src/Assetic/Filter/ReactJsxFilter.php | 75 + .../assetic/src/Assetic/Filter/RooleFilter.php | 84 + .../src/Assetic/Filter/Sass/BaseSassFilter.php | 95 + .../assetic/src/Assetic/Filter/Sass/SassFilter.php | 186 + .../assetic/src/Assetic/Filter/Sass/ScssFilter.php | 28 + .../assetic/src/Assetic/Filter/SassphpFilter.php | 132 + .../assetic/src/Assetic/Filter/ScssphpFilter.php | 147 + .../assetic/src/Assetic/Filter/SeparatorFilter.php | 47 + .../assetic/src/Assetic/Filter/SprocketsFilter.php | 152 + .../assetic/src/Assetic/Filter/StylusFilter.php | 126 + .../src/Assetic/Filter/TypeScriptFilter.php | 80 + .../assetic/src/Assetic/Filter/UglifyCssFilter.php | 120 + .../assetic/src/Assetic/Filter/UglifyJs2Filter.php | 152 + .../assetic/src/Assetic/Filter/UglifyJsFilter.php | 160 + .../Assetic/Filter/Yui/BaseCompressorFilter.php | 117 + .../src/Assetic/Filter/Yui/CssCompressorFilter.php | 28 + .../src/Assetic/Filter/Yui/JsCompressorFilter.php | 61 + .../assetic/src/Assetic/FilterManager.php | 64 + .../assetic/src/Assetic/Util/CssUtils.php | 138 + .../assetic/src/Assetic/Util/FilesystemUtils.php | 84 + .../assetic/src/Assetic/Util/LessUtils.php | 24 + .../assetic/src/Assetic/Util/SassUtils.php | 22 + .../assetic/src/Assetic/Util/TraversableString.php | 44 + .../assetic/src/Assetic/Util/VarUtils.php | 84 + .../assetic/src/Assetic/ValueSupplierInterface.php | 29 + .../vendor/kriswallsmith/assetic/src/functions.php | 125 + .../php/vendor/lmammino/jsmin4assetic/LICENSE | 19 + .../vendor/lmammino/jsmin4assetic/README.markdown | 21 + .../vendor/lmammino/jsmin4assetic/composer.json | 26 + .../vendor/lmammino/jsmin4assetic/src/JSMin.php | 379 ++ .../src/JSMinUnterminatedCommentException.php | 5 + .../src/JSMinUnterminatedRegExpException.php | 5 + .../src/JSMinUnterminatedStringException.php | 5 + .../vendor/matthiasmullie/minify/CONTRIBUTING.md | 59 + .../php/vendor/matthiasmullie/minify/Dockerfile | 13 + .../php/vendor/matthiasmullie/minify/LICENSE | 18 + .../php/vendor/matthiasmullie/minify/bin/minifycss | 45 + .../php/vendor/matthiasmullie/minify/bin/minifyjs | 45 + .../php/vendor/matthiasmullie/minify/composer.json | 38 + .../minify/data/js/keywords_after.txt | 7 + .../minify/data/js/keywords_before.txt | 26 + .../minify/data/js/keywords_reserved.txt | 63 + .../matthiasmullie/minify/data/js/operators.txt | 46 + .../minify/data/js/operators_after.txt | 43 + .../minify/data/js/operators_before.txt | 43 + .../matthiasmullie/minify/docker-compose.yml | 31 + .../php/vendor/matthiasmullie/minify/src/CSS.php | 736 +++ .../vendor/matthiasmullie/minify/src/Exception.php | 20 + .../minify/src/Exceptions/BasicException.php | 23 + .../minify/src/Exceptions/FileImportException.php | 21 + .../minify/src/Exceptions/IOException.php | 21 + .../php/vendor/matthiasmullie/minify/src/JS.php | 598 ++ .../vendor/matthiasmullie/minify/src/Minify.php | 454 ++ .../vendor/matthiasmullie/path-converter/LICENSE | 18 + .../matthiasmullie/path-converter/composer.json | 28 + .../path-converter/src/Converter.php | 195 + .../path-converter/src/ConverterInterface.php | 24 + .../path-converter/src/NoConverter.php | 23 + .../SiteAssets/php/vendor/natxet/CssMin/README | 3 + .../php/vendor/natxet/CssMin/composer.json | 26 + .../php/vendor/natxet/CssMin/src/CssMin.php | 5155 +++++++++++++++++ .../nubs/random-name-generator/.coveralls.yml | 3 + .../vendor/nubs/random-name-generator/.gitignore | 2 + .../nubs/random-name-generator/.scrutinizer.yml | 19 + .../vendor/nubs/random-name-generator/.travis.yml | 9 + .../nubs/random-name-generator/CONTRIBUTING.md | 38 + .../nubs/random-name-generator/Dockerfile.tests | 5 + .../php/vendor/nubs/random-name-generator/LICENSE | 21 + .../vendor/nubs/random-name-generator/README.md | 91 + .../vendor/nubs/random-name-generator/build.php | 25 + .../nubs/random-name-generator/composer.json | 30 + .../nubs/random-name-generator/composer.lock | 1963 +++++++ .../nubs/random-name-generator/docker-compose.yml | 8 + .../vendor/nubs/random-name-generator/phpunit.xml | 10 + .../src/AbstractGenerator.php | 19 + .../vendor/nubs/random-name-generator/src/All.php | 62 + .../random-name-generator/src/Alliteration.php | 59 + .../nubs/random-name-generator/src/Generator.php | 16 + .../vendor/nubs/random-name-generator/src/Vgng.php | 138 + .../nubs/random-name-generator/src/adjectives.txt | 233 + .../nubs/random-name-generator/src/nouns.txt | 313 ++ .../random-name-generator/src/video_game_names.txt | 1276 +++++ .../nubs/random-name-generator/tests/AllTest.php | 72 + .../tests/AlliterationTest.php | 66 + .../nubs/random-name-generator/tests/VgngTest.php | 67 + .../php/vendor/paragonie/random_compat/LICENSE | 22 + .../vendor/paragonie/random_compat/build-phar.sh | 5 + .../vendor/paragonie/random_compat/composer.json | 37 + .../random_compat/dist/random_compat.phar.pubkey | 5 + .../dist/random_compat.phar.pubkey.asc | 11 + .../random_compat/lib/byte_safe_strings.php | 181 + .../paragonie/random_compat/lib/cast_to_int.php | 75 + .../paragonie/random_compat/lib/error_polyfill.php | 49 + .../vendor/paragonie/random_compat/lib/random.php | 225 + .../random_compat/lib/random_bytes_com_dotnet.php | 88 + .../random_compat/lib/random_bytes_dev_urandom.php | 167 + .../random_compat/lib/random_bytes_libsodium.php | 88 + .../lib/random_bytes_libsodium_legacy.php | 92 + .../random_compat/lib/random_bytes_mcrypt.php | 77 + .../paragonie/random_compat/lib/random_int.php | 190 + .../paragonie/random_compat/other/build_phar.php | 57 + .../paragonie/random_compat/psalm-autoload.php | 9 + .../php/vendor/paragonie/random_compat/psalm.xml | 18 + .../php/vendor/psr/http-message/CHANGELOG.md | 36 + .../SiteAssets/php/vendor/psr/http-message/LICENSE | 19 + .../php/vendor/psr/http-message/README.md | 13 + .../php/vendor/psr/http-message/composer.json | 26 + .../psr/http-message/src/MessageInterface.php | 187 + .../psr/http-message/src/RequestInterface.php | 129 + .../psr/http-message/src/ResponseInterface.php | 68 + .../http-message/src/ServerRequestInterface.php | 261 + .../psr/http-message/src/StreamInterface.php | 158 + .../psr/http-message/src/UploadedFileInterface.php | 123 + .../vendor/psr/http-message/src/UriInterface.php | 323 ++ .../php/vendor/ratchet/rfc6455/.gitignore | 4 + .../php/vendor/ratchet/rfc6455/.travis.yml | 20 + .../SiteAssets/php/vendor/ratchet/rfc6455/LICENSE | 19 + .../php/vendor/ratchet/rfc6455/README.md | 13 + .../php/vendor/ratchet/rfc6455/composer.json | 32 + .../php/vendor/ratchet/rfc6455/phpunit.xml.dist | 27 + .../rfc6455/src/Handshake/ClientNegotiator.php | 53 + .../rfc6455/src/Handshake/NegotiatorInterface.php | 47 + .../rfc6455/src/Handshake/RequestVerifier.php | 140 + .../rfc6455/src/Handshake/ResponseVerifier.php | 52 + .../rfc6455/src/Handshake/ServerNegotiator.php | 136 + .../rfc6455/src/Messaging/CloseFrameChecker.php | 24 + .../rfc6455/src/Messaging/DataInterface.php | 34 + .../vendor/ratchet/rfc6455/src/Messaging/Frame.php | 473 ++ .../rfc6455/src/Messaging/FrameInterface.php | 38 + .../ratchet/rfc6455/src/Messaging/Message.php | 123 + .../rfc6455/src/Messaging/MessageBuffer.php | 231 + .../rfc6455/src/Messaging/MessageInterface.php | 20 + .../vendor/ratchet/rfc6455/tests/AbResultsTest.php | 30 + .../ratchet/rfc6455/tests/ab/clientRunner.php | 228 + .../ratchet/rfc6455/tests/ab/fuzzingclient.json | 14 + .../ratchet/rfc6455/tests/ab/fuzzingserver.json | 10 + .../ratchet/rfc6455/tests/ab/run_ab_tests.sh | 11 + .../ratchet/rfc6455/tests/ab/startServer.php | 55 + .../php/vendor/ratchet/rfc6455/tests/bootstrap.php | 19 + .../tests/unit/Handshake/RequestVerifierTest.php | 177 + .../tests/unit/Handshake/ResponseVerifierTest.php | 34 + .../tests/unit/Handshake/ServerNegotiatorTest.php | 175 + .../rfc6455/tests/unit/Messaging/FrameTest.php | 501 ++ .../tests/unit/Messaging/MessageBufferTest.php | 39 + .../rfc6455/tests/unit/Messaging/MessageTest.php | 58 + .../SiteAssets/php/vendor/react/cache/.gitignore | 2 + .../SiteAssets/php/vendor/react/cache/.travis.yml | 25 + .../SiteAssets/php/vendor/react/cache/CHANGELOG.md | 35 + .../SiteAssets/php/vendor/react/cache/LICENSE | 19 + .../SiteAssets/php/vendor/react/cache/README.md | 171 + .../php/vendor/react/cache/composer.json | 19 + .../php/vendor/react/cache/phpunit.xml.dist | 20 + .../php/vendor/react/cache/src/ArrayCache.php | 29 + .../php/vendor/react/cache/src/CacheInterface.php | 13 + .../vendor/react/cache/tests/ArrayCacheTest.php | 60 + .../php/vendor/react/cache/tests/CallableStub.php | 10 + .../php/vendor/react/cache/tests/TestCase.php | 43 + .../SiteAssets/php/vendor/react/dns/.gitignore | 2 + .../SiteAssets/php/vendor/react/dns/.travis.yml | 29 + .../SiteAssets/php/vendor/react/dns/CHANGELOG.md | 179 + .../assets/SiteAssets/php/vendor/react/dns/LICENSE | 19 + .../SiteAssets/php/vendor/react/dns/README.md | 209 + .../SiteAssets/php/vendor/react/dns/composer.json | 24 + .../php/vendor/react/dns/examples/01-one.php | 22 + .../vendor/react/dns/examples/02-concurrent.php | 27 + .../php/vendor/react/dns/examples/03-cached.php | 40 + .../react/dns/examples/04-query-a-and-aaaa.php | 32 + .../php/vendor/react/dns/phpunit.xml.dist | 25 + .../vendor/react/dns/src/BadServerException.php | 7 + .../php/vendor/react/dns/src/Config/Config.php | 127 + .../react/dns/src/Config/FilesystemFactory.php | 73 + .../php/vendor/react/dns/src/Config/HostsFile.php | 151 + .../php/vendor/react/dns/src/Model/HeaderBag.php | 56 + .../php/vendor/react/dns/src/Model/Message.php | 100 + .../php/vendor/react/dns/src/Model/Record.php | 21 + .../vendor/react/dns/src/Protocol/BinaryDumper.php | 62 + .../php/vendor/react/dns/src/Protocol/Parser.php | 254 + .../vendor/react/dns/src/Query/CachedExecutor.php | 55 + .../react/dns/src/Query/CancellationException.php | 7 + .../php/vendor/react/dns/src/Query/Executor.php | 156 + .../react/dns/src/Query/ExecutorInterface.php | 8 + .../react/dns/src/Query/HostsFileExecutor.php | 89 + .../php/vendor/react/dns/src/Query/Query.php | 19 + .../php/vendor/react/dns/src/Query/RecordBag.php | 27 + .../php/vendor/react/dns/src/Query/RecordCache.php | 82 + .../vendor/react/dns/src/Query/RetryExecutor.php | 44 + .../react/dns/src/Query/TimeoutException.php | 7 + .../vendor/react/dns/src/Query/TimeoutExecutor.php | 32 + .../react/dns/src/RecordNotFoundException.php | 7 + .../php/vendor/react/dns/src/Resolver/Factory.php | 103 + .../php/vendor/react/dns/src/Resolver/Resolver.php | 100 + .../php/vendor/react/dns/tests/CallableStub.php | 10 + .../vendor/react/dns/tests/Config/ConfigTest.php | 189 + .../dns/tests/Config/FilesystemFactoryTest.php | 70 + .../react/dns/tests/Config/HostsFileTest.php | 170 + .../react/dns/tests/Fixtures/etc/resolv.conf | 1 + .../react/dns/tests/FunctionalResolverTest.php | 71 + .../vendor/react/dns/tests/Model/MessageTest.php | 30 + .../react/dns/tests/Protocol/BinaryDumperTest.php | 48 + .../vendor/react/dns/tests/Protocol/ParserTest.php | 343 ++ .../react/dns/tests/Query/CachedExecutorTest.php | 100 + .../vendor/react/dns/tests/Query/ExecutorTest.php | 308 + .../dns/tests/Query/HostsFileExecutorTest.php | 126 + .../vendor/react/dns/tests/Query/RecordBagTest.php | 64 + .../react/dns/tests/Query/RecordCacheTest.php | 123 + .../react/dns/tests/Query/RetryExecutorTest.php | 197 + .../react/dns/tests/Query/TimeoutExecutorTest.php | 115 + .../react/dns/tests/Resolver/FactoryTest.php | 131 + .../dns/tests/Resolver/ResolveAliasesTest.php | 100 + .../react/dns/tests/Resolver/ResolverTest.php | 129 + .../php/vendor/react/dns/tests/TestCase.php | 61 + .../php/vendor/react/event-loop/.gitignore | 3 + .../php/vendor/react/event-loop/.travis.yml | 39 + .../php/vendor/react/event-loop/CHANGELOG.md | 316 ++ .../SiteAssets/php/vendor/react/event-loop/LICENSE | 19 + .../php/vendor/react/event-loop/README.md | 702 +++ .../php/vendor/react/event-loop/composer.json | 21 + .../vendor/react/event-loop/examples/01-timers.php | 15 + .../react/event-loop/examples/02-periodic.php | 16 + .../vendor/react/event-loop/examples/03-ticks.php | 15 + .../react/event-loop/examples/04-signals.php | 19 + .../react/event-loop/examples/11-consume-stdin.php | 30 + .../react/event-loop/examples/12-generate-yes.php | 41 + .../examples/13-http-client-blocking.php | 35 + .../event-loop/examples/14-http-client-async.php | 63 + .../react/event-loop/examples/21-http-server.php | 36 + .../event-loop/examples/91-benchmark-ticks.php | 15 + .../event-loop/examples/92-benchmark-timers.php | 15 + .../examples/93-benchmark-ticks-delay.php | 22 + .../examples/94-benchmark-timers-delay.php | 22 + .../event-loop/examples/95-benchmark-memory.php | 67 + .../php/vendor/react/event-loop/phpunit.xml.dist | 25 + .../php/vendor/react/event-loop/src/ExtEvLoop.php | 252 + .../vendor/react/event-loop/src/ExtEventLoop.php | 259 + .../vendor/react/event-loop/src/ExtLibevLoop.php | 199 + .../react/event-loop/src/ExtLibeventLoop.php | 283 + .../php/vendor/react/event-loop/src/Factory.php | 41 + .../vendor/react/event-loop/src/LoopInterface.php | 463 ++ .../vendor/react/event-loop/src/SignalsHandler.php | 63 + .../react/event-loop/src/StreamSelectLoop.php | 275 + .../react/event-loop/src/Tick/FutureTickQueue.php | 60 + .../vendor/react/event-loop/src/Timer/Timer.php | 55 + .../vendor/react/event-loop/src/Timer/Timers.php | 109 + .../vendor/react/event-loop/src/TimerInterface.php | 27 + .../react/event-loop/tests/AbstractLoopTest.php | 621 +++ .../vendor/react/event-loop/tests/CallableStub.php | 10 + .../react/event-loop/tests/ExtEvLoopTest.php | 17 + .../react/event-loop/tests/ExtEventLoopTest.php | 84 + .../react/event-loop/tests/ExtLibevLoopTest.php | 22 + .../react/event-loop/tests/ExtLibeventLoopTest.php | 58 + .../react/event-loop/tests/SignalsHandlerTest.php | 55 + .../event-loop/tests/StreamSelectLoopTest.php | 148 + .../php/vendor/react/event-loop/tests/TestCase.php | 53 + .../event-loop/tests/Timer/AbstractTimerTest.php | 122 + .../event-loop/tests/Timer/ExtEvTimerTest.php | 17 + .../event-loop/tests/Timer/ExtEventTimerTest.php | 17 + .../event-loop/tests/Timer/ExtLibevTimerTest.php | 17 + .../tests/Timer/ExtLibeventTimerTest.php | 17 + .../tests/Timer/StreamSelectTimerTest.php | 13 + .../react/event-loop/tests/Timer/TimersTest.php | 27 + .../vendor/react/event-loop/tests/bootstrap.php | 15 + .../php/vendor/react/event-loop/travis-init.sh | 42 + .../php/vendor/react/promise-timer/.gitignore | 2 + .../php/vendor/react/promise-timer/.travis.yml | 26 + .../php/vendor/react/promise-timer/CHANGELOG.md | 40 + .../php/vendor/react/promise-timer/LICENSE | 21 + .../php/vendor/react/promise-timer/README.md | 372 ++ .../php/vendor/react/promise-timer/composer.json | 28 + .../vendor/react/promise-timer/phpunit.xml.dist | 19 + .../react/promise-timer/src/TimeoutException.php | 22 + .../vendor/react/promise-timer/src/functions.php | 70 + .../react/promise-timer/tests/CallableStub.php | 10 + .../promise-timer/tests/FunctionRejectTest.php | 49 + .../promise-timer/tests/FunctionResolveTest.php | 71 + .../promise-timer/tests/FunctionTimeoutTest.php | 169 + .../vendor/react/promise-timer/tests/TestCase.php | 61 + .../promise-timer/tests/TimeoutExceptionTest.php | 15 + .../SiteAssets/php/vendor/react/promise/.gitignore | 5 + .../php/vendor/react/promise/.travis.yml | 22 + .../php/vendor/react/promise/CHANGELOG.md | 96 + .../SiteAssets/php/vendor/react/promise/LICENSE | 22 + .../SiteAssets/php/vendor/react/promise/README.md | 840 +++ .../php/vendor/react/promise/composer.json | 29 + .../php/vendor/react/promise/phpunit.xml.dist | 28 + .../promise/src/CancellablePromiseInterface.php | 11 + .../vendor/react/promise/src/CancellationQueue.php | 55 + .../php/vendor/react/promise/src/Deferred.php | 60 + .../promise/src/Exception/LengthException.php | 7 + .../react/promise/src/ExtendedPromiseInterface.php | 26 + .../vendor/react/promise/src/FulfilledPromise.php | 68 + .../php/vendor/react/promise/src/LazyPromise.php | 63 + .../php/vendor/react/promise/src/Promise.php | 216 + .../vendor/react/promise/src/PromiseInterface.php | 11 + .../vendor/react/promise/src/PromisorInterface.php | 11 + .../vendor/react/promise/src/RejectedPromise.php | 76 + .../promise/src/UnhandledRejectionException.php | 31 + .../php/vendor/react/promise/src/functions.php | 244 + .../vendor/react/promise/src/functions_include.php | 5 + .../react/promise/tests/CancellationQueueTest.php | 100 + .../vendor/react/promise/tests/DeferredTest.php | 42 + .../react/promise/tests/FulfilledPromiseTest.php | 50 + .../vendor/react/promise/tests/FunctionAllTest.php | 114 + .../vendor/react/promise/tests/FunctionAnyTest.php | 204 + .../promise/tests/FunctionCheckTypehintTest.php | 118 + .../vendor/react/promise/tests/FunctionMapTest.php | 198 + .../react/promise/tests/FunctionRaceTest.php | 211 + .../react/promise/tests/FunctionReduceTest.php | 347 ++ .../react/promise/tests/FunctionRejectTest.php | 64 + .../react/promise/tests/FunctionResolveTest.php | 171 + .../react/promise/tests/FunctionSomeTest.php | 258 + .../vendor/react/promise/tests/LazyPromiseTest.php | 107 + .../PromiseAdapter/CallbackPromiseAdapter.php | 40 + .../PromiseAdapter/PromiseAdapterInterface.php | 14 + .../php/vendor/react/promise/tests/PromiseTest.php | 84 + .../promise/tests/PromiseTest/CancelTestTrait.php | 231 + .../promise/tests/PromiseTest/FullTestTrait.php | 15 + .../promise/tests/PromiseTest/NotifyTestTrait.php | 336 ++ .../PromiseTest/PromiseFulfilledTestTrait.php | 351 ++ .../tests/PromiseTest/PromisePendingTestTrait.php | 68 + .../tests/PromiseTest/PromiseRejectedTestTrait.php | 512 ++ .../tests/PromiseTest/PromiseSettledTestTrait.php | 86 + .../promise/tests/PromiseTest/RejectTestTrait.php | 368 ++ .../promise/tests/PromiseTest/ResolveTestTrait.php | 312 ++ .../react/promise/tests/RejectedPromiseTest.php | 50 + .../react/promise/tests/Stub/CallableStub.php | 10 + .../php/vendor/react/promise/tests/TestCase.php | 43 + .../php/vendor/react/promise/tests/bootstrap.php | 7 + .../tests/fixtures/SimpleFulfilledTestPromise.php | 21 + .../tests/fixtures/SimpleFulfilledTestThenable.php | 21 + .../tests/fixtures/SimpleRejectedTestPromise.php | 21 + .../tests/fixtures/SimpleTestCancellable.php | 13 + .../fixtures/SimpleTestCancellableThenable.php | 18 + .../SiteAssets/php/vendor/react/socket/.gitignore | 2 + .../SiteAssets/php/vendor/react/socket/.travis.yml | 49 + .../php/vendor/react/socket/CHANGELOG.md | 451 ++ .../SiteAssets/php/vendor/react/socket/LICENSE | 19 + .../SiteAssets/php/vendor/react/socket/README.md | 1419 +++++ .../php/vendor/react/socket/composer.json | 29 + .../react/socket/examples/01-echo-server.php | 42 + .../react/socket/examples/02-chat-server.php | 59 + .../react/socket/examples/03-http-server.php | 57 + .../react/socket/examples/11-http-client.php | 36 + .../react/socket/examples/12-https-client.php | 36 + .../react/socket/examples/21-netcat-client.php | 68 + .../react/socket/examples/22-http-client.php | 60 + .../react/socket/examples/91-benchmark-server.php | 60 + .../socket/examples/99-generate-self-signed.php | 31 + .../php/vendor/react/socket/examples/localhost.pem | 49 + .../react/socket/examples/localhost_swordfish.pem | 51 + .../php/vendor/react/socket/phpunit.xml.dist | 25 + .../php/vendor/react/socket/src/Connection.php | 178 + .../react/socket/src/ConnectionInterface.php | 119 + .../php/vendor/react/socket/src/Connector.php | 136 + .../vendor/react/socket/src/ConnectorInterface.php | 58 + .../php/vendor/react/socket/src/DnsConnector.php | 111 + .../vendor/react/socket/src/FixedUriConnector.php | 41 + .../php/vendor/react/socket/src/LimitingServer.php | 203 + .../vendor/react/socket/src/SecureConnector.php | 64 + .../php/vendor/react/socket/src/SecureServer.php | 192 + .../php/vendor/react/socket/src/Server.php | 73 + .../vendor/react/socket/src/ServerInterface.php | 151 + .../vendor/react/socket/src/StreamEncryption.php | 146 + .../php/vendor/react/socket/src/TcpConnector.php | 122 + .../php/vendor/react/socket/src/TcpServer.php | 236 + .../vendor/react/socket/src/TimeoutConnector.php | 25 + .../php/vendor/react/socket/src/UnixConnector.php | 44 + .../php/vendor/react/socket/src/UnixServer.php | 130 + .../vendor/react/socket/tests/ConnectionTest.php | 47 + .../vendor/react/socket/tests/ConnectorTest.php | 128 + .../vendor/react/socket/tests/DnsConnectorTest.php | 111 + .../react/socket/tests/FixedUriConnectorTest.php | 19 + .../react/socket/tests/FunctionalConnectorTest.php | 32 + .../socket/tests/FunctionalSecureServerTest.php | 438 ++ .../react/socket/tests/FunctionalTcpServerTest.php | 324 ++ .../vendor/react/socket/tests/IntegrationTest.php | 171 + .../react/socket/tests/LimitingServerTest.php | 195 + .../react/socket/tests/SecureConnectorTest.php | 74 + .../react/socket/tests/SecureIntegrationTest.php | 204 + .../vendor/react/socket/tests/SecureServerTest.php | 105 + .../php/vendor/react/socket/tests/ServerTest.php | 173 + .../react/socket/tests/Stub/CallableStub.php | 10 + .../react/socket/tests/Stub/ConnectionStub.php | 63 + .../vendor/react/socket/tests/Stub/ServerStub.php | 18 + .../vendor/react/socket/tests/TcpConnectorTest.php | 255 + .../vendor/react/socket/tests/TcpServerTest.php | 285 + .../php/vendor/react/socket/tests/TestCase.php | 101 + .../react/socket/tests/TimeoutConnectorTest.php | 103 + .../react/socket/tests/UnixConnectorTest.php | 64 + .../vendor/react/socket/tests/UnixServerTest.php | 283 + .../SiteAssets/php/vendor/react/stream/.gitignore | 2 + .../SiteAssets/php/vendor/react/stream/.travis.yml | 50 + .../php/vendor/react/stream/CHANGELOG.md | 377 ++ .../SiteAssets/php/vendor/react/stream/LICENSE | 19 + .../SiteAssets/php/vendor/react/stream/README.md | 1224 ++++ .../php/vendor/react/stream/composer.json | 25 + .../php/vendor/react/stream/examples/01-http.php | 40 + .../php/vendor/react/stream/examples/02-https.php | 40 + .../php/vendor/react/stream/examples/11-cat.php | 28 + .../stream/examples/91-benchmark-throughput.php | 62 + .../php/vendor/react/stream/phpunit.xml.dist | 25 + .../vendor/react/stream/src/CompositeStream.php | 82 + .../react/stream/src/DuplexResourceStream.php | 224 + .../react/stream/src/DuplexStreamInterface.php | 39 + .../react/stream/src/ReadableResourceStream.php | 177 + .../react/stream/src/ReadableStreamInterface.php | 362 ++ .../php/vendor/react/stream/src/ThroughStream.php | 190 + .../php/vendor/react/stream/src/Util.php | 75 + .../react/stream/src/WritableResourceStream.php | 171 + .../react/stream/src/WritableStreamInterface.php | 347 ++ .../php/vendor/react/stream/tests/CallableStub.php | 10 + .../react/stream/tests/CompositeStreamTest.php | 267 + .../tests/DuplexResourceStreamIntegrationTest.php | 352 ++ .../stream/tests/DuplexResourceStreamTest.php | 495 ++ .../react/stream/tests/EnforceBlockingWrapper.php | 35 + .../react/stream/tests/FunctionalInternetTest.php | 122 + .../stream/tests/ReadableResourceStreamTest.php | 372 ++ .../react/stream/tests/Stub/ReadableStreamStub.php | 61 + .../php/vendor/react/stream/tests/TestCase.php | 54 + .../react/stream/tests/ThroughStreamTest.php | 267 + .../php/vendor/react/stream/tests/UtilTest.php | 273 + .../stream/tests/WritableStreamResourceTest.php | 534 ++ .../php/vendor/symfony/http-foundation/.gitignore | 3 + .../symfony/http-foundation/AcceptHeader.php | 168 + .../symfony/http-foundation/AcceptHeaderItem.php | 209 + .../symfony/http-foundation/ApacheRequest.php | 43 + .../symfony/http-foundation/BinaryFileResponse.php | 359 ++ .../vendor/symfony/http-foundation/CHANGELOG.md | 159 + .../php/vendor/symfony/http-foundation/Cookie.php | 289 + .../Exception/ConflictingHeadersException.php | 21 + .../Exception/RequestExceptionInterface.php | 21 + .../Exception/SuspiciousOperationException.php | 20 + .../http-foundation/ExpressionRequestMatcher.php | 47 + .../File/Exception/AccessDeniedException.php | 28 + .../File/Exception/FileException.php | 21 + .../File/Exception/FileNotFoundException.php | 28 + .../File/Exception/UnexpectedTypeException.php | 20 + .../File/Exception/UploadException.php | 21 + .../vendor/symfony/http-foundation/File/File.php | 136 + .../File/MimeType/ExtensionGuesser.php | 94 + .../File/MimeType/ExtensionGuesserInterface.php | 27 + .../File/MimeType/FileBinaryMimeTypeGuesser.php | 85 + .../File/MimeType/FileinfoMimeTypeGuesser.php | 69 + .../File/MimeType/MimeTypeExtensionGuesser.php | 808 +++ .../File/MimeType/MimeTypeGuesser.php | 142 + .../File/MimeType/MimeTypeGuesserInterface.php | 35 + .../vendor/symfony/http-foundation/File/Stream.php | 28 + .../symfony/http-foundation/File/UploadedFile.php | 266 + .../php/vendor/symfony/http-foundation/FileBag.php | 144 + .../vendor/symfony/http-foundation/HeaderBag.php | 331 ++ .../php/vendor/symfony/http-foundation/IpUtils.php | 156 + .../symfony/http-foundation/JsonResponse.php | 220 + .../php/vendor/symfony/http-foundation/LICENSE | 19 + .../symfony/http-foundation/ParameterBag.php | 234 + .../php/vendor/symfony/http-foundation/README.md | 14 + .../symfony/http-foundation/RedirectResponse.php | 109 + .../php/vendor/symfony/http-foundation/Request.php | 2154 +++++++ .../symfony/http-foundation/RequestMatcher.php | 178 + .../http-foundation/RequestMatcherInterface.php | 27 + .../symfony/http-foundation/RequestStack.php | 103 + .../vendor/symfony/http-foundation/Response.php | 1298 +++++ .../symfony/http-foundation/ResponseHeaderBag.php | 340 ++ .../vendor/symfony/http-foundation/ServerBag.php | 102 + .../Session/Attribute/AttributeBag.php | 148 + .../Session/Attribute/AttributeBagInterface.php | 72 + .../Session/Attribute/NamespacedAttributeBag.php | 153 + .../Session/Flash/AutoExpireFlashBag.php | 161 + .../http-foundation/Session/Flash/FlashBag.php | 152 + .../Session/Flash/FlashBagInterface.php | 93 + .../symfony/http-foundation/Session/Session.php | 273 + .../Session/SessionBagInterface.php | 46 + .../http-foundation/Session/SessionBagProxy.php | 82 + .../http-foundation/Session/SessionInterface.php | 180 + .../Storage/Handler/AbstractSessionHandler.php | 168 + .../Storage/Handler/MemcacheSessionHandler.php | 120 + .../Storage/Handler/MemcachedSessionHandler.php | 124 + .../Storage/Handler/MongoDbSessionHandler.php | 255 + .../Storage/Handler/NativeFileSessionHandler.php | 55 + .../Storage/Handler/NativeSessionHandler.php | 24 + .../Session/Storage/Handler/NullSessionHandler.php | 76 + .../Session/Storage/Handler/PdoSessionHandler.php | 910 +++ .../Storage/Handler/StrictSessionHandler.php | 103 + .../Storage/Handler/WriteCheckSessionHandler.php | 92 + .../Session/Storage/MetadataBag.php | 168 + .../Session/Storage/MockArraySessionStorage.php | 256 + .../Session/Storage/MockFileSessionStorage.php | 152 + .../Session/Storage/NativeSessionStorage.php | 445 ++ .../Session/Storage/PhpBridgeSessionStorage.php | 59 + .../Session/Storage/Proxy/AbstractProxy.php | 122 + .../Session/Storage/Proxy/NativeProxy.php | 40 + .../Session/Storage/Proxy/SessionHandlerProxy.php | 85 + .../Session/Storage/SessionStorageInterface.php | 137 + .../symfony/http-foundation/StreamedResponse.php | 144 + .../http-foundation/Tests/AcceptHeaderItemTest.php | 113 + .../http-foundation/Tests/AcceptHeaderTest.php | 103 + .../http-foundation/Tests/ApacheRequestTest.php | 93 + .../Tests/BinaryFileResponseTest.php | 352 ++ .../symfony/http-foundation/Tests/CookieTest.php | 223 + .../Tests/ExpressionRequestMatcherTest.php | 69 + .../http-foundation/Tests/File/FakeFile.php | 45 + .../http-foundation/Tests/File/FileTest.php | 180 + .../Tests/File/Fixtures/.unknownextension | 1 + .../Tests/File/Fixtures/directory/.empty | 0 .../Tests/File/Fixtures/other-file.example | 0 .../http-foundation/Tests/File/Fixtures/test | Bin 0 -> 35 bytes .../http-foundation/Tests/File/Fixtures/test.gif | Bin 0 -> 35 bytes .../Tests/File/MimeType/MimeTypeTest.php | 90 + .../Tests/File/UploadedFileTest.php | 273 + .../symfony/http-foundation/Tests/FileBagTest.php | 175 + .../http-foundation/Tests/HeaderBagTest.php | 205 + .../symfony/http-foundation/Tests/IpUtilsTest.php | 104 + .../http-foundation/Tests/JsonResponseTest.php | 257 + .../http-foundation/Tests/ParameterBagTest.php | 194 + .../http-foundation/Tests/RedirectResponseTest.php | 97 + .../http-foundation/Tests/RequestMatcherTest.php | 151 + .../http-foundation/Tests/RequestStackTest.php | 70 + .../symfony/http-foundation/Tests/RequestTest.php | 2329 ++++++++ .../Tests/ResponseHeaderBagTest.php | 363 ++ .../symfony/http-foundation/Tests/ResponseTest.php | 1003 ++++ .../http-foundation/Tests/ResponseTestCase.php | 89 + .../http-foundation/Tests/ServerBagTest.php | 170 + .../Tests/Session/Attribute/AttributeBagTest.php | 186 + .../Attribute/NamespacedAttributeBagTest.php | 182 + .../Tests/Session/Flash/AutoExpireFlashBagTest.php | 161 + .../Tests/Session/Flash/FlashBagTest.php | 132 + .../http-foundation/Tests/Session/SessionTest.php | 242 + .../Storage/Handler/AbstractSessionHandlerTest.php | 61 + .../Session/Storage/Handler/Fixtures/common.inc | 151 + .../Handler/Fixtures/empty_destroys.expected | 17 + .../Storage/Handler/Fixtures/empty_destroys.php | 8 + .../Storage/Handler/Fixtures/read_only.expected | 14 + .../Session/Storage/Handler/Fixtures/read_only.php | 8 + .../Storage/Handler/Fixtures/regenerate.expected | 24 + .../Storage/Handler/Fixtures/regenerate.php | 10 + .../Storage/Handler/Fixtures/storage.expected | 20 + .../Session/Storage/Handler/Fixtures/storage.php | 24 + .../Storage/Handler/Fixtures/with_cookie.expected | 15 + .../Storage/Handler/Fixtures/with_cookie.php | 8 + .../Fixtures/with_cookie_and_session.expected | 24 + .../Handler/Fixtures/with_cookie_and_session.php | 13 + .../Storage/Handler/MemcacheSessionHandlerTest.php | 135 + .../Handler/MemcachedSessionHandlerTest.php | 139 + .../Storage/Handler/MongoDbSessionHandlerTest.php | 333 ++ .../Handler/NativeFileSessionHandlerTest.php | 77 + .../Storage/Handler/NativeSessionHandlerTest.php | 38 + .../Storage/Handler/NullSessionHandlerTest.php | 59 + .../Storage/Handler/PdoSessionHandlerTest.php | 411 ++ .../Storage/Handler/StrictSessionHandlerTest.php | 189 + .../Handler/WriteCheckSessionHandlerTest.php | 97 + .../Tests/Session/Storage/MetadataBagTest.php | 139 + .../Storage/MockArraySessionStorageTest.php | 131 + .../Session/Storage/MockFileSessionStorageTest.php | 127 + .../Session/Storage/NativeSessionStorageTest.php | 277 + .../Storage/PhpBridgeSessionStorageTest.php | 96 + .../Session/Storage/Proxy/AbstractProxyTest.php | 113 + .../Session/Storage/Proxy/NativeProxyTest.php | 38 + .../Storage/Proxy/SessionHandlerProxyTest.php | 124 + .../http-foundation/Tests/StreamedResponseTest.php | 126 + .../Tests/schema/http-status-codes.rng | 31 + .../http-foundation/Tests/schema/iana-registry.rng | 198 + .../vendor/symfony/http-foundation/composer.json | 38 + .../symfony/http-foundation/phpunit.xml.dist | 31 + .../php/vendor/symfony/polyfill-mbstring/LICENSE | 19 + .../vendor/symfony/polyfill-mbstring/Mbstring.php | 791 +++ .../php/vendor/symfony/polyfill-mbstring/README.md | 13 + .../Resources/unidata/lowerCase.php | 1101 ++++ .../Resources/unidata/upperCase.php | 1109 ++++ .../vendor/symfony/polyfill-mbstring/bootstrap.php | 58 + .../vendor/symfony/polyfill-mbstring/composer.json | 34 + .../php/vendor/symfony/polyfill-php70/LICENSE | 19 + .../php/vendor/symfony/polyfill-php70/Php70.php | 74 + .../php/vendor/symfony/polyfill-php70/README.md | 28 + .../Resources/stubs/ArithmeticError.php | 5 + .../Resources/stubs/AssertionError.php | 5 + .../Resources/stubs/DivisionByZeroError.php | 5 + .../polyfill-php70/Resources/stubs/Error.php | 5 + .../polyfill-php70/Resources/stubs/ParseError.php | 5 + .../SessionUpdateTimestampHandlerInterface.php | 23 + .../polyfill-php70/Resources/stubs/TypeError.php | 5 + .../vendor/symfony/polyfill-php70/bootstrap.php | 27 + .../vendor/symfony/polyfill-php70/composer.json | 33 + .../php/vendor/symfony/process/.gitignore | 3 + .../php/vendor/symfony/process/CHANGELOG.md | 57 + .../process/Exception/ExceptionInterface.php | 21 + .../process/Exception/InvalidArgumentException.php | 21 + .../symfony/process/Exception/LogicException.php | 21 + .../process/Exception/ProcessFailedException.php | 54 + .../process/Exception/ProcessTimedOutException.php | 69 + .../symfony/process/Exception/RuntimeException.php | 21 + .../vendor/symfony/process/ExecutableFinder.php | 88 + .../php/vendor/symfony/process/InputStream.php | 92 + .../SiteAssets/php/vendor/symfony/process/LICENSE | 19 + .../vendor/symfony/process/PhpExecutableFinder.php | 94 + .../php/vendor/symfony/process/PhpProcess.php | 76 + .../vendor/symfony/process/Pipes/AbstractPipes.php | 168 + .../symfony/process/Pipes/PipesInterface.php | 67 + .../php/vendor/symfony/process/Pipes/UnixPipes.php | 150 + .../vendor/symfony/process/Pipes/WindowsPipes.php | 196 + .../php/vendor/symfony/process/Process.php | 1751 ++++++ .../php/vendor/symfony/process/ProcessBuilder.php | 280 + .../php/vendor/symfony/process/ProcessUtils.php | 123 + .../php/vendor/symfony/process/README.md | 13 + .../symfony/process/Tests/ExecutableFinderTest.php | 133 + .../symfony/process/Tests/NonStopableProcess.php | 47 + .../process/Tests/PhpExecutableFinderTest.php | 72 + .../symfony/process/Tests/PhpProcessTest.php | 48 + .../Tests/PipeStdinInStdoutStdErrStreamSelect.php | 72 + .../symfony/process/Tests/ProcessBuilderTest.php | 226 + .../process/Tests/ProcessFailedExceptionTest.php | 137 + .../vendor/symfony/process/Tests/ProcessTest.php | 1621 ++++++ .../symfony/process/Tests/ProcessUtilsTest.php | 53 + .../symfony/process/Tests/SignalListener.php | 21 + .../php/vendor/symfony/process/composer.json | 33 + .../php/vendor/symfony/process/phpunit.xml.dist | 30 + .../php/vendor/symfony/routing/.gitignore | 3 + .../vendor/symfony/routing/Annotation/Route.php | 144 + .../php/vendor/symfony/routing/CHANGELOG.md | 228 + .../php/vendor/symfony/routing/CompiledRoute.php | 169 + .../DependencyInjection/RoutingResolverPass.php | 49 + .../routing/Exception/ExceptionInterface.php | 21 + .../Exception/InvalidParameterException.php | 21 + .../Exception/MethodNotAllowedException.php | 41 + .../MissingMandatoryParametersException.php | 22 + .../routing/Exception/NoConfigurationException.php | 21 + .../Exception/ResourceNotFoundException.php | 23 + .../routing/Exception/RouteNotFoundException.php | 21 + .../ConfigurableRequirementsInterface.php | 55 + .../routing/Generator/Dumper/GeneratorDumper.php | 37 + .../Generator/Dumper/GeneratorDumperInterface.php | 39 + .../Generator/Dumper/PhpGeneratorDumper.php | 118 + .../symfony/routing/Generator/UrlGenerator.php | 321 ++ .../routing/Generator/UrlGeneratorInterface.php | 86 + .../SiteAssets/php/vendor/symfony/routing/LICENSE | 19 + .../routing/Loader/AnnotationClassLoader.php | 269 + .../routing/Loader/AnnotationDirectoryLoader.php | 93 + .../routing/Loader/AnnotationFileLoader.php | 142 + .../symfony/routing/Loader/ClosureLoader.php | 46 + .../Loader/Configurator/CollectionConfigurator.php | 81 + .../Loader/Configurator/ImportConfigurator.php | 49 + .../Loader/Configurator/RouteConfigurator.php | 34 + .../Loader/Configurator/RoutingConfigurator.php | 62 + .../Loader/Configurator/Traits/AddTrait.php | 55 + .../Loader/Configurator/Traits/RouteTrait.php | 131 + .../DependencyInjection/ServiceRouterLoader.php | 40 + .../symfony/routing/Loader/DirectoryLoader.php | 58 + .../symfony/routing/Loader/GlobFileLoader.php | 47 + .../symfony/routing/Loader/ObjectRouteLoader.php | 95 + .../symfony/routing/Loader/PhpFileLoader.php | 75 + .../symfony/routing/Loader/XmlFileLoader.php | 359 ++ .../symfony/routing/Loader/YamlFileLoader.php | 233 + .../routing/Loader/schema/routing/routing-1.0.xsd | 148 + .../routing/Matcher/Dumper/DumperCollection.php | 159 + .../symfony/routing/Matcher/Dumper/DumperRoute.php | 57 + .../routing/Matcher/Dumper/MatcherDumper.php | 37 + .../Matcher/Dumper/MatcherDumperInterface.php | 39 + .../routing/Matcher/Dumper/PhpMatcherDumper.php | 429 ++ .../Matcher/Dumper/StaticPrefixCollection.php | 238 + .../routing/Matcher/RedirectableUrlMatcher.php | 65 + .../Matcher/RedirectableUrlMatcherInterface.php | 31 + .../routing/Matcher/RequestMatcherInterface.php | 39 + .../routing/Matcher/TraceableUrlMatcher.php | 141 + .../vendor/symfony/routing/Matcher/UrlMatcher.php | 252 + .../routing/Matcher/UrlMatcherInterface.php | 41 + .../php/vendor/symfony/routing/README.md | 13 + .../php/vendor/symfony/routing/RequestContext.php | 336 ++ .../routing/RequestContextAwareInterface.php | 27 + .../php/vendor/symfony/routing/Route.php | 558 ++ .../php/vendor/symfony/routing/RouteCollection.php | 280 + .../symfony/routing/RouteCollectionBuilder.php | 380 ++ .../php/vendor/symfony/routing/RouteCompiler.php | 316 ++ .../symfony/routing/RouteCompilerInterface.php | 30 + .../php/vendor/symfony/routing/Router.php | 388 ++ .../php/vendor/symfony/routing/RouterInterface.php | 32 + .../symfony/routing/Tests/Annotation/RouteTest.php | 50 + .../symfony/routing/Tests/CompiledRouteTest.php | 27 + .../RoutingResolverPassTest.php | 36 + .../Fixtures/AnnotatedClasses/AbstractClass.php | 16 + .../Tests/Fixtures/AnnotatedClasses/BarClass.php | 19 + .../Tests/Fixtures/AnnotatedClasses/BazClass.php | 19 + .../Tests/Fixtures/AnnotatedClasses/FooClass.php | 16 + .../Tests/Fixtures/AnnotatedClasses/FooTrait.php | 13 + .../routing/Tests/Fixtures/CustomCompiledRoute.php | 18 + .../routing/Tests/Fixtures/CustomRouteCompiler.php | 26 + .../routing/Tests/Fixtures/CustomXmlFileLoader.php | 26 + .../AnonymousClassInTrait.php | 24 + .../OtherAnnotatedClasses/NoStartTagClass.php | 3 + .../OtherAnnotatedClasses/VariadicClass.php | 19 + .../Tests/Fixtures/RedirectableUrlMatcher.php | 30 + .../symfony/routing/Tests/Fixtures/annotated.php | 0 .../symfony/routing/Tests/Fixtures/bad_format.yml | 3 + .../vendor/symfony/routing/Tests/Fixtures/bar.xml | 0 .../Fixtures/controller/import__controller.xml | 10 + .../Fixtures/controller/import__controller.yml | 4 + .../Fixtures/controller/import_controller.xml | 8 + .../Fixtures/controller/import_controller.yml | 3 + .../controller/import_override_defaults.xml | 10 + .../controller/import_override_defaults.yml | 5 + .../Fixtures/controller/override_defaults.xml | 10 + .../Fixtures/controller/override_defaults.yml | 5 + .../routing/Tests/Fixtures/controller/routing.xml | 14 + .../routing/Tests/Fixtures/controller/routing.yml | 11 + .../Tests/Fixtures/directory/recurse/routes1.yml | 2 + .../Tests/Fixtures/directory/recurse/routes2.yml | 2 + .../routing/Tests/Fixtures/directory/routes3.yml | 2 + .../Tests/Fixtures/directory_import/import.yml | 3 + .../routing/Tests/Fixtures/dumper/url_matcher0.php | 37 + .../routing/Tests/Fixtures/dumper/url_matcher1.php | 318 ++ .../routing/Tests/Fixtures/dumper/url_matcher2.php | 380 ++ .../routing/Tests/Fixtures/dumper/url_matcher3.php | 55 + .../routing/Tests/Fixtures/dumper/url_matcher4.php | 112 + .../routing/Tests/Fixtures/dumper/url_matcher5.php | 209 + .../routing/Tests/Fixtures/dumper/url_matcher6.php | 213 + .../routing/Tests/Fixtures/dumper/url_matcher7.php | 249 + .../symfony/routing/Tests/Fixtures/empty.yml | 0 .../routing/Tests/Fixtures/file_resource.yml | 0 .../vendor/symfony/routing/Tests/Fixtures/foo.xml | 0 .../vendor/symfony/routing/Tests/Fixtures/foo1.xml | 0 .../symfony/routing/Tests/Fixtures/glob/bar.xml | 8 + .../symfony/routing/Tests/Fixtures/glob/bar.yml | 4 + .../symfony/routing/Tests/Fixtures/glob/baz.xml | 8 + .../symfony/routing/Tests/Fixtures/glob/baz.yml | 4 + .../Tests/Fixtures/glob/import_multiple.xml | 8 + .../Tests/Fixtures/glob/import_multiple.yml | 2 + .../routing/Tests/Fixtures/glob/import_single.xml | 8 + .../routing/Tests/Fixtures/glob/import_single.yml | 2 + .../routing/Tests/Fixtures/glob/php_dsl.php | 7 + .../routing/Tests/Fixtures/glob/php_dsl_bar.php | 12 + .../routing/Tests/Fixtures/glob/php_dsl_baz.php | 12 + .../symfony/routing/Tests/Fixtures/incomplete.yml | 2 + .../routing/Tests/Fixtures/list_defaults.xml | 20 + .../Tests/Fixtures/list_in_list_defaults.xml | 22 + .../Tests/Fixtures/list_in_map_defaults.xml | 22 + .../routing/Tests/Fixtures/list_null_values.xml | 22 + .../routing/Tests/Fixtures/map_defaults.xml | 20 + .../Tests/Fixtures/map_in_list_defaults.xml | 22 + .../routing/Tests/Fixtures/map_in_map_defaults.xml | 22 + .../routing/Tests/Fixtures/map_null_values.xml | 22 + .../symfony/routing/Tests/Fixtures/missing_id.xml | 8 + .../routing/Tests/Fixtures/missing_path.xml | 8 + .../routing/Tests/Fixtures/namespaceprefix.xml | 16 + .../Fixtures/nonesense_resource_plus_path.yml | 3 + .../Fixtures/nonesense_type_without_resource.yml | 3 + .../symfony/routing/Tests/Fixtures/nonvalid.xml | 10 + .../symfony/routing/Tests/Fixtures/nonvalid.yml | 1 + .../symfony/routing/Tests/Fixtures/nonvalid2.yml | 1 + .../routing/Tests/Fixtures/nonvalidkeys.yml | 3 + .../routing/Tests/Fixtures/nonvalidnode.xml | 8 + .../routing/Tests/Fixtures/nonvalidroute.xml | 12 + .../symfony/routing/Tests/Fixtures/null_values.xml | 12 + .../symfony/routing/Tests/Fixtures/php_dsl.php | 22 + .../symfony/routing/Tests/Fixtures/php_dsl_sub.php | 14 + .../routing/Tests/Fixtures/scalar_defaults.xml | 33 + .../routing/Tests/Fixtures/special_route_name.yml | 2 + .../routing/Tests/Fixtures/validpattern.php | 18 + .../routing/Tests/Fixtures/validpattern.xml | 15 + .../routing/Tests/Fixtures/validpattern.yml | 13 + .../routing/Tests/Fixtures/validresource.php | 18 + .../routing/Tests/Fixtures/validresource.xml | 13 + .../routing/Tests/Fixtures/validresource.yml | 8 + .../Tests/Fixtures/with_define_path_variable.php | 5 + .../symfony/routing/Tests/Fixtures/withdoctype.xml | 3 + .../Generator/Dumper/PhpGeneratorDumperTest.php | 181 + .../routing/Tests/Generator/UrlGeneratorTest.php | 724 +++ .../Tests/Loader/AbstractAnnotationLoaderTest.php | 33 + .../Tests/Loader/AnnotationClassLoaderTest.php | 255 + .../Tests/Loader/AnnotationDirectoryLoaderTest.php | 98 + .../Tests/Loader/AnnotationFileLoaderTest.php | 91 + .../routing/Tests/Loader/ClosureLoaderTest.php | 49 + .../routing/Tests/Loader/DirectoryLoaderTest.php | 74 + .../routing/Tests/Loader/GlobFileLoaderTest.php | 45 + .../routing/Tests/Loader/ObjectRouteLoaderTest.php | 123 + .../routing/Tests/Loader/PhpFileLoaderTest.php | 133 + .../routing/Tests/Loader/XmlFileLoaderTest.php | 385 ++ .../routing/Tests/Loader/YamlFileLoaderTest.php | 206 + .../Matcher/DumpedRedirectableUrlMatcherTest.php | 43 + .../routing/Tests/Matcher/DumpedUrlMatcherTest.php | 48 + .../Tests/Matcher/Dumper/DumperCollectionTest.php | 34 + .../Tests/Matcher/Dumper/PhpMatcherDumperTest.php | 459 ++ .../Matcher/Dumper/StaticPrefixCollectionTest.php | 175 + .../Tests/Matcher/RedirectableUrlMatcherTest.php | 124 + .../Tests/Matcher/TraceableUrlMatcherTest.php | 122 + .../routing/Tests/Matcher/UrlMatcherTest.php | 509 ++ .../symfony/routing/Tests/RequestContextTest.php | 160 + .../routing/Tests/RouteCollectionBuilderTest.php | 364 ++ .../symfony/routing/Tests/RouteCollectionTest.php | 305 + .../symfony/routing/Tests/RouteCompilerTest.php | 389 ++ .../php/vendor/symfony/routing/Tests/RouteTest.php | 258 + .../vendor/symfony/routing/Tests/RouterTest.php | 163 + .../php/vendor/symfony/routing/composer.json | 56 + .../php/vendor/symfony/routing/phpunit.xml.dist | 30 + .../assets/font-starcraft/css/font-starcraft.css | 61 + .../assets/font-starcraft/fonts/font-starcraft.eot | Bin 0 -> 48420 bytes .../assets/font-starcraft/fonts/font-starcraft.svg | 22 + .../assets/font-starcraft/fonts/font-starcraft.ttf | Bin 0 -> 48248 bytes .../font-starcraft/fonts/font-starcraft.woff | Bin 0 -> 39164 bytes .../assets/userfrosting/css/AdminLTE-skins-all.css | 1770 ++++++ .../core/assets/userfrosting/css/AdminLTE.css | 5894 ++++++++++++++++++++ .../assets/userfrosting/css/tablesorter-reflow.css | 61 + .../core/assets/userfrosting/css/uf-alerts.css | 23 + .../core/assets/userfrosting/css/uf-collection.css | 15 + .../userfrosting/css/uf-jqueryvalidation.css | 10 + .../core/assets/userfrosting/css/userfrosting.css | 204 + .../core/assets/userfrosting/favicons/README.md | 7 + .../favicons/android-chrome-144x144.png | Bin 0 -> 20991 bytes .../favicons/android-chrome-192x192.png | Bin 0 -> 29935 bytes .../favicons/android-chrome-256x256.png | Bin 0 -> 42828 bytes .../userfrosting/favicons/android-chrome-36x36.png | Bin 0 -> 3150 bytes .../favicons/android-chrome-384x384.png | Bin 0 -> 69092 bytes .../userfrosting/favicons/android-chrome-48x48.png | Bin 0 -> 4736 bytes .../favicons/android-chrome-512x512.png | Bin 0 -> 96438 bytes .../userfrosting/favicons/android-chrome-72x72.png | Bin 0 -> 8367 bytes .../userfrosting/favicons/android-chrome-96x96.png | Bin 0 -> 12470 bytes .../apple-touch-icon-114x114-precomposed.png | Bin 0 -> 14098 bytes .../favicons/apple-touch-icon-114x114.png | Bin 0 -> 12666 bytes .../apple-touch-icon-120x120-precomposed.png | Bin 0 -> 15188 bytes .../favicons/apple-touch-icon-120x120.png | Bin 0 -> 13650 bytes .../apple-touch-icon-144x144-precomposed.png | Bin 0 -> 19338 bytes .../favicons/apple-touch-icon-144x144.png | Bin 0 -> 17393 bytes .../apple-touch-icon-152x152-precomposed.png | Bin 0 -> 20475 bytes .../favicons/apple-touch-icon-152x152.png | Bin 0 -> 18420 bytes .../apple-touch-icon-180x180-precomposed.png | Bin 0 -> 25283 bytes .../favicons/apple-touch-icon-180x180.png | Bin 0 -> 22957 bytes .../apple-touch-icon-57x57-precomposed.png | Bin 0 -> 5279 bytes .../favicons/apple-touch-icon-57x57.png | Bin 0 -> 4660 bytes .../apple-touch-icon-60x60-precomposed.png | Bin 0 -> 5706 bytes .../favicons/apple-touch-icon-60x60.png | Bin 0 -> 5017 bytes .../apple-touch-icon-72x72-precomposed.png | Bin 0 -> 7498 bytes .../favicons/apple-touch-icon-72x72.png | Bin 0 -> 6615 bytes .../apple-touch-icon-76x76-precomposed.png | Bin 0 -> 8080 bytes .../favicons/apple-touch-icon-76x76.png | Bin 0 -> 7180 bytes .../favicons/apple-touch-icon-precomposed.png | Bin 0 -> 25283 bytes .../userfrosting/favicons/apple-touch-icon.png | Bin 0 -> 22957 bytes .../assets/userfrosting/favicons/favicon-16x16.png | Bin 0 -> 1374 bytes .../assets/userfrosting/favicons/favicon-32x32.png | Bin 0 -> 2678 bytes .../core/assets/userfrosting/favicons/favicon.ico | Bin 0 -> 15086 bytes .../core/assets/userfrosting/favicons/ieconfig.xml | 12 + .../assets/userfrosting/favicons/manifest.json | 51 + .../userfrosting/favicons/mstile-144x144.png | Bin 0 -> 20296 bytes .../userfrosting/favicons/mstile-150x150.png | Bin 0 -> 18368 bytes .../userfrosting/favicons/mstile-310x150.png | Bin 0 -> 19511 bytes .../userfrosting/favicons/mstile-310x310.png | Bin 0 -> 46338 bytes .../assets/userfrosting/favicons/mstile-70x70.png | Bin 0 -> 12070 bytes .../userfrosting/favicons/safari-pinned-tab.svg | 246 + .../core/assets/userfrosting/images/cupcake.png | Bin 0 -> 19738 bytes .../core/assets/userfrosting/images/logo.svg | 514 ++ .../core/assets/userfrosting/js/AdminLTE-custom.js | 117 + .../core/assets/userfrosting/js/AdminLTE.js | 763 +++ .../core/assets/userfrosting/js/attrchange.js | 124 + .../js/fortress-jqueryvalidation-methods.js | 57 + .../assets/userfrosting/js/handlebars-helpers.js | 119 + .../core/assets/userfrosting/js/query-string.js | 65 + .../js/tablesorter/widget-sort2Hash.js | 271 + .../core/assets/userfrosting/js/uf-alerts.js | 289 + .../core/assets/userfrosting/js/uf-captcha.js | 13 + .../core/assets/userfrosting/js/uf-collection.js | 345 ++ .../core/assets/userfrosting/js/uf-copy.js | 50 + .../core/assets/userfrosting/js/uf-form.js | 297 + .../core/assets/userfrosting/js/uf-init.js | 26 + .../userfrosting/js/uf-jqueryvalidation-config.js | 49 + .../core/assets/userfrosting/js/uf-modal.js | 192 + .../core/assets/userfrosting/js/uf-table.js | 704 +++ .../userfrosting/js/uf-tablesorter-parsers.js | 53 + main/app/sprinkles/core/bower.json | 47 + main/app/sprinkles/core/composer.json | 44 + main/app/sprinkles/core/config/default.php | 182 + main/app/sprinkles/core/config/dev.php | 30 + main/app/sprinkles/core/config/production.php | 40 + main/app/sprinkles/core/config/testing.php | 23 + main/app/sprinkles/core/extra/adjectives.php | 1221 ++++ main/app/sprinkles/core/extra/nouns.php | 90 + main/app/sprinkles/core/locale/ar/errors.php | 51 + main/app/sprinkles/core/locale/ar/messages.php | 112 + main/app/sprinkles/core/locale/ar/validate.php | 25 + main/app/sprinkles/core/locale/de_DE/errors.php | 53 + main/app/sprinkles/core/locale/de_DE/messages.php | 123 + main/app/sprinkles/core/locale/de_DE/validate.php | 32 + main/app/sprinkles/core/locale/en_US/errors.php | 51 + main/app/sprinkles/core/locale/en_US/messages.php | 120 + main/app/sprinkles/core/locale/en_US/validate.php | 33 + main/app/sprinkles/core/locale/es_ES/errors.php | 51 + main/app/sprinkles/core/locale/es_ES/messages.php | 115 + main/app/sprinkles/core/locale/es_ES/validate.php | 35 + main/app/sprinkles/core/locale/fa/errors.php | 52 + main/app/sprinkles/core/locale/fa/messages.php | 110 + main/app/sprinkles/core/locale/fa/validate.php | 31 + main/app/sprinkles/core/locale/fr_FR/errors.php | 51 + main/app/sprinkles/core/locale/fr_FR/messages.php | 105 + main/app/sprinkles/core/locale/fr_FR/validate.php | 33 + main/app/sprinkles/core/locale/it_IT/errors.php | 53 + main/app/sprinkles/core/locale/it_IT/messages.php | 123 + main/app/sprinkles/core/locale/it_IT/validate.php | 32 + main/app/sprinkles/core/locale/pt_PT/errors.php | 51 + main/app/sprinkles/core/locale/pt_PT/messages.php | 102 + main/app/sprinkles/core/locale/pt_PT/validate.php | 25 + main/app/sprinkles/core/locale/ru_RU/errors.php | 51 + main/app/sprinkles/core/locale/ru_RU/messages.php | 120 + main/app/sprinkles/core/locale/ru_RU/validate.php | 33 + main/app/sprinkles/core/locale/th_TH/errors.php | 51 + main/app/sprinkles/core/locale/th_TH/messages.php | 102 + main/app/sprinkles/core/locale/th_TH/validate.php | 25 + main/app/sprinkles/core/locale/valitron/ar.php | 28 + main/app/sprinkles/core/locale/valitron/de.php | 33 + main/app/sprinkles/core/locale/valitron/el.php | 34 + main/app/sprinkles/core/locale/valitron/en.php | 34 + main/app/sprinkles/core/locale/valitron/es.php | 34 + main/app/sprinkles/core/locale/valitron/fr.php | 34 + main/app/sprinkles/core/locale/valitron/id.php | 33 + main/app/sprinkles/core/locale/valitron/it.php | 31 + main/app/sprinkles/core/locale/valitron/ja.php | 33 + main/app/sprinkles/core/locale/valitron/lv.php | 31 + main/app/sprinkles/core/locale/valitron/pt-br.php | 28 + main/app/sprinkles/core/locale/valitron/ro.php | 33 + main/app/sprinkles/core/locale/valitron/ru.php | 33 + main/app/sprinkles/core/locale/valitron/th.php | 34 + main/app/sprinkles/core/locale/valitron/zh-cn.php | 28 + main/app/sprinkles/core/locale/valitron/zh-tw.php | 28 + main/app/sprinkles/core/locale/zh_CN/errors.php | 49 + main/app/sprinkles/core/locale/zh_CN/messages.php | 105 + main/app/sprinkles/core/locale/zh_CN/validate.php | 29 + main/app/sprinkles/core/routes/routes.php | 24 + main/app/sprinkles/core/schema/.gitkeep | 0 main/app/sprinkles/core/src/Alert/AlertStream.php | 144 + .../sprinkles/core/src/Alert/CacheAlertStream.php | 84 + .../core/src/Alert/SessionAlertStream.php | 70 + .../core/src/Controller/CoreController.php | 90 + .../core/src/Controller/SimpleController.php | 36 + main/app/sprinkles/core/src/Core.php | 121 + main/app/sprinkles/core/src/Database/Builder.php | 210 + .../core/src/Database/DatabaseInvalidException.php | 20 + .../src/Database/Migrations/v400/SessionsTable.php | 48 + .../Database/Migrations/v400/ThrottlesTable.php | 52 + .../Database/Models/Concerns/HasRelationships.php | 278 + .../sprinkles/core/src/Database/Models/Model.php | 140 + .../core/src/Database/Models/Throttle.php | 36 + .../Relations/BelongsToManyConstrained.php | 122 + .../Database/Relations/BelongsToManyThrough.php | 232 + .../src/Database/Relations/BelongsToManyUnique.php | 22 + .../src/Database/Relations/Concerns/Syncable.php | 132 + .../src/Database/Relations/Concerns/Unique.php | 563 ++ .../src/Database/Relations/HasManySyncable.php | 22 + .../src/Database/Relations/MorphManySyncable.php | 22 + .../src/Database/Relations/MorphToManyUnique.php | 22 + .../core/src/Error/ExceptionHandlerManager.php | 93 + .../core/src/Error/Handler/ExceptionHandler.php | 275 + .../Error/Handler/ExceptionHandlerInterface.php | 32 + .../src/Error/Handler/HttpExceptionHandler.php | 64 + .../src/Error/Handler/NotFoundExceptionHandler.php | 38 + .../Error/Handler/PhpMailerExceptionHandler.php | 30 + .../core/src/Error/Renderer/ErrorRenderer.php | 64 + .../src/Error/Renderer/ErrorRendererInterface.php | 29 + .../core/src/Error/Renderer/HtmlRenderer.php | 151 + .../core/src/Error/Renderer/JsonRenderer.php | 57 + .../core/src/Error/Renderer/PlainTextRenderer.php | 65 + .../core/src/Error/Renderer/WhoopsRenderer.php | 712 +++ .../core/src/Error/Renderer/XmlRenderer.php | 48 + main/app/sprinkles/core/src/Facades/Debug.php | 28 + main/app/sprinkles/core/src/Facades/Translator.php | 28 + .../src/Http/Concerns/DeterminesContentType.php | 76 + .../app/sprinkles/core/src/Log/DatabaseHandler.php | 53 + main/app/sprinkles/core/src/Log/MixedFormatter.php | 59 + .../app/sprinkles/core/src/Mail/EmailRecipient.php | 136 + main/app/sprinkles/core/src/Mail/MailMessage.php | 186 + main/app/sprinkles/core/src/Mail/Mailer.php | 204 + .../sprinkles/core/src/Mail/StaticMailMessage.php | 78 + .../sprinkles/core/src/Mail/TwigMailMessage.php | 93 + main/app/sprinkles/core/src/Model/UFModel.php | 27 + main/app/sprinkles/core/src/Router.php | 101 + .../core/src/ServicesProvider/ServicesProvider.php | 618 ++ main/app/sprinkles/core/src/Sprunje/Sprunje.php | 566 ++ .../sprinkles/core/src/Throttle/ThrottleRule.php | 140 + main/app/sprinkles/core/src/Throttle/Throttler.php | 178 + .../core/src/Throttle/ThrottlerException.php | 18 + main/app/sprinkles/core/src/Twig/CacheHelper.php | 58 + main/app/sprinkles/core/src/Twig/CoreExtension.php | 124 + .../core/src/Util/BadClassNameException.php | 18 + main/app/sprinkles/core/src/Util/Captcha.php | 159 + .../sprinkles/core/src/Util/CheckEnvironment.php | 340 ++ main/app/sprinkles/core/src/Util/ClassMapper.php | 94 + .../sprinkles/core/src/Util/EnvironmentInfo.php | 68 + .../sprinkles/core/src/Util/ShutdownHandler.php | 167 + main/app/sprinkles/core/src/Util/Util.php | 173 + .../sprinkles/core/templates/forms/csrf.html.twig | 2 + main/app/sprinkles/core/templates/mail/.gitkeep | 0 .../core/templates/modals/modal.html.twig | 27 + .../core/templates/navigation/breadcrumb.html.twig | 4 + .../core/templates/navigation/main-nav.html.twig | 31 + .../sprinkles/core/templates/pages/about.html.twig | 173 + .../core/templates/pages/abstract/base.html.twig | 103 + .../templates/pages/abstract/default.html.twig | 45 + .../core/templates/pages/abstract/error.html.twig | 32 + .../templates/pages/abstract/mainsite.html.twig | 93 + .../core/templates/pages/error/400.html.twig | 9 + .../core/templates/pages/error/404.html.twig | 16 + .../templates/pages/error/config-errors.html.twig | 22 + .../sprinkles/core/templates/pages/index.html.twig | 112 + .../sprinkles/core/templates/pages/legal.html.twig | 12 + .../core/templates/pages/partials/alerts.html.twig | 13 + .../templates/pages/partials/analytics.html.twig | 15 + .../core/templates/pages/partials/config.js.twig | 13 + .../templates/pages/partials/favicons.html.twig | 39 + .../core/templates/pages/partials/footer.html.twig | 8 + .../core/templates/pages/partials/legal.html.twig | 95 + .../core/templates/pages/partials/page.js.twig | 4 + .../templates/pages/partials/privacy.html.twig | 35 + .../core/templates/pages/privacy.html.twig | 12 + .../templates/tables/table-paginated.html.twig | 59 + .../templates/tables/table-tool-menu.html.twig | 25 + .../core/tests/Integration/DatabaseTests.php | 1315 +++++ .../core/tests/Unit/BelongsToManyThroughTest.php | 103 + .../core/tests/Unit/DatabaseSyncableTest.php | 119 + main/app/sprinkles/core/tests/Unit/SprunjeTest.php | 100 + main/app/sprinkles/extend-user/.gitignore | 3 + main/app/sprinkles/extend-user/README.md | 29 + main/app/sprinkles/extend-user/composer.json | 22 + main/app/sprinkles/extend-user/routes/member.php | 7 + .../extend-user/schema/requests/user/create.yaml | 86 + .../schema/requests/user/edit-info.yaml | 50 + .../src/Controller/MemberController.php | 123 + .../src/Database/Migrations/v400/MembersTable.php | 34 + .../extend-user/src/Database/Models/Member.php | 124 + .../extend-user/src/Database/Models/MemberAux.php | 20 + .../src/Database/Scopes/MemberAuxScope.php | 36 + .../src/ServicesProvider/ServicesProvider.php | 26 + .../extend-user/templates/forms/user.html.twig | 145 + .../extend-user/templates/pages/user.html.twig | 11 + main/app/system/Bakery/Bakery.php | 166 + main/app/system/Bakery/BaseCommand.php | 58 + main/app/system/Bakery/Command/Bake.php | 77 + main/app/system/Bakery/Command/BuildAssets.php | 180 + main/app/system/Bakery/Command/ClearCache.php | 95 + main/app/system/Bakery/Command/Debug.php | 185 + main/app/system/Bakery/Command/Migrate.php | 48 + main/app/system/Bakery/Command/MigrateRefresh.php | 52 + main/app/system/Bakery/Command/MigrateReset.php | 49 + main/app/system/Bakery/Command/MigrateRollback.php | 51 + main/app/system/Bakery/Command/Setup.php | 223 + main/app/system/Bakery/Command/Test.php | 56 + main/app/system/Bakery/DatabaseTest.php | 52 + main/app/system/Bakery/Migration.php | 64 + main/app/system/Bakery/Migrator.php | 584 ++ .../Database/Migrations/v410/MigrationTable.php | 59 + main/app/system/Database/Model/Migrations.php | 55 + main/app/system/Facade.php | 247 + main/app/system/ServicesProvider.php | 104 + main/app/system/SlimAppEvent.php | 29 + main/app/system/Sprinkle/Sprinkle.php | 56 + main/app/system/Sprinkle/SprinkleManager.php | 236 + main/app/system/UserFrosting.php | 187 + main/app/tests/DatabaseTransactions.php | 48 + main/app/tests/TestCase.php | 236 + main/app/tests/Unit/ExampleTest.php | 19 + main/bakery | 34 + main/build/before_install.sh | 39 + main/build/gulpfile.js | 183 + main/build/package-lock.json | 5512 ++++++++++++++++++ main/build/package.json | 33 + main/composer.json | 60 + main/docker-compose.yml | 56 + main/docker/README.md | 35 + main/docker/nginx/Dockerfile | 2 + main/docker/nginx/default.conf | 28 + main/docker/node/Dockerfile | 4 + main/docker/php/Dockerfile | 11 + main/phpunit.xml | 21 + main/public/.htaccess | 185 + main/public/index.php | 23 + main/screenshots/login.png | Bin 0 -> 3464 bytes main/screenshots/permissions.png | Bin 0 -> 31896 bytes main/screenshots/users.png | Bin 0 -> 43380 bytes main/sponsors/nextgi.png | Bin 0 -> 5673 bytes main/sponsors/usor.png | Bin 0 -> 12098 bytes main/webserver-configs/htaccess.txt | 185 + main/webserver-configs/nginx.conf | 127 + main/webserver-configs/web.config | 92 + 1611 files changed, 193194 insertions(+) create mode 100755 main/.github/CONTRIBUTING.md create mode 100755 main/.github/ISSUE_TEMPLATE.md create mode 100755 main/.travis.yml create mode 100755 main/CHANGELOG.md create mode 100755 main/LICENSE.md create mode 100755 main/README.md create mode 100755 main/STYLE-GUIDE.md create mode 100755 main/app/.env.example create mode 100755 main/app/.htaccess create mode 100755 main/app/cache/.gitkeep create mode 100755 main/app/defines.php create mode 100755 main/app/logs/.gitkeep create mode 100755 main/app/sessions/.gitkeep create mode 100755 main/app/sprinkles.example.json create mode 100755 main/app/sprinkles/ConfigManager/CHANGELOG.md create mode 100755 main/app/sprinkles/ConfigManager/LICENSE create mode 100755 main/app/sprinkles/ConfigManager/README.md create mode 100755 main/app/sprinkles/ConfigManager/asset-bundles.json create mode 100755 main/app/sprinkles/ConfigManager/assets/js/ConfigManager.js create mode 100755 main/app/sprinkles/ConfigManager/composer.json create mode 100755 main/app/sprinkles/ConfigManager/locale/en_US/AdminLTE.php create mode 100755 main/app/sprinkles/ConfigManager/locale/en_US/ConfigManager.php create mode 100755 main/app/sprinkles/ConfigManager/locale/fr_FR/AdminLTE.php create mode 100755 main/app/sprinkles/ConfigManager/locale/fr_FR/ConfigManager.php create mode 100755 main/app/sprinkles/ConfigManager/routes/ConfigManager.php create mode 100755 main/app/sprinkles/ConfigManager/schema/config/AdminLTE.json create mode 100755 main/app/sprinkles/ConfigManager/schema/config/site.json create mode 100755 main/app/sprinkles/ConfigManager/src/ConfigManager.php create mode 100755 main/app/sprinkles/ConfigManager/src/Controller/ConfigManagerController.php create mode 100755 main/app/sprinkles/ConfigManager/src/Database/Migrations/v100/SettingsTable.php create mode 100755 main/app/sprinkles/ConfigManager/src/Database/Migrations/v101/SettingsPermissions.php create mode 100755 main/app/sprinkles/ConfigManager/src/Database/Models/Config.php create mode 100755 main/app/sprinkles/ConfigManager/src/ServicesProvider/ServicesProvider.php create mode 100755 main/app/sprinkles/ConfigManager/src/Util/ConfigManager.php create mode 100755 main/app/sprinkles/ConfigManager/templates/pages/ConfigManager.html.twig create mode 100755 main/app/sprinkles/FormGenerator/.gitignore create mode 100755 main/app/sprinkles/FormGenerator/CHANGELOG.md create mode 100755 main/app/sprinkles/FormGenerator/LICENSE create mode 100755 main/app/sprinkles/FormGenerator/README.md create mode 100755 main/app/sprinkles/FormGenerator/asset-bundles.json create mode 100755 main/app/sprinkles/FormGenerator/assets/js/widget-formGenerator.js create mode 100755 main/app/sprinkles/FormGenerator/bower.json create mode 100755 main/app/sprinkles/FormGenerator/composer.json create mode 100755 main/app/sprinkles/FormGenerator/locale/en_US/FormGenerator.php create mode 100755 main/app/sprinkles/FormGenerator/locale/fr_FR/FormGenerator.php create mode 100755 main/app/sprinkles/FormGenerator/routes/FormGenerator.php create mode 100755 main/app/sprinkles/FormGenerator/src/Controller/FormGeneratorController.php create mode 100755 main/app/sprinkles/FormGenerator/src/Element/Alert.php create mode 100755 main/app/sprinkles/FormGenerator/src/Element/BaseInput.php create mode 100755 main/app/sprinkles/FormGenerator/src/Element/Checkbox.php create mode 100755 main/app/sprinkles/FormGenerator/src/Element/Hidden.php create mode 100755 main/app/sprinkles/FormGenerator/src/Element/InputInterface.php create mode 100755 main/app/sprinkles/FormGenerator/src/Element/Select.php create mode 100755 main/app/sprinkles/FormGenerator/src/Element/Text.php create mode 100755 main/app/sprinkles/FormGenerator/src/Element/Textarea.php create mode 100755 main/app/sprinkles/FormGenerator/src/Form.php create mode 100755 main/app/sprinkles/FormGenerator/templates/FormGenerator/FormGenerator.html.twig create mode 100755 main/app/sprinkles/FormGenerator/templates/FormGenerator/confirm.html.twig create mode 100755 main/app/sprinkles/FormGenerator/templates/FormGenerator/macros/alert.html.twig create mode 100755 main/app/sprinkles/FormGenerator/templates/FormGenerator/macros/checkbox.html.twig create mode 100755 main/app/sprinkles/FormGenerator/templates/FormGenerator/macros/hidden.html.twig create mode 100755 main/app/sprinkles/FormGenerator/templates/FormGenerator/macros/select.html.twig create mode 100755 main/app/sprinkles/FormGenerator/templates/FormGenerator/macros/text.html.twig create mode 100755 main/app/sprinkles/FormGenerator/templates/FormGenerator/macros/textarea.html.twig create mode 100755 main/app/sprinkles/FormGenerator/templates/FormGenerator/modal-large.html.twig create mode 100755 main/app/sprinkles/FormGenerator/templates/FormGenerator/modal.html.twig create mode 100755 main/app/sprinkles/FormGenerator/templates/FormGenerator/typehead.html.twig create mode 100755 main/app/sprinkles/FormGenerator/tests/Unit/FormGeneratorTest.php create mode 100755 main/app/sprinkles/FormGenerator/tests/Unit/data/bad.json create mode 100755 main/app/sprinkles/FormGenerator/tests/Unit/data/good.json create mode 100755 main/app/sprinkles/account/asset-bundles.json create mode 100755 main/app/sprinkles/account/assets/userfrosting/js/pages/account-settings.js create mode 100755 main/app/sprinkles/account/assets/userfrosting/js/pages/forgot-password.js create mode 100755 main/app/sprinkles/account/assets/userfrosting/js/pages/register.js create mode 100755 main/app/sprinkles/account/assets/userfrosting/js/pages/resend-verification.js create mode 100755 main/app/sprinkles/account/assets/userfrosting/js/pages/set-or-reset-password.js create mode 100755 main/app/sprinkles/account/assets/userfrosting/js/pages/sign-in.js create mode 100755 main/app/sprinkles/account/bower.json create mode 100755 main/app/sprinkles/account/composer.json create mode 100755 main/app/sprinkles/account/config/default.php create mode 100755 main/app/sprinkles/account/config/production.php create mode 100755 main/app/sprinkles/account/factories/Permissions.php create mode 100755 main/app/sprinkles/account/factories/Roles.php create mode 100755 main/app/sprinkles/account/factories/Users.php create mode 100755 main/app/sprinkles/account/locale/ar/messages.php create mode 100755 main/app/sprinkles/account/locale/ar/validate.php create mode 100755 main/app/sprinkles/account/locale/de_DE/messages.php create mode 100755 main/app/sprinkles/account/locale/de_DE/validate.php create mode 100755 main/app/sprinkles/account/locale/en_US/messages.php create mode 100755 main/app/sprinkles/account/locale/en_US/validate.php create mode 100755 main/app/sprinkles/account/locale/es_ES/messages.php create mode 100755 main/app/sprinkles/account/locale/es_ES/validate.php create mode 100755 main/app/sprinkles/account/locale/fa/messages.php create mode 100755 main/app/sprinkles/account/locale/fa/validate.php create mode 100755 main/app/sprinkles/account/locale/fr_FR/messages.php create mode 100755 main/app/sprinkles/account/locale/fr_FR/validate.php create mode 100755 main/app/sprinkles/account/locale/it_IT/messages.php create mode 100755 main/app/sprinkles/account/locale/it_IT/validate.php create mode 100755 main/app/sprinkles/account/locale/pt_PT/messages.php create mode 100755 main/app/sprinkles/account/locale/pt_PT/validate.php create mode 100755 main/app/sprinkles/account/locale/ru_RU/messages.php create mode 100755 main/app/sprinkles/account/locale/ru_RU/validate.php create mode 100755 main/app/sprinkles/account/locale/th_TH/messages.php create mode 100755 main/app/sprinkles/account/locale/th_TH/validate.php create mode 100755 main/app/sprinkles/account/locale/tr/messages.php create mode 100755 main/app/sprinkles/account/locale/tr/validate.php create mode 100755 main/app/sprinkles/account/locale/zh_CN/messages.php create mode 100755 main/app/sprinkles/account/locale/zh_CN/validate.php create mode 100755 main/app/sprinkles/account/routes/routes.php create mode 100755 main/app/sprinkles/account/schema/requests/account-settings.yaml create mode 100755 main/app/sprinkles/account/schema/requests/account-verify.yaml create mode 100755 main/app/sprinkles/account/schema/requests/check-username.yaml create mode 100755 main/app/sprinkles/account/schema/requests/deny-password.yaml create mode 100755 main/app/sprinkles/account/schema/requests/forgot-password.yaml create mode 100755 main/app/sprinkles/account/schema/requests/login.yaml create mode 100755 main/app/sprinkles/account/schema/requests/profile-settings.yaml create mode 100755 main/app/sprinkles/account/schema/requests/register.yaml create mode 100755 main/app/sprinkles/account/schema/requests/resend-verification.yaml create mode 100755 main/app/sprinkles/account/schema/requests/set-password.yaml create mode 100755 main/app/sprinkles/account/src/Account.php create mode 100755 main/app/sprinkles/account/src/Authenticate/AuthGuard.php create mode 100755 main/app/sprinkles/account/src/Authenticate/Authenticator.php create mode 100755 main/app/sprinkles/account/src/Authenticate/Exception/AccountDisabledException.php create mode 100755 main/app/sprinkles/account/src/Authenticate/Exception/AccountInvalidException.php create mode 100755 main/app/sprinkles/account/src/Authenticate/Exception/AccountNotVerifiedException.php create mode 100755 main/app/sprinkles/account/src/Authenticate/Exception/AuthCompromisedException.php create mode 100755 main/app/sprinkles/account/src/Authenticate/Exception/AuthExpiredException.php create mode 100755 main/app/sprinkles/account/src/Authenticate/Exception/InvalidCredentialsException.php create mode 100755 main/app/sprinkles/account/src/Authenticate/Hasher.php create mode 100755 main/app/sprinkles/account/src/Authorize/AccessConditionExpression.php create mode 100755 main/app/sprinkles/account/src/Authorize/AuthorizationException.php create mode 100755 main/app/sprinkles/account/src/Authorize/AuthorizationManager.php create mode 100755 main/app/sprinkles/account/src/Authorize/ParserNodeFunctionEvaluator.php create mode 100755 main/app/sprinkles/account/src/Bakery/CreateAdminUser.php create mode 100755 main/app/sprinkles/account/src/Controller/AccountController.php create mode 100755 main/app/sprinkles/account/src/Controller/Exception/SpammyRequestException.php create mode 100755 main/app/sprinkles/account/src/Database/Migrations/v400/ActivitiesTable.php create mode 100755 main/app/sprinkles/account/src/Database/Migrations/v400/GroupsTable.php create mode 100755 main/app/sprinkles/account/src/Database/Migrations/v400/PasswordResetsTable.php create mode 100755 main/app/sprinkles/account/src/Database/Migrations/v400/PermissionRolesTable.php create mode 100755 main/app/sprinkles/account/src/Database/Migrations/v400/PermissionsTable.php create mode 100755 main/app/sprinkles/account/src/Database/Migrations/v400/PersistencesTable.php create mode 100755 main/app/sprinkles/account/src/Database/Migrations/v400/RoleUsersTable.php create mode 100755 main/app/sprinkles/account/src/Database/Migrations/v400/RolesTable.php create mode 100755 main/app/sprinkles/account/src/Database/Migrations/v400/UsersTable.php create mode 100755 main/app/sprinkles/account/src/Database/Migrations/v400/VerificationsTable.php create mode 100755 main/app/sprinkles/account/src/Database/Models/Activity.php create mode 100755 main/app/sprinkles/account/src/Database/Models/Group.php create mode 100755 main/app/sprinkles/account/src/Database/Models/PasswordReset.php create mode 100755 main/app/sprinkles/account/src/Database/Models/Permission.php create mode 100755 main/app/sprinkles/account/src/Database/Models/Role.php create mode 100755 main/app/sprinkles/account/src/Database/Models/User.php create mode 100755 main/app/sprinkles/account/src/Database/Models/Verification.php create mode 100755 main/app/sprinkles/account/src/Error/Handler/AuthCompromisedExceptionHandler.php create mode 100755 main/app/sprinkles/account/src/Error/Handler/AuthExpiredExceptionHandler.php create mode 100755 main/app/sprinkles/account/src/Error/Handler/ForbiddenExceptionHandler.php create mode 100755 main/app/sprinkles/account/src/Facades/Password.php create mode 100755 main/app/sprinkles/account/src/Log/UserActivityDatabaseHandler.php create mode 100755 main/app/sprinkles/account/src/Log/UserActivityProcessor.php create mode 100755 main/app/sprinkles/account/src/Repository/PasswordResetRepository.php create mode 100755 main/app/sprinkles/account/src/Repository/TokenRepository.php create mode 100755 main/app/sprinkles/account/src/Repository/VerificationRepository.php create mode 100755 main/app/sprinkles/account/src/ServicesProvider/ServicesProvider.php create mode 100755 main/app/sprinkles/account/src/Twig/AccountExtension.php create mode 100755 main/app/sprinkles/account/src/Util/HashFailedException.php create mode 100755 main/app/sprinkles/account/src/Util/Util.php create mode 100755 main/app/sprinkles/account/templates/forms/settings-account.html.twig create mode 100755 main/app/sprinkles/account/templates/forms/settings-profile.html.twig create mode 100755 main/app/sprinkles/account/templates/mail/password-reset.html.twig create mode 100755 main/app/sprinkles/account/templates/mail/resend-verification.html.twig create mode 100755 main/app/sprinkles/account/templates/mail/verify-account.html.twig create mode 100755 main/app/sprinkles/account/templates/modals/tos.html.twig create mode 100755 main/app/sprinkles/account/templates/navigation/main-nav.html.twig create mode 100755 main/app/sprinkles/account/templates/navigation/user-card.html.twig create mode 100755 main/app/sprinkles/account/templates/pages/account-settings.html.twig create mode 100755 main/app/sprinkles/account/templates/pages/error/compromised.html.twig create mode 100755 main/app/sprinkles/account/templates/pages/forgot-password.html.twig create mode 100755 main/app/sprinkles/account/templates/pages/register.html.twig create mode 100755 main/app/sprinkles/account/templates/pages/resend-verification.html.twig create mode 100755 main/app/sprinkles/account/templates/pages/reset-password.html.twig create mode 100755 main/app/sprinkles/account/templates/pages/set-password.html.twig create mode 100755 main/app/sprinkles/account/templates/pages/sign-in.html.twig create mode 100755 main/app/sprinkles/account/tests/Unit/FactoriesTest.php create mode 100755 main/app/sprinkles/account/tests/Unit/HasherTest.php create mode 100755 main/app/sprinkles/admin/asset-bundles.json create mode 100755 main/app/sprinkles/admin/assets/userfrosting/css/tablesorter-custom.css create mode 100755 main/app/sprinkles/admin/assets/userfrosting/js/pages/activities.js create mode 100755 main/app/sprinkles/admin/assets/userfrosting/js/pages/dashboard.js create mode 100755 main/app/sprinkles/admin/assets/userfrosting/js/pages/group.js create mode 100755 main/app/sprinkles/admin/assets/userfrosting/js/pages/groups.js create mode 100755 main/app/sprinkles/admin/assets/userfrosting/js/pages/permission.js create mode 100755 main/app/sprinkles/admin/assets/userfrosting/js/pages/permissions.js create mode 100755 main/app/sprinkles/admin/assets/userfrosting/js/pages/role.js create mode 100755 main/app/sprinkles/admin/assets/userfrosting/js/pages/roles.js create mode 100755 main/app/sprinkles/admin/assets/userfrosting/js/pages/user.js create mode 100755 main/app/sprinkles/admin/assets/userfrosting/js/pages/users.js create mode 100755 main/app/sprinkles/admin/assets/userfrosting/js/widgets/groups.js create mode 100755 main/app/sprinkles/admin/assets/userfrosting/js/widgets/roles.js create mode 100755 main/app/sprinkles/admin/assets/userfrosting/js/widgets/users.js create mode 100755 main/app/sprinkles/admin/composer.json create mode 100755 main/app/sprinkles/admin/locale/ar/messages.php create mode 100755 main/app/sprinkles/admin/locale/de_DE/messages.php create mode 100755 main/app/sprinkles/admin/locale/en_US/messages.php create mode 100755 main/app/sprinkles/admin/locale/es_ES/messages.php create mode 100755 main/app/sprinkles/admin/locale/fa/messages.php create mode 100755 main/app/sprinkles/admin/locale/fr_FR/messages.php create mode 100755 main/app/sprinkles/admin/locale/it_IT/messages.php create mode 100755 main/app/sprinkles/admin/locale/pt_PT/messages.php create mode 100755 main/app/sprinkles/admin/locale/ru_RU/messages.php create mode 100755 main/app/sprinkles/admin/locale/th_TH/messages.php create mode 100755 main/app/sprinkles/admin/locale/tr/messages.php create mode 100755 main/app/sprinkles/admin/locale/zh_CN/messages.php create mode 100755 main/app/sprinkles/admin/routes/activities.php create mode 100755 main/app/sprinkles/admin/routes/admin.php create mode 100755 main/app/sprinkles/admin/routes/groups.php create mode 100755 main/app/sprinkles/admin/routes/permissions.php create mode 100755 main/app/sprinkles/admin/routes/roles.php create mode 100755 main/app/sprinkles/admin/routes/users.php create mode 100755 main/app/sprinkles/admin/schema/requests/group/create.yaml create mode 100755 main/app/sprinkles/admin/schema/requests/group/edit-info.yaml create mode 100755 main/app/sprinkles/admin/schema/requests/group/get-by-slug.yaml create mode 100755 main/app/sprinkles/admin/schema/requests/role/create.yaml create mode 100755 main/app/sprinkles/admin/schema/requests/role/edit-field.yaml create mode 100755 main/app/sprinkles/admin/schema/requests/role/edit-info.yaml create mode 100755 main/app/sprinkles/admin/schema/requests/role/get-by-slug.yaml create mode 100755 main/app/sprinkles/admin/schema/requests/user/create.yaml create mode 100755 main/app/sprinkles/admin/schema/requests/user/edit-field.yaml create mode 100755 main/app/sprinkles/admin/schema/requests/user/edit-info.yaml create mode 100755 main/app/sprinkles/admin/schema/requests/user/edit-password.yaml create mode 100755 main/app/sprinkles/admin/schema/requests/user/get-by-username.yaml create mode 100755 main/app/sprinkles/admin/src/Admin.php create mode 100755 main/app/sprinkles/admin/src/Controller/ActivityController.php create mode 100755 main/app/sprinkles/admin/src/Controller/AdminController.php create mode 100755 main/app/sprinkles/admin/src/Controller/GroupController.php create mode 100755 main/app/sprinkles/admin/src/Controller/PermissionController.php create mode 100755 main/app/sprinkles/admin/src/Controller/RoleController.php create mode 100755 main/app/sprinkles/admin/src/Controller/UserController.php create mode 100755 main/app/sprinkles/admin/src/ServicesProvider/ServicesProvider.php create mode 100755 main/app/sprinkles/admin/src/Sprunje/ActivitySprunje.php create mode 100755 main/app/sprinkles/admin/src/Sprunje/GroupSprunje.php create mode 100755 main/app/sprinkles/admin/src/Sprunje/PermissionSprunje.php create mode 100755 main/app/sprinkles/admin/src/Sprunje/PermissionUserSprunje.php create mode 100755 main/app/sprinkles/admin/src/Sprunje/RoleSprunje.php create mode 100755 main/app/sprinkles/admin/src/Sprunje/UserPermissionSprunje.php create mode 100755 main/app/sprinkles/admin/src/Sprunje/UserSprunje.php create mode 100755 main/app/sprinkles/admin/templates/forms/group.html.twig create mode 100755 main/app/sprinkles/admin/templates/forms/role.html.twig create mode 100755 main/app/sprinkles/admin/templates/forms/user.html.twig create mode 100755 main/app/sprinkles/admin/templates/mail/password-create.html.twig create mode 100755 main/app/sprinkles/admin/templates/modals/confirm-clear-cache.html.twig create mode 100755 main/app/sprinkles/admin/templates/modals/confirm-delete-group.html.twig create mode 100755 main/app/sprinkles/admin/templates/modals/confirm-delete-role.html.twig create mode 100755 main/app/sprinkles/admin/templates/modals/confirm-delete-user.html.twig create mode 100755 main/app/sprinkles/admin/templates/modals/group.html.twig create mode 100755 main/app/sprinkles/admin/templates/modals/role-manage-permissions.html.twig create mode 100755 main/app/sprinkles/admin/templates/modals/role.html.twig create mode 100755 main/app/sprinkles/admin/templates/modals/user-manage-roles.html.twig create mode 100755 main/app/sprinkles/admin/templates/modals/user-set-password.html.twig create mode 100755 main/app/sprinkles/admin/templates/modals/user.html.twig create mode 100755 main/app/sprinkles/admin/templates/navigation/navbar.html.twig create mode 100755 main/app/sprinkles/admin/templates/navigation/sidebar-menu.html.twig create mode 100755 main/app/sprinkles/admin/templates/navigation/sidebar-user.html.twig create mode 100755 main/app/sprinkles/admin/templates/navigation/sidebar.html.twig create mode 100755 main/app/sprinkles/admin/templates/navigation/user-card.html.twig create mode 100755 main/app/sprinkles/admin/templates/pages/abstract/dashboard.html.twig create mode 100755 main/app/sprinkles/admin/templates/pages/activities.html.twig create mode 100755 main/app/sprinkles/admin/templates/pages/dashboard.html.twig create mode 100755 main/app/sprinkles/admin/templates/pages/group.html.twig create mode 100755 main/app/sprinkles/admin/templates/pages/groups.html.twig create mode 100755 main/app/sprinkles/admin/templates/pages/permission.html.twig create mode 100755 main/app/sprinkles/admin/templates/pages/permissions.html.twig create mode 100755 main/app/sprinkles/admin/templates/pages/role.html.twig create mode 100755 main/app/sprinkles/admin/templates/pages/roles.html.twig create mode 100755 main/app/sprinkles/admin/templates/pages/user.html.twig create mode 100755 main/app/sprinkles/admin/templates/pages/users.html.twig create mode 100755 main/app/sprinkles/admin/templates/tables/activities.html.twig create mode 100755 main/app/sprinkles/admin/templates/tables/groups.html.twig create mode 100755 main/app/sprinkles/admin/templates/tables/permissions.html.twig create mode 100755 main/app/sprinkles/admin/templates/tables/roles.html.twig create mode 100755 main/app/sprinkles/admin/templates/tables/users.html.twig create mode 100755 main/app/sprinkles/admin/tests/Integration/SprunjeTests.php create mode 100755 main/app/sprinkles/core/asset-bundles.json create mode 100644 main/app/sprinkles/core/assets/SiteAssets/css/main.css create mode 100644 main/app/sprinkles/core/assets/SiteAssets/css/slick.css create mode 100644 main/app/sprinkles/core/assets/SiteAssets/icons/BurgerMenuShort.svg create mode 100644 main/app/sprinkles/core/assets/SiteAssets/icons/ExploreGlobeOutline.svg create mode 100644 main/app/sprinkles/core/assets/SiteAssets/icons/FriendFeedOutline.svg create mode 100644 main/app/sprinkles/core/assets/SiteAssets/icons/MessageBubbleOutline.svg create mode 100644 main/app/sprinkles/core/assets/SiteAssets/icons/UserGroupOutline.svg create mode 100644 main/app/sprinkles/core/assets/SiteAssets/icons/UserOutline.svg create mode 100644 main/app/sprinkles/core/assets/SiteAssets/js/chat.js create mode 100644 main/app/sprinkles/core/assets/SiteAssets/js/encryption.js create mode 100644 main/app/sprinkles/core/assets/SiteAssets/js/fontawesome.js create mode 100644 main/app/sprinkles/core/assets/SiteAssets/js/jquery.js create mode 100644 main/app/sprinkles/core/assets/SiteAssets/js/language.js create mode 100644 main/app/sprinkles/core/assets/SiteAssets/js/linkify.js create mode 100644 main/app/sprinkles/core/assets/SiteAssets/js/main.js create mode 100644 main/app/sprinkles/core/assets/SiteAssets/js/modernizr.js create mode 100644 main/app/sprinkles/core/assets/SiteAssets/js/slick.js create mode 100644 main/app/sprinkles/core/assets/SiteAssets/languages/ExcelFile.xls create mode 100644 main/app/sprinkles/core/assets/SiteAssets/languages/json/Translations.json create mode 100644 main/app/sprinkles/core/assets/SiteAssets/languages/json/de.json create mode 100644 main/app/sprinkles/core/assets/SiteAssets/languages/json/en.json create mode 100644 main/app/sprinkles/core/assets/SiteAssets/languages/json/fr.json create mode 100644 main/app/sprinkles/core/assets/SiteAssets/languages/json/kl.json create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/Chatserver/bin/WebChatServer.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/Chatserver/src/ChatProcessor.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/SavePublicKey.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/composer.json create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/composer.lock create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/scripts.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/stylesheet.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/autoload.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/cboden/ratchet/.gitignore create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/cboden/ratchet/.travis.yml create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/cboden/ratchet/CHANGELOG.md create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/cboden/ratchet/LICENSE create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/cboden/ratchet/Makefile create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/cboden/ratchet/README.md create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/cboden/ratchet/composer.json create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/cboden/ratchet/phpunit.xml.dist create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/cboden/ratchet/src/Ratchet/AbstractConnectionDecorator.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/cboden/ratchet/src/Ratchet/App.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/cboden/ratchet/src/Ratchet/ComponentInterface.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/cboden/ratchet/src/Ratchet/ConnectionInterface.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/cboden/ratchet/src/Ratchet/Http/CloseResponseTrait.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/cboden/ratchet/src/Ratchet/Http/HttpRequestParser.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/cboden/ratchet/src/Ratchet/Http/HttpServer.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/cboden/ratchet/src/Ratchet/Http/HttpServerInterface.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/cboden/ratchet/src/Ratchet/Http/NoOpHttpServerController.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/cboden/ratchet/src/Ratchet/Http/OriginCheck.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/cboden/ratchet/src/Ratchet/Http/Router.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/cboden/ratchet/src/Ratchet/MessageComponentInterface.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/cboden/ratchet/src/Ratchet/MessageInterface.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/cboden/ratchet/src/Ratchet/Server/EchoServer.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/cboden/ratchet/src/Ratchet/Server/FlashPolicy.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/cboden/ratchet/src/Ratchet/Server/IoConnection.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/cboden/ratchet/src/Ratchet/Server/IoServer.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/cboden/ratchet/src/Ratchet/Server/IpBlackList.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/cboden/ratchet/src/Ratchet/Session/Serialize/HandlerInterface.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/cboden/ratchet/src/Ratchet/Session/Serialize/PhpBinaryHandler.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/cboden/ratchet/src/Ratchet/Session/Serialize/PhpHandler.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/cboden/ratchet/src/Ratchet/Session/SessionProvider.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/cboden/ratchet/src/Ratchet/Session/Storage/Proxy/VirtualProxy.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/cboden/ratchet/src/Ratchet/Session/Storage/VirtualSessionStorage.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/cboden/ratchet/src/Ratchet/Wamp/Exception.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/cboden/ratchet/src/Ratchet/Wamp/JsonException.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/cboden/ratchet/src/Ratchet/Wamp/ServerProtocol.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/cboden/ratchet/src/Ratchet/Wamp/Topic.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/cboden/ratchet/src/Ratchet/Wamp/TopicManager.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/cboden/ratchet/src/Ratchet/Wamp/WampConnection.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/cboden/ratchet/src/Ratchet/Wamp/WampServer.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/cboden/ratchet/src/Ratchet/Wamp/WampServerInterface.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/cboden/ratchet/src/Ratchet/WebSocket/ConnContext.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/cboden/ratchet/src/Ratchet/WebSocket/MessageCallableInterface.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/cboden/ratchet/src/Ratchet/WebSocket/MessageComponentInterface.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/cboden/ratchet/src/Ratchet/WebSocket/WsConnection.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/cboden/ratchet/src/Ratchet/WebSocket/WsServer.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/cboden/ratchet/src/Ratchet/WebSocket/WsServerInterface.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/cboden/ratchet/tests/autobahn/bin/fuzzingserver.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/cboden/ratchet/tests/autobahn/fuzzingclient-all.json create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/cboden/ratchet/tests/autobahn/fuzzingclient-profile.json create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/cboden/ratchet/tests/autobahn/fuzzingclient-quick.json create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/cboden/ratchet/tests/bootstrap.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/cboden/ratchet/tests/helpers/Ratchet/AbstractMessageComponentTestCase.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/cboden/ratchet/tests/helpers/Ratchet/Mock/Component.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/cboden/ratchet/tests/helpers/Ratchet/Mock/Connection.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/cboden/ratchet/tests/helpers/Ratchet/Mock/ConnectionDecorator.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/cboden/ratchet/tests/helpers/Ratchet/Mock/WampComponent.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/cboden/ratchet/tests/helpers/Ratchet/NullComponent.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/cboden/ratchet/tests/helpers/Ratchet/Wamp/Stub/WsWampServerInterface.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/cboden/ratchet/tests/helpers/Ratchet/WebSocket/Stub/WsMessageComponentInterface.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/cboden/ratchet/tests/unit/AbstractConnectionDecoratorTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/cboden/ratchet/tests/unit/Http/HttpRequestParserTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/cboden/ratchet/tests/unit/Http/HttpServerTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/cboden/ratchet/tests/unit/Http/OriginCheckTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/cboden/ratchet/tests/unit/Http/RouterTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/cboden/ratchet/tests/unit/Server/EchoServerTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/cboden/ratchet/tests/unit/Server/FlashPolicyComponentTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/cboden/ratchet/tests/unit/Server/IoConnectionTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/cboden/ratchet/tests/unit/Server/IoServerTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/cboden/ratchet/tests/unit/Server/IpBlackListComponentTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/cboden/ratchet/tests/unit/Session/Serialize/PhpHandlerTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/cboden/ratchet/tests/unit/Session/SessionComponentTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/cboden/ratchet/tests/unit/Session/Storage/VirtualSessionStoragePDOTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/cboden/ratchet/tests/unit/Wamp/ServerProtocolTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/cboden/ratchet/tests/unit/Wamp/TopicManagerTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/cboden/ratchet/tests/unit/Wamp/TopicTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/cboden/ratchet/tests/unit/Wamp/WampConnectionTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/cboden/ratchet/tests/unit/Wamp/WampServerTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/composer/ClassLoader.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/composer/LICENSE create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/composer/autoload_classmap.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/composer/autoload_files.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/composer/autoload_namespaces.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/composer/autoload_psr4.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/composer/autoload_real.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/composer/autoload_static.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/composer/installed.json create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/evenement/evenement/.gitignore create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/evenement/evenement/.travis.yml create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/evenement/evenement/CHANGELOG.md create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/evenement/evenement/LICENSE create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/evenement/evenement/README.md create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/evenement/evenement/composer.json create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/evenement/evenement/doc/00-intro.md create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/evenement/evenement/doc/01-api.md create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/evenement/evenement/doc/02-plugin-system.md create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/evenement/evenement/examples/benchmark-emit-no-arguments.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/evenement/evenement/examples/benchmark-emit-once.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/evenement/evenement/examples/benchmark-emit-one-argument.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/evenement/evenement/examples/benchmark-emit.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/evenement/evenement/examples/benchmark-remove-listener-once.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/evenement/evenement/phpunit.xml.dist create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/evenement/evenement/src/Evenement/EventEmitter.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/evenement/evenement/src/Evenement/EventEmitterInterface.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/evenement/evenement/src/Evenement/EventEmitterTrait.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/evenement/evenement/tests/Evenement/Tests/EventEmitterTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/evenement/evenement/tests/Evenement/Tests/Listener.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/evenement/evenement/tests/Evenement/Tests/functions.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/guzzlehttp/psr7/CHANGELOG.md create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/guzzlehttp/psr7/LICENSE create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/guzzlehttp/psr7/README.md create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/guzzlehttp/psr7/composer.json create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/guzzlehttp/psr7/src/AppendStream.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/guzzlehttp/psr7/src/BufferStream.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/guzzlehttp/psr7/src/CachingStream.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/guzzlehttp/psr7/src/DroppingStream.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/guzzlehttp/psr7/src/FnStream.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/guzzlehttp/psr7/src/InflateStream.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/guzzlehttp/psr7/src/LazyOpenStream.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/guzzlehttp/psr7/src/LimitStream.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/guzzlehttp/psr7/src/MessageTrait.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/guzzlehttp/psr7/src/MultipartStream.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/guzzlehttp/psr7/src/NoSeekStream.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/guzzlehttp/psr7/src/PumpStream.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/guzzlehttp/psr7/src/Request.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/guzzlehttp/psr7/src/Response.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/guzzlehttp/psr7/src/ServerRequest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/guzzlehttp/psr7/src/Stream.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/guzzlehttp/psr7/src/StreamDecoratorTrait.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/guzzlehttp/psr7/src/StreamWrapper.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/guzzlehttp/psr7/src/UploadedFile.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/guzzlehttp/psr7/src/Uri.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/guzzlehttp/psr7/src/UriNormalizer.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/guzzlehttp/psr7/src/UriResolver.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/guzzlehttp/psr7/src/functions.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/guzzlehttp/psr7/src/functions_include.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/kriswallsmith/assetic/CHANGELOG-1.0.md create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/kriswallsmith/assetic/CHANGELOG-1.1.md create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/kriswallsmith/assetic/CHANGELOG-1.2.md create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/kriswallsmith/assetic/Gemfile create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/kriswallsmith/assetic/LICENSE create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/kriswallsmith/assetic/README.md create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/kriswallsmith/assetic/composer.json create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/kriswallsmith/assetic/package.json create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/kriswallsmith/assetic/src/Assetic/Asset/AssetCache.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/kriswallsmith/assetic/src/Assetic/Asset/AssetCollection.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/kriswallsmith/assetic/src/Assetic/Asset/AssetCollectionInterface.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/kriswallsmith/assetic/src/Assetic/Asset/AssetInterface.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/kriswallsmith/assetic/src/Assetic/Asset/AssetReference.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/kriswallsmith/assetic/src/Assetic/Asset/BaseAsset.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/kriswallsmith/assetic/src/Assetic/Asset/FileAsset.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/kriswallsmith/assetic/src/Assetic/Asset/GlobAsset.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/kriswallsmith/assetic/src/Assetic/Asset/HttpAsset.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/kriswallsmith/assetic/src/Assetic/Asset/Iterator/AssetCollectionFilterIterator.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/kriswallsmith/assetic/src/Assetic/Asset/Iterator/AssetCollectionIterator.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/kriswallsmith/assetic/src/Assetic/Asset/StringAsset.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/kriswallsmith/assetic/src/Assetic/AssetManager.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/kriswallsmith/assetic/src/Assetic/AssetWriter.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/kriswallsmith/assetic/src/Assetic/Cache/ApcCache.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/kriswallsmith/assetic/src/Assetic/Cache/ArrayCache.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/kriswallsmith/assetic/src/Assetic/Cache/CacheInterface.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/kriswallsmith/assetic/src/Assetic/Cache/ConfigCache.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/kriswallsmith/assetic/src/Assetic/Cache/ExpiringCache.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/kriswallsmith/assetic/src/Assetic/Cache/FilesystemCache.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/kriswallsmith/assetic/src/Assetic/Exception/Exception.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/kriswallsmith/assetic/src/Assetic/Exception/FilterException.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/kriswallsmith/assetic/src/Assetic/Extension/Twig/AsseticExtension.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/kriswallsmith/assetic/src/Assetic/Extension/Twig/AsseticFilterFunction.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/kriswallsmith/assetic/src/Assetic/Extension/Twig/AsseticFilterInvoker.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/kriswallsmith/assetic/src/Assetic/Extension/Twig/AsseticFilterNode.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/kriswallsmith/assetic/src/Assetic/Extension/Twig/AsseticNode.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/kriswallsmith/assetic/src/Assetic/Extension/Twig/AsseticTokenParser.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/kriswallsmith/assetic/src/Assetic/Extension/Twig/TwigFormulaLoader.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/kriswallsmith/assetic/src/Assetic/Extension/Twig/TwigResource.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/kriswallsmith/assetic/src/Assetic/Extension/Twig/ValueContainer.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/kriswallsmith/assetic/src/Assetic/Factory/AssetFactory.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/kriswallsmith/assetic/src/Assetic/Factory/LazyAssetManager.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/kriswallsmith/assetic/src/Assetic/Factory/Loader/BasePhpFormulaLoader.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/kriswallsmith/assetic/src/Assetic/Factory/Loader/CachedFormulaLoader.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/kriswallsmith/assetic/src/Assetic/Factory/Loader/FormulaLoaderInterface.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/kriswallsmith/assetic/src/Assetic/Factory/Loader/FunctionCallsFormulaLoader.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/kriswallsmith/assetic/src/Assetic/Factory/Resource/CoalescingDirectoryResource.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/kriswallsmith/assetic/src/Assetic/Factory/Resource/DirectoryResource.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/kriswallsmith/assetic/src/Assetic/Factory/Resource/FileResource.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/kriswallsmith/assetic/src/Assetic/Factory/Resource/IteratorResourceInterface.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/kriswallsmith/assetic/src/Assetic/Factory/Resource/ResourceInterface.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/kriswallsmith/assetic/src/Assetic/Factory/Worker/CacheBustingWorker.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/kriswallsmith/assetic/src/Assetic/Factory/Worker/EnsureFilterWorker.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/kriswallsmith/assetic/src/Assetic/Factory/Worker/WorkerInterface.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/kriswallsmith/assetic/src/Assetic/Filter/AutoprefixerFilter.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/kriswallsmith/assetic/src/Assetic/Filter/BaseCssFilter.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/kriswallsmith/assetic/src/Assetic/Filter/BaseNodeFilter.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/kriswallsmith/assetic/src/Assetic/Filter/BaseProcessFilter.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/kriswallsmith/assetic/src/Assetic/Filter/CallablesFilter.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/kriswallsmith/assetic/src/Assetic/Filter/CleanCssFilter.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/kriswallsmith/assetic/src/Assetic/Filter/CoffeeScriptFilter.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/kriswallsmith/assetic/src/Assetic/Filter/CompassFilter.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/kriswallsmith/assetic/src/Assetic/Filter/CssCacheBustingFilter.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/kriswallsmith/assetic/src/Assetic/Filter/CssEmbedFilter.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/kriswallsmith/assetic/src/Assetic/Filter/CssImportFilter.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/kriswallsmith/assetic/src/Assetic/Filter/CssMinFilter.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/kriswallsmith/assetic/src/Assetic/Filter/CssRewriteFilter.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/kriswallsmith/assetic/src/Assetic/Filter/DartFilter.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/kriswallsmith/assetic/src/Assetic/Filter/DependencyExtractorInterface.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/kriswallsmith/assetic/src/Assetic/Filter/EmberPrecompileFilter.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/kriswallsmith/assetic/src/Assetic/Filter/FilterCollection.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/kriswallsmith/assetic/src/Assetic/Filter/FilterInterface.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/kriswallsmith/assetic/src/Assetic/Filter/GoogleClosure/BaseCompilerFilter.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/kriswallsmith/assetic/src/Assetic/Filter/GoogleClosure/CompilerApiFilter.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/kriswallsmith/assetic/src/Assetic/Filter/GoogleClosure/CompilerJarFilter.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/kriswallsmith/assetic/src/Assetic/Filter/GssFilter.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/kriswallsmith/assetic/src/Assetic/Filter/HandlebarsFilter.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/kriswallsmith/assetic/src/Assetic/Filter/HashableInterface.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/kriswallsmith/assetic/src/Assetic/Filter/JSMinFilter.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/kriswallsmith/assetic/src/Assetic/Filter/JSMinPlusFilter.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/kriswallsmith/assetic/src/Assetic/Filter/JSqueezeFilter.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/kriswallsmith/assetic/src/Assetic/Filter/JpegoptimFilter.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/kriswallsmith/assetic/src/Assetic/Filter/JpegtranFilter.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/kriswallsmith/assetic/src/Assetic/Filter/LessFilter.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/kriswallsmith/assetic/src/Assetic/Filter/LessphpFilter.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/kriswallsmith/assetic/src/Assetic/Filter/MinifyCssCompressorFilter.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/kriswallsmith/assetic/src/Assetic/Filter/OptiPngFilter.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/kriswallsmith/assetic/src/Assetic/Filter/PackagerFilter.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/kriswallsmith/assetic/src/Assetic/Filter/PackerFilter.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/kriswallsmith/assetic/src/Assetic/Filter/PhpCssEmbedFilter.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/kriswallsmith/assetic/src/Assetic/Filter/PngoutFilter.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/kriswallsmith/assetic/src/Assetic/Filter/ReactJsxFilter.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/kriswallsmith/assetic/src/Assetic/Filter/RooleFilter.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/kriswallsmith/assetic/src/Assetic/Filter/Sass/BaseSassFilter.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/kriswallsmith/assetic/src/Assetic/Filter/Sass/SassFilter.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/kriswallsmith/assetic/src/Assetic/Filter/Sass/ScssFilter.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/kriswallsmith/assetic/src/Assetic/Filter/SassphpFilter.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/kriswallsmith/assetic/src/Assetic/Filter/ScssphpFilter.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/kriswallsmith/assetic/src/Assetic/Filter/SeparatorFilter.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/kriswallsmith/assetic/src/Assetic/Filter/SprocketsFilter.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/kriswallsmith/assetic/src/Assetic/Filter/StylusFilter.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/kriswallsmith/assetic/src/Assetic/Filter/TypeScriptFilter.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/kriswallsmith/assetic/src/Assetic/Filter/UglifyCssFilter.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/kriswallsmith/assetic/src/Assetic/Filter/UglifyJs2Filter.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/kriswallsmith/assetic/src/Assetic/Filter/UglifyJsFilter.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/kriswallsmith/assetic/src/Assetic/Filter/Yui/BaseCompressorFilter.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/kriswallsmith/assetic/src/Assetic/Filter/Yui/CssCompressorFilter.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/kriswallsmith/assetic/src/Assetic/Filter/Yui/JsCompressorFilter.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/kriswallsmith/assetic/src/Assetic/FilterManager.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/kriswallsmith/assetic/src/Assetic/Util/CssUtils.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/kriswallsmith/assetic/src/Assetic/Util/FilesystemUtils.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/kriswallsmith/assetic/src/Assetic/Util/LessUtils.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/kriswallsmith/assetic/src/Assetic/Util/SassUtils.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/kriswallsmith/assetic/src/Assetic/Util/TraversableString.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/kriswallsmith/assetic/src/Assetic/Util/VarUtils.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/kriswallsmith/assetic/src/Assetic/ValueSupplierInterface.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/kriswallsmith/assetic/src/functions.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/lmammino/jsmin4assetic/LICENSE create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/lmammino/jsmin4assetic/README.markdown create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/lmammino/jsmin4assetic/composer.json create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/lmammino/jsmin4assetic/src/JSMin.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/lmammino/jsmin4assetic/src/JSMinUnterminatedCommentException.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/lmammino/jsmin4assetic/src/JSMinUnterminatedRegExpException.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/lmammino/jsmin4assetic/src/JSMinUnterminatedStringException.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/matthiasmullie/minify/CONTRIBUTING.md create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/matthiasmullie/minify/Dockerfile create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/matthiasmullie/minify/LICENSE create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/matthiasmullie/minify/bin/minifycss create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/matthiasmullie/minify/bin/minifyjs create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/matthiasmullie/minify/composer.json create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/matthiasmullie/minify/data/js/keywords_after.txt create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/matthiasmullie/minify/data/js/keywords_before.txt create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/matthiasmullie/minify/data/js/keywords_reserved.txt create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/matthiasmullie/minify/data/js/operators.txt create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/matthiasmullie/minify/data/js/operators_after.txt create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/matthiasmullie/minify/data/js/operators_before.txt create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/matthiasmullie/minify/docker-compose.yml create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/matthiasmullie/minify/src/CSS.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/matthiasmullie/minify/src/Exception.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/matthiasmullie/minify/src/Exceptions/BasicException.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/matthiasmullie/minify/src/Exceptions/FileImportException.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/matthiasmullie/minify/src/Exceptions/IOException.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/matthiasmullie/minify/src/JS.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/matthiasmullie/minify/src/Minify.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/matthiasmullie/path-converter/LICENSE create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/matthiasmullie/path-converter/composer.json create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/matthiasmullie/path-converter/src/Converter.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/matthiasmullie/path-converter/src/ConverterInterface.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/matthiasmullie/path-converter/src/NoConverter.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/natxet/CssMin/README create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/natxet/CssMin/composer.json create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/natxet/CssMin/src/CssMin.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/nubs/random-name-generator/.coveralls.yml create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/nubs/random-name-generator/.gitignore create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/nubs/random-name-generator/.scrutinizer.yml create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/nubs/random-name-generator/.travis.yml create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/nubs/random-name-generator/CONTRIBUTING.md create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/nubs/random-name-generator/Dockerfile.tests create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/nubs/random-name-generator/LICENSE create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/nubs/random-name-generator/README.md create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/nubs/random-name-generator/build.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/nubs/random-name-generator/composer.json create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/nubs/random-name-generator/composer.lock create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/nubs/random-name-generator/docker-compose.yml create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/nubs/random-name-generator/phpunit.xml create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/nubs/random-name-generator/src/AbstractGenerator.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/nubs/random-name-generator/src/All.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/nubs/random-name-generator/src/Alliteration.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/nubs/random-name-generator/src/Generator.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/nubs/random-name-generator/src/Vgng.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/nubs/random-name-generator/src/adjectives.txt create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/nubs/random-name-generator/src/nouns.txt create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/nubs/random-name-generator/src/video_game_names.txt create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/nubs/random-name-generator/tests/AllTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/nubs/random-name-generator/tests/AlliterationTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/nubs/random-name-generator/tests/VgngTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/paragonie/random_compat/LICENSE create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/paragonie/random_compat/build-phar.sh create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/paragonie/random_compat/composer.json create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/paragonie/random_compat/dist/random_compat.phar.pubkey create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/paragonie/random_compat/dist/random_compat.phar.pubkey.asc create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/paragonie/random_compat/lib/byte_safe_strings.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/paragonie/random_compat/lib/cast_to_int.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/paragonie/random_compat/lib/error_polyfill.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/paragonie/random_compat/lib/random.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/paragonie/random_compat/lib/random_bytes_com_dotnet.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/paragonie/random_compat/lib/random_bytes_dev_urandom.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/paragonie/random_compat/lib/random_bytes_libsodium.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/paragonie/random_compat/lib/random_bytes_libsodium_legacy.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/paragonie/random_compat/lib/random_bytes_mcrypt.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/paragonie/random_compat/lib/random_int.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/paragonie/random_compat/other/build_phar.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/paragonie/random_compat/psalm-autoload.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/paragonie/random_compat/psalm.xml create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/psr/http-message/CHANGELOG.md create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/psr/http-message/LICENSE create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/psr/http-message/README.md create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/psr/http-message/composer.json create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/psr/http-message/src/MessageInterface.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/psr/http-message/src/RequestInterface.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/psr/http-message/src/ResponseInterface.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/psr/http-message/src/ServerRequestInterface.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/psr/http-message/src/StreamInterface.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/psr/http-message/src/UploadedFileInterface.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/psr/http-message/src/UriInterface.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/ratchet/rfc6455/.gitignore create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/ratchet/rfc6455/.travis.yml create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/ratchet/rfc6455/LICENSE create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/ratchet/rfc6455/README.md create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/ratchet/rfc6455/composer.json create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/ratchet/rfc6455/phpunit.xml.dist create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/ratchet/rfc6455/src/Handshake/ClientNegotiator.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/ratchet/rfc6455/src/Handshake/NegotiatorInterface.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/ratchet/rfc6455/src/Handshake/RequestVerifier.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/ratchet/rfc6455/src/Handshake/ResponseVerifier.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/ratchet/rfc6455/src/Handshake/ServerNegotiator.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/ratchet/rfc6455/src/Messaging/CloseFrameChecker.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/ratchet/rfc6455/src/Messaging/DataInterface.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/ratchet/rfc6455/src/Messaging/Frame.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/ratchet/rfc6455/src/Messaging/FrameInterface.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/ratchet/rfc6455/src/Messaging/Message.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/ratchet/rfc6455/src/Messaging/MessageBuffer.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/ratchet/rfc6455/src/Messaging/MessageInterface.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/ratchet/rfc6455/tests/AbResultsTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/ratchet/rfc6455/tests/ab/clientRunner.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/ratchet/rfc6455/tests/ab/fuzzingclient.json create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/ratchet/rfc6455/tests/ab/fuzzingserver.json create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/ratchet/rfc6455/tests/ab/run_ab_tests.sh create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/ratchet/rfc6455/tests/ab/startServer.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/ratchet/rfc6455/tests/bootstrap.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/ratchet/rfc6455/tests/unit/Handshake/RequestVerifierTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/ratchet/rfc6455/tests/unit/Handshake/ResponseVerifierTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/ratchet/rfc6455/tests/unit/Handshake/ServerNegotiatorTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/ratchet/rfc6455/tests/unit/Messaging/FrameTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/ratchet/rfc6455/tests/unit/Messaging/MessageBufferTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/ratchet/rfc6455/tests/unit/Messaging/MessageTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/cache/.gitignore create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/cache/.travis.yml create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/cache/CHANGELOG.md create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/cache/LICENSE create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/cache/README.md create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/cache/composer.json create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/cache/phpunit.xml.dist create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/cache/src/ArrayCache.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/cache/src/CacheInterface.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/cache/tests/ArrayCacheTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/cache/tests/CallableStub.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/cache/tests/TestCase.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/dns/.gitignore create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/dns/.travis.yml create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/dns/CHANGELOG.md create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/dns/LICENSE create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/dns/README.md create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/dns/composer.json create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/dns/examples/01-one.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/dns/examples/02-concurrent.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/dns/examples/03-cached.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/dns/examples/04-query-a-and-aaaa.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/dns/phpunit.xml.dist create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/dns/src/BadServerException.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/dns/src/Config/Config.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/dns/src/Config/FilesystemFactory.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/dns/src/Config/HostsFile.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/dns/src/Model/HeaderBag.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/dns/src/Model/Message.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/dns/src/Model/Record.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/dns/src/Protocol/BinaryDumper.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/dns/src/Protocol/Parser.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/dns/src/Query/CachedExecutor.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/dns/src/Query/CancellationException.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/dns/src/Query/Executor.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/dns/src/Query/ExecutorInterface.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/dns/src/Query/HostsFileExecutor.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/dns/src/Query/Query.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/dns/src/Query/RecordBag.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/dns/src/Query/RecordCache.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/dns/src/Query/RetryExecutor.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/dns/src/Query/TimeoutException.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/dns/src/Query/TimeoutExecutor.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/dns/src/RecordNotFoundException.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/dns/src/Resolver/Factory.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/dns/src/Resolver/Resolver.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/dns/tests/CallableStub.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/dns/tests/Config/ConfigTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/dns/tests/Config/FilesystemFactoryTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/dns/tests/Config/HostsFileTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/dns/tests/Fixtures/etc/resolv.conf create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/dns/tests/FunctionalResolverTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/dns/tests/Model/MessageTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/dns/tests/Protocol/BinaryDumperTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/dns/tests/Protocol/ParserTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/dns/tests/Query/CachedExecutorTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/dns/tests/Query/ExecutorTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/dns/tests/Query/HostsFileExecutorTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/dns/tests/Query/RecordBagTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/dns/tests/Query/RecordCacheTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/dns/tests/Query/RetryExecutorTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/dns/tests/Query/TimeoutExecutorTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/dns/tests/Resolver/FactoryTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/dns/tests/Resolver/ResolveAliasesTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/dns/tests/Resolver/ResolverTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/dns/tests/TestCase.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/event-loop/.gitignore create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/event-loop/.travis.yml create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/event-loop/CHANGELOG.md create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/event-loop/LICENSE create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/event-loop/README.md create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/event-loop/composer.json create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/event-loop/examples/01-timers.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/event-loop/examples/02-periodic.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/event-loop/examples/03-ticks.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/event-loop/examples/04-signals.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/event-loop/examples/11-consume-stdin.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/event-loop/examples/12-generate-yes.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/event-loop/examples/13-http-client-blocking.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/event-loop/examples/14-http-client-async.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/event-loop/examples/21-http-server.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/event-loop/examples/91-benchmark-ticks.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/event-loop/examples/92-benchmark-timers.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/event-loop/examples/93-benchmark-ticks-delay.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/event-loop/examples/94-benchmark-timers-delay.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/event-loop/examples/95-benchmark-memory.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/event-loop/phpunit.xml.dist create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/event-loop/src/ExtEvLoop.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/event-loop/src/ExtEventLoop.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/event-loop/src/ExtLibevLoop.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/event-loop/src/ExtLibeventLoop.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/event-loop/src/Factory.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/event-loop/src/LoopInterface.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/event-loop/src/SignalsHandler.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/event-loop/src/StreamSelectLoop.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/event-loop/src/Tick/FutureTickQueue.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/event-loop/src/Timer/Timer.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/event-loop/src/Timer/Timers.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/event-loop/src/TimerInterface.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/event-loop/tests/AbstractLoopTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/event-loop/tests/CallableStub.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/event-loop/tests/ExtEvLoopTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/event-loop/tests/ExtEventLoopTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/event-loop/tests/ExtLibevLoopTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/event-loop/tests/ExtLibeventLoopTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/event-loop/tests/SignalsHandlerTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/event-loop/tests/StreamSelectLoopTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/event-loop/tests/TestCase.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/event-loop/tests/Timer/AbstractTimerTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/event-loop/tests/Timer/ExtEvTimerTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/event-loop/tests/Timer/ExtEventTimerTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/event-loop/tests/Timer/ExtLibevTimerTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/event-loop/tests/Timer/ExtLibeventTimerTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/event-loop/tests/Timer/StreamSelectTimerTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/event-loop/tests/Timer/TimersTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/event-loop/tests/bootstrap.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/event-loop/travis-init.sh create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/promise-timer/.gitignore create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/promise-timer/.travis.yml create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/promise-timer/CHANGELOG.md create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/promise-timer/LICENSE create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/promise-timer/README.md create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/promise-timer/composer.json create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/promise-timer/phpunit.xml.dist create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/promise-timer/src/TimeoutException.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/promise-timer/src/functions.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/promise-timer/tests/CallableStub.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/promise-timer/tests/FunctionRejectTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/promise-timer/tests/FunctionResolveTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/promise-timer/tests/FunctionTimeoutTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/promise-timer/tests/TestCase.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/promise-timer/tests/TimeoutExceptionTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/promise/.gitignore create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/promise/.travis.yml create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/promise/CHANGELOG.md create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/promise/LICENSE create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/promise/README.md create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/promise/composer.json create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/promise/phpunit.xml.dist create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/promise/src/CancellablePromiseInterface.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/promise/src/CancellationQueue.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/promise/src/Deferred.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/promise/src/Exception/LengthException.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/promise/src/ExtendedPromiseInterface.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/promise/src/FulfilledPromise.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/promise/src/LazyPromise.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/promise/src/Promise.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/promise/src/PromiseInterface.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/promise/src/PromisorInterface.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/promise/src/RejectedPromise.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/promise/src/UnhandledRejectionException.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/promise/src/functions.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/promise/src/functions_include.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/promise/tests/CancellationQueueTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/promise/tests/DeferredTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/promise/tests/FulfilledPromiseTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/promise/tests/FunctionAllTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/promise/tests/FunctionAnyTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/promise/tests/FunctionCheckTypehintTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/promise/tests/FunctionMapTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/promise/tests/FunctionRaceTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/promise/tests/FunctionReduceTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/promise/tests/FunctionRejectTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/promise/tests/FunctionResolveTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/promise/tests/FunctionSomeTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/promise/tests/LazyPromiseTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/promise/tests/PromiseAdapter/CallbackPromiseAdapter.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/promise/tests/PromiseAdapter/PromiseAdapterInterface.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/promise/tests/PromiseTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/promise/tests/PromiseTest/CancelTestTrait.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/promise/tests/PromiseTest/FullTestTrait.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/promise/tests/PromiseTest/NotifyTestTrait.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/promise/tests/PromiseTest/PromiseFulfilledTestTrait.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/promise/tests/PromiseTest/PromisePendingTestTrait.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/promise/tests/PromiseTest/PromiseRejectedTestTrait.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/promise/tests/PromiseTest/PromiseSettledTestTrait.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/promise/tests/PromiseTest/RejectTestTrait.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/promise/tests/PromiseTest/ResolveTestTrait.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/promise/tests/RejectedPromiseTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/promise/tests/Stub/CallableStub.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/promise/tests/TestCase.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/promise/tests/bootstrap.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/promise/tests/fixtures/SimpleFulfilledTestPromise.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/promise/tests/fixtures/SimpleFulfilledTestThenable.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/promise/tests/fixtures/SimpleRejectedTestPromise.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/promise/tests/fixtures/SimpleTestCancellable.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/promise/tests/fixtures/SimpleTestCancellableThenable.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/socket/.gitignore create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/socket/.travis.yml create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/socket/CHANGELOG.md create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/socket/LICENSE create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/socket/README.md create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/socket/composer.json create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/socket/examples/01-echo-server.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/socket/examples/02-chat-server.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/socket/examples/03-http-server.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/socket/examples/11-http-client.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/socket/examples/12-https-client.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/socket/examples/21-netcat-client.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/socket/examples/22-http-client.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/socket/examples/91-benchmark-server.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/socket/examples/99-generate-self-signed.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/socket/examples/localhost.pem create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/socket/examples/localhost_swordfish.pem create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/socket/phpunit.xml.dist create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/socket/src/Connection.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/socket/src/ConnectionInterface.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/socket/src/Connector.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/socket/src/ConnectorInterface.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/socket/src/DnsConnector.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/socket/src/FixedUriConnector.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/socket/src/LimitingServer.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/socket/src/SecureConnector.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/socket/src/SecureServer.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/socket/src/Server.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/socket/src/ServerInterface.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/socket/src/StreamEncryption.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/socket/src/TcpConnector.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/socket/src/TcpServer.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/socket/src/TimeoutConnector.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/socket/src/UnixConnector.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/socket/src/UnixServer.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/socket/tests/ConnectionTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/socket/tests/ConnectorTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/socket/tests/DnsConnectorTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/socket/tests/FixedUriConnectorTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/socket/tests/FunctionalConnectorTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/socket/tests/FunctionalSecureServerTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/socket/tests/FunctionalTcpServerTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/socket/tests/IntegrationTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/socket/tests/LimitingServerTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/socket/tests/SecureConnectorTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/socket/tests/SecureIntegrationTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/socket/tests/SecureServerTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/socket/tests/ServerTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/socket/tests/Stub/CallableStub.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/socket/tests/Stub/ConnectionStub.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/socket/tests/Stub/ServerStub.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/socket/tests/TcpConnectorTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/socket/tests/TcpServerTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/socket/tests/TestCase.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/socket/tests/TimeoutConnectorTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/socket/tests/UnixConnectorTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/socket/tests/UnixServerTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/stream/.gitignore create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/stream/.travis.yml create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/stream/CHANGELOG.md create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/stream/LICENSE create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/stream/README.md create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/stream/composer.json create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/stream/examples/01-http.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/stream/examples/02-https.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/stream/examples/11-cat.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/stream/examples/91-benchmark-throughput.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/stream/phpunit.xml.dist create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/stream/src/CompositeStream.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/stream/src/DuplexResourceStream.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/stream/src/DuplexStreamInterface.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/stream/src/ReadableResourceStream.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/stream/src/ReadableStreamInterface.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/stream/src/ThroughStream.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/stream/src/Util.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/stream/src/WritableResourceStream.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/stream/src/WritableStreamInterface.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/stream/tests/CallableStub.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/stream/tests/CompositeStreamTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/stream/tests/DuplexResourceStreamIntegrationTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/stream/tests/DuplexResourceStreamTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/stream/tests/EnforceBlockingWrapper.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/stream/tests/FunctionalInternetTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/stream/tests/ReadableResourceStreamTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/stream/tests/Stub/ReadableStreamStub.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/stream/tests/TestCase.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/stream/tests/ThroughStreamTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/stream/tests/UtilTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/react/stream/tests/WritableStreamResourceTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/.gitignore create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/AcceptHeader.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/AcceptHeaderItem.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/ApacheRequest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/BinaryFileResponse.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/CHANGELOG.md create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/Cookie.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/Exception/ConflictingHeadersException.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/Exception/RequestExceptionInterface.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/Exception/SuspiciousOperationException.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/ExpressionRequestMatcher.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/File/Exception/AccessDeniedException.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/File/Exception/FileException.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/File/Exception/FileNotFoundException.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/File/Exception/UnexpectedTypeException.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/File/Exception/UploadException.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/File/File.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/File/MimeType/ExtensionGuesser.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/File/MimeType/ExtensionGuesserInterface.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/File/MimeType/FileBinaryMimeTypeGuesser.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/File/MimeType/FileinfoMimeTypeGuesser.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/File/MimeType/MimeTypeExtensionGuesser.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/File/MimeType/MimeTypeGuesser.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/File/MimeType/MimeTypeGuesserInterface.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/File/Stream.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/File/UploadedFile.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/FileBag.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/HeaderBag.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/IpUtils.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/JsonResponse.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/LICENSE create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/ParameterBag.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/README.md create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/RedirectResponse.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/Request.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/RequestMatcher.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/RequestMatcherInterface.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/RequestStack.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/Response.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/ResponseHeaderBag.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/ServerBag.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/Session/Attribute/AttributeBag.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/Session/Attribute/AttributeBagInterface.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/Session/Attribute/NamespacedAttributeBag.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/Session/Flash/AutoExpireFlashBag.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/Session/Flash/FlashBag.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/Session/Flash/FlashBagInterface.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/Session/Session.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/Session/SessionBagInterface.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/Session/SessionBagProxy.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/Session/SessionInterface.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/Session/Storage/Handler/AbstractSessionHandler.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/Session/Storage/Handler/MemcacheSessionHandler.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/Session/Storage/Handler/MemcachedSessionHandler.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/Session/Storage/Handler/MongoDbSessionHandler.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/Session/Storage/Handler/NativeFileSessionHandler.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/Session/Storage/Handler/NativeSessionHandler.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/Session/Storage/Handler/NullSessionHandler.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/Session/Storage/Handler/PdoSessionHandler.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/Session/Storage/Handler/StrictSessionHandler.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/Session/Storage/Handler/WriteCheckSessionHandler.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/Session/Storage/MetadataBag.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/Session/Storage/MockArraySessionStorage.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/Session/Storage/MockFileSessionStorage.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/Session/Storage/NativeSessionStorage.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/Session/Storage/PhpBridgeSessionStorage.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/Session/Storage/Proxy/AbstractProxy.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/Session/Storage/Proxy/NativeProxy.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/Session/Storage/Proxy/SessionHandlerProxy.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/Session/Storage/SessionStorageInterface.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/StreamedResponse.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/Tests/AcceptHeaderItemTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/Tests/AcceptHeaderTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/Tests/ApacheRequestTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/Tests/BinaryFileResponseTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/Tests/CookieTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/Tests/ExpressionRequestMatcherTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/Tests/File/FakeFile.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/Tests/File/FileTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/Tests/File/Fixtures/.unknownextension create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/Tests/File/Fixtures/directory/.empty create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/Tests/File/Fixtures/other-file.example create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/Tests/File/Fixtures/test create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/Tests/File/Fixtures/test.gif create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/Tests/File/MimeType/MimeTypeTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/Tests/File/UploadedFileTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/Tests/FileBagTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/Tests/HeaderBagTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/Tests/IpUtilsTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/Tests/JsonResponseTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/Tests/ParameterBagTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/Tests/RedirectResponseTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/Tests/RequestMatcherTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/Tests/RequestStackTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/Tests/RequestTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/Tests/ResponseHeaderBagTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/Tests/ResponseTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/Tests/ResponseTestCase.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/Tests/ServerBagTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/Tests/Session/Attribute/AttributeBagTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/Tests/Session/Attribute/NamespacedAttributeBagTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/Tests/Session/Flash/AutoExpireFlashBagTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/Tests/Session/Flash/FlashBagTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/Tests/Session/SessionTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/Tests/Session/Storage/Handler/AbstractSessionHandlerTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/Tests/Session/Storage/Handler/Fixtures/common.inc create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/Tests/Session/Storage/Handler/Fixtures/empty_destroys.expected create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/Tests/Session/Storage/Handler/Fixtures/empty_destroys.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/Tests/Session/Storage/Handler/Fixtures/read_only.expected create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/Tests/Session/Storage/Handler/Fixtures/read_only.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/Tests/Session/Storage/Handler/Fixtures/regenerate.expected create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/Tests/Session/Storage/Handler/Fixtures/regenerate.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/Tests/Session/Storage/Handler/Fixtures/storage.expected create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/Tests/Session/Storage/Handler/Fixtures/storage.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/Tests/Session/Storage/Handler/Fixtures/with_cookie.expected create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/Tests/Session/Storage/Handler/Fixtures/with_cookie.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/Tests/Session/Storage/Handler/Fixtures/with_cookie_and_session.expected create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/Tests/Session/Storage/Handler/Fixtures/with_cookie_and_session.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/Tests/Session/Storage/Handler/MemcacheSessionHandlerTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/Tests/Session/Storage/Handler/MemcachedSessionHandlerTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/Tests/Session/Storage/Handler/MongoDbSessionHandlerTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/Tests/Session/Storage/Handler/NativeFileSessionHandlerTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/Tests/Session/Storage/Handler/NativeSessionHandlerTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/Tests/Session/Storage/Handler/NullSessionHandlerTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/Tests/Session/Storage/Handler/PdoSessionHandlerTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/Tests/Session/Storage/Handler/StrictSessionHandlerTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/Tests/Session/Storage/Handler/WriteCheckSessionHandlerTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/Tests/Session/Storage/MetadataBagTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/Tests/Session/Storage/MockArraySessionStorageTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/Tests/Session/Storage/MockFileSessionStorageTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/Tests/Session/Storage/NativeSessionStorageTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/Tests/Session/Storage/PhpBridgeSessionStorageTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/Tests/Session/Storage/Proxy/AbstractProxyTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/Tests/Session/Storage/Proxy/NativeProxyTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/Tests/Session/Storage/Proxy/SessionHandlerProxyTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/Tests/StreamedResponseTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/Tests/schema/http-status-codes.rng create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/Tests/schema/iana-registry.rng create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/composer.json create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/http-foundation/phpunit.xml.dist create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/polyfill-mbstring/LICENSE create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/polyfill-mbstring/Mbstring.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/polyfill-mbstring/README.md create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/polyfill-mbstring/Resources/unidata/lowerCase.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/polyfill-mbstring/Resources/unidata/upperCase.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/polyfill-mbstring/bootstrap.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/polyfill-mbstring/composer.json create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/polyfill-php70/LICENSE create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/polyfill-php70/Php70.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/polyfill-php70/README.md create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/polyfill-php70/Resources/stubs/ArithmeticError.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/polyfill-php70/Resources/stubs/AssertionError.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/polyfill-php70/Resources/stubs/DivisionByZeroError.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/polyfill-php70/Resources/stubs/Error.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/polyfill-php70/Resources/stubs/ParseError.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/polyfill-php70/Resources/stubs/SessionUpdateTimestampHandlerInterface.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/polyfill-php70/Resources/stubs/TypeError.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/polyfill-php70/bootstrap.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/polyfill-php70/composer.json create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/process/.gitignore create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/process/CHANGELOG.md create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/process/Exception/ExceptionInterface.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/process/Exception/InvalidArgumentException.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/process/Exception/LogicException.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/process/Exception/ProcessFailedException.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/process/Exception/ProcessTimedOutException.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/process/Exception/RuntimeException.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/process/ExecutableFinder.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/process/InputStream.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/process/LICENSE create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/process/PhpExecutableFinder.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/process/PhpProcess.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/process/Pipes/AbstractPipes.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/process/Pipes/PipesInterface.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/process/Pipes/UnixPipes.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/process/Pipes/WindowsPipes.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/process/Process.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/process/ProcessBuilder.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/process/ProcessUtils.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/process/README.md create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/process/Tests/ExecutableFinderTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/process/Tests/NonStopableProcess.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/process/Tests/PhpExecutableFinderTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/process/Tests/PhpProcessTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/process/Tests/PipeStdinInStdoutStdErrStreamSelect.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/process/Tests/ProcessBuilderTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/process/Tests/ProcessFailedExceptionTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/process/Tests/ProcessTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/process/Tests/ProcessUtilsTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/process/Tests/SignalListener.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/process/composer.json create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/process/phpunit.xml.dist create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/.gitignore create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Annotation/Route.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/CHANGELOG.md create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/CompiledRoute.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/DependencyInjection/RoutingResolverPass.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Exception/ExceptionInterface.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Exception/InvalidParameterException.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Exception/MethodNotAllowedException.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Exception/MissingMandatoryParametersException.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Exception/NoConfigurationException.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Exception/ResourceNotFoundException.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Exception/RouteNotFoundException.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Generator/ConfigurableRequirementsInterface.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Generator/Dumper/GeneratorDumper.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Generator/Dumper/GeneratorDumperInterface.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Generator/Dumper/PhpGeneratorDumper.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Generator/UrlGenerator.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Generator/UrlGeneratorInterface.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/LICENSE create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Loader/AnnotationClassLoader.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Loader/AnnotationDirectoryLoader.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Loader/AnnotationFileLoader.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Loader/ClosureLoader.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Loader/Configurator/CollectionConfigurator.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Loader/Configurator/ImportConfigurator.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Loader/Configurator/RouteConfigurator.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Loader/Configurator/RoutingConfigurator.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Loader/Configurator/Traits/AddTrait.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Loader/Configurator/Traits/RouteTrait.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Loader/DependencyInjection/ServiceRouterLoader.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Loader/DirectoryLoader.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Loader/GlobFileLoader.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Loader/ObjectRouteLoader.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Loader/PhpFileLoader.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Loader/XmlFileLoader.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Loader/YamlFileLoader.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Loader/schema/routing/routing-1.0.xsd create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Matcher/Dumper/DumperCollection.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Matcher/Dumper/DumperRoute.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Matcher/Dumper/MatcherDumper.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Matcher/Dumper/MatcherDumperInterface.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Matcher/Dumper/PhpMatcherDumper.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Matcher/Dumper/StaticPrefixCollection.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Matcher/RedirectableUrlMatcher.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Matcher/RedirectableUrlMatcherInterface.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Matcher/RequestMatcherInterface.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Matcher/TraceableUrlMatcher.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Matcher/UrlMatcher.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Matcher/UrlMatcherInterface.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/README.md create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/RequestContext.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/RequestContextAwareInterface.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Route.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/RouteCollection.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/RouteCollectionBuilder.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/RouteCompiler.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/RouteCompilerInterface.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Router.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/RouterInterface.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Tests/Annotation/RouteTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Tests/CompiledRouteTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Tests/DependencyInjection/RoutingResolverPassTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Tests/Fixtures/AnnotatedClasses/AbstractClass.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Tests/Fixtures/AnnotatedClasses/BarClass.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Tests/Fixtures/AnnotatedClasses/BazClass.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Tests/Fixtures/AnnotatedClasses/FooClass.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Tests/Fixtures/AnnotatedClasses/FooTrait.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Tests/Fixtures/CustomCompiledRoute.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Tests/Fixtures/CustomRouteCompiler.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Tests/Fixtures/CustomXmlFileLoader.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Tests/Fixtures/OtherAnnotatedClasses/AnonymousClassInTrait.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Tests/Fixtures/OtherAnnotatedClasses/NoStartTagClass.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Tests/Fixtures/OtherAnnotatedClasses/VariadicClass.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Tests/Fixtures/RedirectableUrlMatcher.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Tests/Fixtures/annotated.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Tests/Fixtures/bad_format.yml create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Tests/Fixtures/bar.xml create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Tests/Fixtures/controller/import__controller.xml create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Tests/Fixtures/controller/import__controller.yml create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Tests/Fixtures/controller/import_controller.xml create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Tests/Fixtures/controller/import_controller.yml create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Tests/Fixtures/controller/import_override_defaults.xml create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Tests/Fixtures/controller/import_override_defaults.yml create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Tests/Fixtures/controller/override_defaults.xml create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Tests/Fixtures/controller/override_defaults.yml create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Tests/Fixtures/controller/routing.xml create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Tests/Fixtures/controller/routing.yml create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Tests/Fixtures/directory/recurse/routes1.yml create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Tests/Fixtures/directory/recurse/routes2.yml create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Tests/Fixtures/directory/routes3.yml create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Tests/Fixtures/directory_import/import.yml create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Tests/Fixtures/dumper/url_matcher0.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Tests/Fixtures/dumper/url_matcher1.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Tests/Fixtures/dumper/url_matcher2.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Tests/Fixtures/dumper/url_matcher3.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Tests/Fixtures/dumper/url_matcher4.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Tests/Fixtures/dumper/url_matcher5.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Tests/Fixtures/dumper/url_matcher6.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Tests/Fixtures/dumper/url_matcher7.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Tests/Fixtures/empty.yml create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Tests/Fixtures/file_resource.yml create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Tests/Fixtures/foo.xml create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Tests/Fixtures/foo1.xml create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Tests/Fixtures/glob/bar.xml create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Tests/Fixtures/glob/bar.yml create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Tests/Fixtures/glob/baz.xml create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Tests/Fixtures/glob/baz.yml create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Tests/Fixtures/glob/import_multiple.xml create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Tests/Fixtures/glob/import_multiple.yml create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Tests/Fixtures/glob/import_single.xml create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Tests/Fixtures/glob/import_single.yml create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Tests/Fixtures/glob/php_dsl.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Tests/Fixtures/glob/php_dsl_bar.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Tests/Fixtures/glob/php_dsl_baz.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Tests/Fixtures/incomplete.yml create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Tests/Fixtures/list_defaults.xml create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Tests/Fixtures/list_in_list_defaults.xml create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Tests/Fixtures/list_in_map_defaults.xml create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Tests/Fixtures/list_null_values.xml create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Tests/Fixtures/map_defaults.xml create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Tests/Fixtures/map_in_list_defaults.xml create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Tests/Fixtures/map_in_map_defaults.xml create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Tests/Fixtures/map_null_values.xml create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Tests/Fixtures/missing_id.xml create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Tests/Fixtures/missing_path.xml create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Tests/Fixtures/namespaceprefix.xml create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Tests/Fixtures/nonesense_resource_plus_path.yml create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Tests/Fixtures/nonesense_type_without_resource.yml create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Tests/Fixtures/nonvalid.xml create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Tests/Fixtures/nonvalid.yml create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Tests/Fixtures/nonvalid2.yml create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Tests/Fixtures/nonvalidkeys.yml create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Tests/Fixtures/nonvalidnode.xml create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Tests/Fixtures/nonvalidroute.xml create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Tests/Fixtures/null_values.xml create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Tests/Fixtures/php_dsl.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Tests/Fixtures/php_dsl_sub.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Tests/Fixtures/scalar_defaults.xml create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Tests/Fixtures/special_route_name.yml create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Tests/Fixtures/validpattern.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Tests/Fixtures/validpattern.xml create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Tests/Fixtures/validpattern.yml create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Tests/Fixtures/validresource.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Tests/Fixtures/validresource.xml create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Tests/Fixtures/validresource.yml create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Tests/Fixtures/with_define_path_variable.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Tests/Fixtures/withdoctype.xml create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Tests/Generator/Dumper/PhpGeneratorDumperTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Tests/Generator/UrlGeneratorTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Tests/Loader/AbstractAnnotationLoaderTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Tests/Loader/AnnotationClassLoaderTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Tests/Loader/AnnotationDirectoryLoaderTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Tests/Loader/AnnotationFileLoaderTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Tests/Loader/ClosureLoaderTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Tests/Loader/DirectoryLoaderTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Tests/Loader/GlobFileLoaderTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Tests/Loader/ObjectRouteLoaderTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Tests/Loader/PhpFileLoaderTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Tests/Loader/XmlFileLoaderTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Tests/Loader/YamlFileLoaderTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Tests/Matcher/DumpedRedirectableUrlMatcherTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Tests/Matcher/DumpedUrlMatcherTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Tests/Matcher/Dumper/DumperCollectionTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Tests/Matcher/Dumper/PhpMatcherDumperTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Tests/Matcher/Dumper/StaticPrefixCollectionTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Tests/Matcher/RedirectableUrlMatcherTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Tests/Matcher/TraceableUrlMatcherTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Tests/Matcher/UrlMatcherTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Tests/RequestContextTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Tests/RouteCollectionBuilderTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Tests/RouteCollectionTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Tests/RouteCompilerTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Tests/RouteTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/Tests/RouterTest.php create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/composer.json create mode 100644 main/app/sprinkles/core/assets/SiteAssets/php/vendor/symfony/routing/phpunit.xml.dist create mode 100755 main/app/sprinkles/core/assets/font-starcraft/css/font-starcraft.css create mode 100755 main/app/sprinkles/core/assets/font-starcraft/fonts/font-starcraft.eot create mode 100755 main/app/sprinkles/core/assets/font-starcraft/fonts/font-starcraft.svg create mode 100755 main/app/sprinkles/core/assets/font-starcraft/fonts/font-starcraft.ttf create mode 100755 main/app/sprinkles/core/assets/font-starcraft/fonts/font-starcraft.woff create mode 100755 main/app/sprinkles/core/assets/userfrosting/css/AdminLTE-skins-all.css create mode 100755 main/app/sprinkles/core/assets/userfrosting/css/AdminLTE.css create mode 100755 main/app/sprinkles/core/assets/userfrosting/css/tablesorter-reflow.css create mode 100755 main/app/sprinkles/core/assets/userfrosting/css/uf-alerts.css create mode 100755 main/app/sprinkles/core/assets/userfrosting/css/uf-collection.css create mode 100755 main/app/sprinkles/core/assets/userfrosting/css/uf-jqueryvalidation.css create mode 100755 main/app/sprinkles/core/assets/userfrosting/css/userfrosting.css create mode 100755 main/app/sprinkles/core/assets/userfrosting/favicons/README.md create mode 100755 main/app/sprinkles/core/assets/userfrosting/favicons/android-chrome-144x144.png create mode 100755 main/app/sprinkles/core/assets/userfrosting/favicons/android-chrome-192x192.png create mode 100755 main/app/sprinkles/core/assets/userfrosting/favicons/android-chrome-256x256.png create mode 100755 main/app/sprinkles/core/assets/userfrosting/favicons/android-chrome-36x36.png create mode 100755 main/app/sprinkles/core/assets/userfrosting/favicons/android-chrome-384x384.png create mode 100755 main/app/sprinkles/core/assets/userfrosting/favicons/android-chrome-48x48.png create mode 100755 main/app/sprinkles/core/assets/userfrosting/favicons/android-chrome-512x512.png create mode 100755 main/app/sprinkles/core/assets/userfrosting/favicons/android-chrome-72x72.png create mode 100755 main/app/sprinkles/core/assets/userfrosting/favicons/android-chrome-96x96.png create mode 100755 main/app/sprinkles/core/assets/userfrosting/favicons/apple-touch-icon-114x114-precomposed.png create mode 100755 main/app/sprinkles/core/assets/userfrosting/favicons/apple-touch-icon-114x114.png create mode 100755 main/app/sprinkles/core/assets/userfrosting/favicons/apple-touch-icon-120x120-precomposed.png create mode 100755 main/app/sprinkles/core/assets/userfrosting/favicons/apple-touch-icon-120x120.png create mode 100755 main/app/sprinkles/core/assets/userfrosting/favicons/apple-touch-icon-144x144-precomposed.png create mode 100755 main/app/sprinkles/core/assets/userfrosting/favicons/apple-touch-icon-144x144.png create mode 100755 main/app/sprinkles/core/assets/userfrosting/favicons/apple-touch-icon-152x152-precomposed.png create mode 100755 main/app/sprinkles/core/assets/userfrosting/favicons/apple-touch-icon-152x152.png create mode 100755 main/app/sprinkles/core/assets/userfrosting/favicons/apple-touch-icon-180x180-precomposed.png create mode 100755 main/app/sprinkles/core/assets/userfrosting/favicons/apple-touch-icon-180x180.png create mode 100755 main/app/sprinkles/core/assets/userfrosting/favicons/apple-touch-icon-57x57-precomposed.png create mode 100755 main/app/sprinkles/core/assets/userfrosting/favicons/apple-touch-icon-57x57.png create mode 100755 main/app/sprinkles/core/assets/userfrosting/favicons/apple-touch-icon-60x60-precomposed.png create mode 100755 main/app/sprinkles/core/assets/userfrosting/favicons/apple-touch-icon-60x60.png create mode 100755 main/app/sprinkles/core/assets/userfrosting/favicons/apple-touch-icon-72x72-precomposed.png create mode 100755 main/app/sprinkles/core/assets/userfrosting/favicons/apple-touch-icon-72x72.png create mode 100755 main/app/sprinkles/core/assets/userfrosting/favicons/apple-touch-icon-76x76-precomposed.png create mode 100755 main/app/sprinkles/core/assets/userfrosting/favicons/apple-touch-icon-76x76.png create mode 100755 main/app/sprinkles/core/assets/userfrosting/favicons/apple-touch-icon-precomposed.png create mode 100755 main/app/sprinkles/core/assets/userfrosting/favicons/apple-touch-icon.png create mode 100755 main/app/sprinkles/core/assets/userfrosting/favicons/favicon-16x16.png create mode 100755 main/app/sprinkles/core/assets/userfrosting/favicons/favicon-32x32.png create mode 100755 main/app/sprinkles/core/assets/userfrosting/favicons/favicon.ico create mode 100755 main/app/sprinkles/core/assets/userfrosting/favicons/ieconfig.xml create mode 100755 main/app/sprinkles/core/assets/userfrosting/favicons/manifest.json create mode 100755 main/app/sprinkles/core/assets/userfrosting/favicons/mstile-144x144.png create mode 100755 main/app/sprinkles/core/assets/userfrosting/favicons/mstile-150x150.png create mode 100755 main/app/sprinkles/core/assets/userfrosting/favicons/mstile-310x150.png create mode 100755 main/app/sprinkles/core/assets/userfrosting/favicons/mstile-310x310.png create mode 100755 main/app/sprinkles/core/assets/userfrosting/favicons/mstile-70x70.png create mode 100755 main/app/sprinkles/core/assets/userfrosting/favicons/safari-pinned-tab.svg create mode 100755 main/app/sprinkles/core/assets/userfrosting/images/cupcake.png create mode 100755 main/app/sprinkles/core/assets/userfrosting/images/logo.svg create mode 100755 main/app/sprinkles/core/assets/userfrosting/js/AdminLTE-custom.js create mode 100755 main/app/sprinkles/core/assets/userfrosting/js/AdminLTE.js create mode 100755 main/app/sprinkles/core/assets/userfrosting/js/attrchange.js create mode 100755 main/app/sprinkles/core/assets/userfrosting/js/fortress-jqueryvalidation-methods.js create mode 100755 main/app/sprinkles/core/assets/userfrosting/js/handlebars-helpers.js create mode 100755 main/app/sprinkles/core/assets/userfrosting/js/query-string.js create mode 100755 main/app/sprinkles/core/assets/userfrosting/js/tablesorter/widget-sort2Hash.js create mode 100755 main/app/sprinkles/core/assets/userfrosting/js/uf-alerts.js create mode 100755 main/app/sprinkles/core/assets/userfrosting/js/uf-captcha.js create mode 100755 main/app/sprinkles/core/assets/userfrosting/js/uf-collection.js create mode 100755 main/app/sprinkles/core/assets/userfrosting/js/uf-copy.js create mode 100755 main/app/sprinkles/core/assets/userfrosting/js/uf-form.js create mode 100755 main/app/sprinkles/core/assets/userfrosting/js/uf-init.js create mode 100755 main/app/sprinkles/core/assets/userfrosting/js/uf-jqueryvalidation-config.js create mode 100755 main/app/sprinkles/core/assets/userfrosting/js/uf-modal.js create mode 100755 main/app/sprinkles/core/assets/userfrosting/js/uf-table.js create mode 100755 main/app/sprinkles/core/assets/userfrosting/js/uf-tablesorter-parsers.js create mode 100755 main/app/sprinkles/core/bower.json create mode 100755 main/app/sprinkles/core/composer.json create mode 100755 main/app/sprinkles/core/config/default.php create mode 100755 main/app/sprinkles/core/config/dev.php create mode 100755 main/app/sprinkles/core/config/production.php create mode 100755 main/app/sprinkles/core/config/testing.php create mode 100755 main/app/sprinkles/core/extra/adjectives.php create mode 100755 main/app/sprinkles/core/extra/nouns.php create mode 100755 main/app/sprinkles/core/locale/ar/errors.php create mode 100755 main/app/sprinkles/core/locale/ar/messages.php create mode 100755 main/app/sprinkles/core/locale/ar/validate.php create mode 100755 main/app/sprinkles/core/locale/de_DE/errors.php create mode 100755 main/app/sprinkles/core/locale/de_DE/messages.php create mode 100755 main/app/sprinkles/core/locale/de_DE/validate.php create mode 100755 main/app/sprinkles/core/locale/en_US/errors.php create mode 100755 main/app/sprinkles/core/locale/en_US/messages.php create mode 100755 main/app/sprinkles/core/locale/en_US/validate.php create mode 100755 main/app/sprinkles/core/locale/es_ES/errors.php create mode 100755 main/app/sprinkles/core/locale/es_ES/messages.php create mode 100755 main/app/sprinkles/core/locale/es_ES/validate.php create mode 100755 main/app/sprinkles/core/locale/fa/errors.php create mode 100755 main/app/sprinkles/core/locale/fa/messages.php create mode 100755 main/app/sprinkles/core/locale/fa/validate.php create mode 100755 main/app/sprinkles/core/locale/fr_FR/errors.php create mode 100755 main/app/sprinkles/core/locale/fr_FR/messages.php create mode 100755 main/app/sprinkles/core/locale/fr_FR/validate.php create mode 100755 main/app/sprinkles/core/locale/it_IT/errors.php create mode 100755 main/app/sprinkles/core/locale/it_IT/messages.php create mode 100755 main/app/sprinkles/core/locale/it_IT/validate.php create mode 100755 main/app/sprinkles/core/locale/pt_PT/errors.php create mode 100755 main/app/sprinkles/core/locale/pt_PT/messages.php create mode 100755 main/app/sprinkles/core/locale/pt_PT/validate.php create mode 100755 main/app/sprinkles/core/locale/ru_RU/errors.php create mode 100755 main/app/sprinkles/core/locale/ru_RU/messages.php create mode 100755 main/app/sprinkles/core/locale/ru_RU/validate.php create mode 100755 main/app/sprinkles/core/locale/th_TH/errors.php create mode 100755 main/app/sprinkles/core/locale/th_TH/messages.php create mode 100755 main/app/sprinkles/core/locale/th_TH/validate.php create mode 100755 main/app/sprinkles/core/locale/valitron/ar.php create mode 100755 main/app/sprinkles/core/locale/valitron/de.php create mode 100755 main/app/sprinkles/core/locale/valitron/el.php create mode 100755 main/app/sprinkles/core/locale/valitron/en.php create mode 100755 main/app/sprinkles/core/locale/valitron/es.php create mode 100755 main/app/sprinkles/core/locale/valitron/fr.php create mode 100755 main/app/sprinkles/core/locale/valitron/id.php create mode 100755 main/app/sprinkles/core/locale/valitron/it.php create mode 100755 main/app/sprinkles/core/locale/valitron/ja.php create mode 100755 main/app/sprinkles/core/locale/valitron/lv.php create mode 100755 main/app/sprinkles/core/locale/valitron/pt-br.php create mode 100755 main/app/sprinkles/core/locale/valitron/ro.php create mode 100755 main/app/sprinkles/core/locale/valitron/ru.php create mode 100755 main/app/sprinkles/core/locale/valitron/th.php create mode 100755 main/app/sprinkles/core/locale/valitron/zh-cn.php create mode 100755 main/app/sprinkles/core/locale/valitron/zh-tw.php create mode 100755 main/app/sprinkles/core/locale/zh_CN/errors.php create mode 100755 main/app/sprinkles/core/locale/zh_CN/messages.php create mode 100755 main/app/sprinkles/core/locale/zh_CN/validate.php create mode 100755 main/app/sprinkles/core/routes/routes.php create mode 100755 main/app/sprinkles/core/schema/.gitkeep create mode 100755 main/app/sprinkles/core/src/Alert/AlertStream.php create mode 100755 main/app/sprinkles/core/src/Alert/CacheAlertStream.php create mode 100755 main/app/sprinkles/core/src/Alert/SessionAlertStream.php create mode 100755 main/app/sprinkles/core/src/Controller/CoreController.php create mode 100755 main/app/sprinkles/core/src/Controller/SimpleController.php create mode 100755 main/app/sprinkles/core/src/Core.php create mode 100755 main/app/sprinkles/core/src/Database/Builder.php create mode 100755 main/app/sprinkles/core/src/Database/DatabaseInvalidException.php create mode 100755 main/app/sprinkles/core/src/Database/Migrations/v400/SessionsTable.php create mode 100755 main/app/sprinkles/core/src/Database/Migrations/v400/ThrottlesTable.php create mode 100755 main/app/sprinkles/core/src/Database/Models/Concerns/HasRelationships.php create mode 100755 main/app/sprinkles/core/src/Database/Models/Model.php create mode 100755 main/app/sprinkles/core/src/Database/Models/Throttle.php create mode 100755 main/app/sprinkles/core/src/Database/Relations/BelongsToManyConstrained.php create mode 100755 main/app/sprinkles/core/src/Database/Relations/BelongsToManyThrough.php create mode 100755 main/app/sprinkles/core/src/Database/Relations/BelongsToManyUnique.php create mode 100755 main/app/sprinkles/core/src/Database/Relations/Concerns/Syncable.php create mode 100755 main/app/sprinkles/core/src/Database/Relations/Concerns/Unique.php create mode 100755 main/app/sprinkles/core/src/Database/Relations/HasManySyncable.php create mode 100755 main/app/sprinkles/core/src/Database/Relations/MorphManySyncable.php create mode 100755 main/app/sprinkles/core/src/Database/Relations/MorphToManyUnique.php create mode 100755 main/app/sprinkles/core/src/Error/ExceptionHandlerManager.php create mode 100755 main/app/sprinkles/core/src/Error/Handler/ExceptionHandler.php create mode 100755 main/app/sprinkles/core/src/Error/Handler/ExceptionHandlerInterface.php create mode 100755 main/app/sprinkles/core/src/Error/Handler/HttpExceptionHandler.php create mode 100755 main/app/sprinkles/core/src/Error/Handler/NotFoundExceptionHandler.php create mode 100755 main/app/sprinkles/core/src/Error/Handler/PhpMailerExceptionHandler.php create mode 100755 main/app/sprinkles/core/src/Error/Renderer/ErrorRenderer.php create mode 100755 main/app/sprinkles/core/src/Error/Renderer/ErrorRendererInterface.php create mode 100755 main/app/sprinkles/core/src/Error/Renderer/HtmlRenderer.php create mode 100755 main/app/sprinkles/core/src/Error/Renderer/JsonRenderer.php create mode 100755 main/app/sprinkles/core/src/Error/Renderer/PlainTextRenderer.php create mode 100755 main/app/sprinkles/core/src/Error/Renderer/WhoopsRenderer.php create mode 100755 main/app/sprinkles/core/src/Error/Renderer/XmlRenderer.php create mode 100755 main/app/sprinkles/core/src/Facades/Debug.php create mode 100755 main/app/sprinkles/core/src/Facades/Translator.php create mode 100755 main/app/sprinkles/core/src/Http/Concerns/DeterminesContentType.php create mode 100755 main/app/sprinkles/core/src/Log/DatabaseHandler.php create mode 100755 main/app/sprinkles/core/src/Log/MixedFormatter.php create mode 100755 main/app/sprinkles/core/src/Mail/EmailRecipient.php create mode 100755 main/app/sprinkles/core/src/Mail/MailMessage.php create mode 100755 main/app/sprinkles/core/src/Mail/Mailer.php create mode 100755 main/app/sprinkles/core/src/Mail/StaticMailMessage.php create mode 100755 main/app/sprinkles/core/src/Mail/TwigMailMessage.php create mode 100755 main/app/sprinkles/core/src/Model/UFModel.php create mode 100755 main/app/sprinkles/core/src/Router.php create mode 100755 main/app/sprinkles/core/src/ServicesProvider/ServicesProvider.php create mode 100755 main/app/sprinkles/core/src/Sprunje/Sprunje.php create mode 100755 main/app/sprinkles/core/src/Throttle/ThrottleRule.php create mode 100755 main/app/sprinkles/core/src/Throttle/Throttler.php create mode 100755 main/app/sprinkles/core/src/Throttle/ThrottlerException.php create mode 100755 main/app/sprinkles/core/src/Twig/CacheHelper.php create mode 100755 main/app/sprinkles/core/src/Twig/CoreExtension.php create mode 100755 main/app/sprinkles/core/src/Util/BadClassNameException.php create mode 100755 main/app/sprinkles/core/src/Util/Captcha.php create mode 100755 main/app/sprinkles/core/src/Util/CheckEnvironment.php create mode 100755 main/app/sprinkles/core/src/Util/ClassMapper.php create mode 100755 main/app/sprinkles/core/src/Util/EnvironmentInfo.php create mode 100755 main/app/sprinkles/core/src/Util/ShutdownHandler.php create mode 100755 main/app/sprinkles/core/src/Util/Util.php create mode 100755 main/app/sprinkles/core/templates/forms/csrf.html.twig create mode 100755 main/app/sprinkles/core/templates/mail/.gitkeep create mode 100755 main/app/sprinkles/core/templates/modals/modal.html.twig create mode 100755 main/app/sprinkles/core/templates/navigation/breadcrumb.html.twig create mode 100755 main/app/sprinkles/core/templates/navigation/main-nav.html.twig create mode 100755 main/app/sprinkles/core/templates/pages/about.html.twig create mode 100755 main/app/sprinkles/core/templates/pages/abstract/base.html.twig create mode 100755 main/app/sprinkles/core/templates/pages/abstract/default.html.twig create mode 100755 main/app/sprinkles/core/templates/pages/abstract/error.html.twig create mode 100644 main/app/sprinkles/core/templates/pages/abstract/mainsite.html.twig create mode 100755 main/app/sprinkles/core/templates/pages/error/400.html.twig create mode 100755 main/app/sprinkles/core/templates/pages/error/404.html.twig create mode 100755 main/app/sprinkles/core/templates/pages/error/config-errors.html.twig create mode 100755 main/app/sprinkles/core/templates/pages/index.html.twig create mode 100755 main/app/sprinkles/core/templates/pages/legal.html.twig create mode 100755 main/app/sprinkles/core/templates/pages/partials/alerts.html.twig create mode 100755 main/app/sprinkles/core/templates/pages/partials/analytics.html.twig create mode 100755 main/app/sprinkles/core/templates/pages/partials/config.js.twig create mode 100755 main/app/sprinkles/core/templates/pages/partials/favicons.html.twig create mode 100755 main/app/sprinkles/core/templates/pages/partials/footer.html.twig create mode 100755 main/app/sprinkles/core/templates/pages/partials/legal.html.twig create mode 100755 main/app/sprinkles/core/templates/pages/partials/page.js.twig create mode 100755 main/app/sprinkles/core/templates/pages/partials/privacy.html.twig create mode 100755 main/app/sprinkles/core/templates/pages/privacy.html.twig create mode 100755 main/app/sprinkles/core/templates/tables/table-paginated.html.twig create mode 100755 main/app/sprinkles/core/templates/tables/table-tool-menu.html.twig create mode 100755 main/app/sprinkles/core/tests/Integration/DatabaseTests.php create mode 100755 main/app/sprinkles/core/tests/Unit/BelongsToManyThroughTest.php create mode 100755 main/app/sprinkles/core/tests/Unit/DatabaseSyncableTest.php create mode 100755 main/app/sprinkles/core/tests/Unit/SprunjeTest.php create mode 100755 main/app/sprinkles/extend-user/.gitignore create mode 100755 main/app/sprinkles/extend-user/README.md create mode 100755 main/app/sprinkles/extend-user/composer.json create mode 100755 main/app/sprinkles/extend-user/routes/member.php create mode 100755 main/app/sprinkles/extend-user/schema/requests/user/create.yaml create mode 100755 main/app/sprinkles/extend-user/schema/requests/user/edit-info.yaml create mode 100755 main/app/sprinkles/extend-user/src/Controller/MemberController.php create mode 100755 main/app/sprinkles/extend-user/src/Database/Migrations/v400/MembersTable.php create mode 100755 main/app/sprinkles/extend-user/src/Database/Models/Member.php create mode 100755 main/app/sprinkles/extend-user/src/Database/Models/MemberAux.php create mode 100755 main/app/sprinkles/extend-user/src/Database/Scopes/MemberAuxScope.php create mode 100755 main/app/sprinkles/extend-user/src/ServicesProvider/ServicesProvider.php create mode 100755 main/app/sprinkles/extend-user/templates/forms/user.html.twig create mode 100755 main/app/sprinkles/extend-user/templates/pages/user.html.twig create mode 100755 main/app/system/Bakery/Bakery.php create mode 100755 main/app/system/Bakery/BaseCommand.php create mode 100755 main/app/system/Bakery/Command/Bake.php create mode 100755 main/app/system/Bakery/Command/BuildAssets.php create mode 100755 main/app/system/Bakery/Command/ClearCache.php create mode 100755 main/app/system/Bakery/Command/Debug.php create mode 100755 main/app/system/Bakery/Command/Migrate.php create mode 100755 main/app/system/Bakery/Command/MigrateRefresh.php create mode 100755 main/app/system/Bakery/Command/MigrateReset.php create mode 100755 main/app/system/Bakery/Command/MigrateRollback.php create mode 100755 main/app/system/Bakery/Command/Setup.php create mode 100755 main/app/system/Bakery/Command/Test.php create mode 100755 main/app/system/Bakery/DatabaseTest.php create mode 100755 main/app/system/Bakery/Migration.php create mode 100755 main/app/system/Bakery/Migrator.php create mode 100755 main/app/system/Database/Migrations/v410/MigrationTable.php create mode 100755 main/app/system/Database/Model/Migrations.php create mode 100755 main/app/system/Facade.php create mode 100755 main/app/system/ServicesProvider.php create mode 100755 main/app/system/SlimAppEvent.php create mode 100755 main/app/system/Sprinkle/Sprinkle.php create mode 100755 main/app/system/Sprinkle/SprinkleManager.php create mode 100755 main/app/system/UserFrosting.php create mode 100755 main/app/tests/DatabaseTransactions.php create mode 100755 main/app/tests/TestCase.php create mode 100755 main/app/tests/Unit/ExampleTest.php create mode 100755 main/bakery create mode 100755 main/build/before_install.sh create mode 100755 main/build/gulpfile.js create mode 100755 main/build/package-lock.json create mode 100755 main/build/package.json create mode 100755 main/composer.json create mode 100755 main/docker-compose.yml create mode 100755 main/docker/README.md create mode 100755 main/docker/nginx/Dockerfile create mode 100755 main/docker/nginx/default.conf create mode 100755 main/docker/node/Dockerfile create mode 100755 main/docker/php/Dockerfile create mode 100755 main/phpunit.xml create mode 100755 main/public/.htaccess create mode 100755 main/public/index.php create mode 100755 main/screenshots/login.png create mode 100755 main/screenshots/permissions.png create mode 100755 main/screenshots/users.png create mode 100755 main/sponsors/nextgi.png create mode 100755 main/sponsors/usor.png create mode 100755 main/webserver-configs/htaccess.txt create mode 100755 main/webserver-configs/nginx.conf create mode 100755 main/webserver-configs/web.config (limited to 'main') diff --git a/main/.github/CONTRIBUTING.md b/main/.github/CONTRIBUTING.md new file mode 100755 index 0000000..99b92e0 --- /dev/null +++ b/main/.github/CONTRIBUTING.md @@ -0,0 +1,76 @@ +# Guidelines for Getting Help with UserFrosting + +**Before** you open a new issue or ask a question in chat, you **must** read these guidelines. If it is evident from your issue that you failed to research your question properly, your issue may be closed without being answered. + +## Troubleshooting + +There are a few common stumbling blocks that new users face when setting up UserFrosting for the first time. If you are new to the current version of UserFrosting, please first look at the [basic requirements and installation instructions](https://learn.userfrosting.com/basics/requirements/basic-stack). + +If you don't find what you're looking for in the troubleshooting page, then please check the [wiki](https://github.com/userfrosting/UserFrosting/wiki) and [existing issues](https://github.com/alexweissman/UserFrosting/issues?utf8=%E2%9C%93&q=is%3Aissue), both opened and closed. Your question may have already been asked and answered before! + +You can also search for help on Stack Overflow or in our [Forums](https://forums.userfrosting.com/). In addition to the tags for the components that UF builds upon, such as [Slim](http://stackoverflow.com/questions/tagged/slim), [Twig](http://stackoverflow.com/questions/tagged/twig), [Eloquent](http://stackoverflow.com/questions/tagged/eloquent), [jQuery Validate](http://stackoverflow.com/questions/tagged/jquery-validate), [Select2](http://stackoverflow.com/questions/tagged/jquery-select2), there is now a [UserFrosting tag](http://stackoverflow.com/questions/tagged/userfrosting) as well. + +There are also tags for the utilities upon which UserFrosting depends, such as [Composer](http://stackoverflow.com/questions/tagged/composer-php) and [Git](http://stackoverflow.com/questions/tagged/git). + +## Asking for Help + +In general, the Github issue tracker should only be used for bug reports and feature requests. If you're just having trouble getting something to work, you should ask on Stack Overflow or in our [Forums](https://forums.userfrosting.com/) instead. Tag your question with the `userfrosting` tag, and optionally with any tags specific to the relevant underlying technologies, such as `slim`, `twig`, `eloquent`, `composer`, etc. You should also mention the version of UserFrosting that you are using. + +After posting a question on Stack Overflow or in our [Forums](https://forums.userfrosting.com/), please [link to it in chat](https://chat.userfrosting.com). This will ensure that more people see it, and provide a place where we can discuss and help clarify your question. + +On Github, Chat, and Stack Overflow, please keep in mind the following: + +1. Remember that courtesy and proper grammar go a long way. Please take the time to craft a **precise, polite issue**. We will do our best to help, but remember that this is an open-source project - none of us are getting paid a salary to develop this project, or act as your personal support hotline :wink: + +2. Report any errors in detail. Vague issues like "it doesn't work when I do this" are not helpful. Show that you have put some effort into identifying the cause of the error. + +3. There are three main places where you may find error messages: + +- Backend (PHP-related) fatal errors: in your PHP error log. This is usually a file called `php_error_log` or something like that. In XAMPP, the default location of this file is `XAMPP/xamppfiles/logs/`. For other web hosting platforms, please consult the documentation or do a quick Google search (i.e. "where is the php error log in _____"). Some web hosts may provide a special interface for accessing the php error log, through ssh, cpanel, etc. Please ask them directly for help with this. + +- Non-fatal PHP errors will be logged in your UserFrosting error log. Look for your `app/logs/errors.log` file. + +- Frontend (Javascript-related) errors: in your browser's Javascript console. See [this guide](https://learn.userfrosting.com/background/client-side) to using your browser console. + +You should also try testing your code in a local development environment, to separate **code-related** issues from **server** issues. In general, we recommend that you install a local development server on your computer, rather than [testing your code directly on the production server](https://pbs.twimg.com/media/BxfENwpIYAAcHqQ.png). This means you can test your code directly on your own computer, making development faster and without the risk of exposing sensitive information to the public. We recommend installing [XAMPP](https://www.apachefriends.org) if you don't already have a local server set up. + +## Contributing to the Codebase + +We welcome your technical expertise! But first, please join us in [chat](https://chat.userfrosting.com) to discuss your proposed changes/fixes/enhancements before you get started. At least one member of our development team will usually be around. + +Please also be sure to read our [style guidelines](../STYLE-GUIDE.md). + +When it's time to integrate changes, our git flow more or less follows http://nvie.com/posts/a-successful-git-branching-model/. + +### Branches + +- `master`: The current release or release candidate. Always numbered as `major.minor.revision`, possibly with an `-alpha` or `-beta` extension as well. +- `develop`: During alpha/beta, contains major changes to a release candidate. After beta, contains breaking changes that will need to wait for the next version to be integrated. Always numbered as `major.minor.x`, possibly with an `-alpha` or `-beta` extension as well. + +### Changes + +#### Hotfixes + +Hotfixes should be created in a separate branch, and then merged into both **master** and **develop**. + +#### Features + +New features that introduce some breaking changes should be created in a separate branch. When they are ready, they can be merged into `develop`. + +### Releases + +After every release, the `master` branch (and possibly `develop`, for minor/major releases) should immediately be version-bumped. That way, new changes can be accumulated until the next release. + +When a new version is created, the version number need to be changed in `app/define.php`. `CHANGELOG.md` should also be updated and the associated tag should be created on Github. + +#### Alpha/beta releases + +During alpha/beta, a release candidate sits on the `master` branch. Minor improvements should be treated as hotfixes, while major changes should be treated as features. In alpha/beta, major changes can still be integrated into `master` from `develop`. However, this should bump the revision number instead of the minor/major number. + +## Building the API documentation + +To build the API documentation, install [ApiGen](http://www.apigen.org/) globally and then run: + +`apigen generate --source UserFrosting/app,userfrosting-assets/src,userfrosting-config/Config,userfrosting-fortress/Fortress,userfrosting-i18n/I18n,userfrosting-session/Session,userfrosting-support/Support --destination userfrosting-api --exclude *vendor*,*_meta* --template-theme "bootstrap"` + +from inside your dev directory. diff --git a/main/.github/ISSUE_TEMPLATE.md b/main/.github/ISSUE_TEMPLATE.md new file mode 100755 index 0000000..e46fc9f --- /dev/null +++ b/main/.github/ISSUE_TEMPLATE.md @@ -0,0 +1,2 @@ + \ No newline at end of file diff --git a/main/.travis.yml b/main/.travis.yml new file mode 100755 index 0000000..4b516f7 --- /dev/null +++ b/main/.travis.yml @@ -0,0 +1,36 @@ +sudo: false +dist: trusty +language: php + +services: + - mysql + - postgresql + +php: + - 5.6 + - 7 + - 7.1 + +env: + matrix: + - DB=mysql + - DB=sqlite + - DB=pgsql + +before_install: + # copy sprinkles.json + - cp app/sprinkles.example.json app/sprinkles.json + # set up db + - bash build/before_install.sh $DB + +before_script: + # install deps and UF + - composer install + - php bakery migrate + +script: + # run unit tests + - composer test + +after_failure: + - cat app/log/userfrosting.log diff --git a/main/CHANGELOG.md b/main/CHANGELOG.md new file mode 100755 index 0000000..44f7530 --- /dev/null +++ b/main/CHANGELOG.md @@ -0,0 +1,553 @@ +# Change Log + +All notable changes to this project will be documented in this file. + +The format is based on [Keep a Changelog](http://keepachangelog.com/en/1.0.0/) +and this project adheres to [Semantic Versioning](http://semver.org/spec/v2.0.0.html). + +## v4.1.17-alpha +- Lock `gulp-uf-bundle-assets` at v2.28.0 until Silic0nS0ldier/gulp-uf-bundle-assets#5 is resolved (see #859) +- Add missing getInfo methods for GroupController and RoleController (#837) + +## v4.1.16-alpha +- Fix for `merge` bundling rule (#660) +- Fix for undefined variable exception under strict mode in `ufAlerts` (#809) +- Fix for site cache reset upon login (#828) +- Changed global cache tag to proper prefix +- Fix broken alert message in registration process (#843) +- Add partial Turkish translation + +## v4.1.15-alpha +- Refactor `Password` into a instantiable `Hasher` class, service, and `Password` facade (#827) +- Change default hash cost back to 10 and fix legacy hash detection issue + +## v4.1.14-alpha +- Fix issue with scopes being applied twice in `Unique::getPaginatedQuery` (https://github.com/userfrosting/extend-user/issues/2) +- Update Bower dependencies in core Sprinkle +- Refactor the `Password` class to use `hash_equals` for legacy passwords (prevent timing-based attacks) and factor out the default cost (#814) +- Check if `require_email_verification` is set in `Authenticator` and sign-in page (#815) +- Factor out hardcoded `sprinkles.json` filename (partially addresses #813) +- Add Farsi translations (#816) +- `ufTable`: Make `tableElement` configurable, and check for existence (#824) +- Put AdminLTE menu toggle button back (Revert e8a26fb and part of a46205f) + +## v4.1.13-alpha +- `ufTable`: Implement `rowTemplate` for customizing how rows are rendered (#787) +- `ufTable`: Support for passing callbacks for column templates instead of Handlebars templates +- Remove explicit references to `id` columns (#803) +- Fix Docker +- Append full name to User API + +## v4.1.12-alpha +- Separate out user action column from user status column +- Improve table row menus in mobile views (#724) +- Hide side menu toggle button in desktop sizes +- Add chevron to user menu +- Change "remember me" text +- Improve table tool buttons +- Twig extensions now implement `Twig_Extension_GlobalsInterface` as required by https://twig.symfony.com/doc/2.x/advanced.html#id1 (#788) +- Display element based on permissions for group list/info pages +- Factor the admin user creation out of migrations and into its own Bakery command (See #778) +- Bakery `clear-cache` command now clears Twig and router cache (Fix #750) +- Add Russian translations +- Add Travis for automated test/build on push + +## v4.1.11-alpha +- Updated `composer/installers` dependency. +- Patch `composer.json` to fix `illuminate/*` dependencies at 5.4 for now + +## v4.1.10-alpha +- Add support for PHP7 runtime errors to be handled in the same way as Exceptions +- Implement NotFoundExceptionHandler and pass through all NotFoundExceptions to this handler. +- Implement `redirect.onAlreadyLoggedIn` service (fixes #680) +- Deprecate `determineRedirectOnLogin` and replace with `redirect.onLogin` service +- Fix some PSR-2 compliance issues + +## v4.1.9-alpha +- Fixes #780, and more efficient way to collect ids in Unique::getPaginatedQuery +- Show "user deleted" in activities table (#782) +- Patched version of `widget-sort2Hash.js` to prevent writing extraneous browser history entries (#712) +- Improve handling of fatal/parse errors + +## v4.1.8-alpha +- Normalize paths to always have a leading slash when comparing against the CSRF blacklist (#775) (possible breaking change for some environments - please see updated docs at https://learn.userfrosting.com/routes-and-controllers/client-input/csrf-guard#blacklisting-routes) +- Set `display_errors` to `true` for development configs (#762), move php settings into a common `php` subkey in config files +- `ShutdownHandler` no longer responsible for logging fatal errors +- Set up PHP config values in `Core.php` instead of inside the `config` service definition. +- Reimplement `Builder::exclude` to maintain a list of excluded columns, and then automatically update list of columns to fetch in `get()` +- Deprecate `Model::queryBuilder` and `Model::export` +- Update nginx config file from spdy to http2 +- Add Pagespeed block (commented out) to nginx config file +- Make fpm-php7.0 the default CGI nginx config file + +## v4.1.7-alpha +- Add the `withTernary` method to the `Unique` trait to allow loading nested child models on ternary relationships. +- Add skip, take, limit, offset and deprecate withLimit and withOffset in `Unique` trait +- Support for `withPivot` on `Unique` relationships (in tertiary models) +- Factor out common code from `PermissionUserSprunje` into `UserSprunje` +- Rework internals of `Sprunje` to make it more testable. Filters, sorts, and paginations are now applied to a clone of the original queriable object. Deprecated `getResults` and added `getArray` and `getModels`. Result keys can now be customized. +- Table of user permissions on user info page +- Simplify by combining `permission-users.html.twig` into options on `users.html.twig` +- Add Chinese translations +- Deprecate User::exists() (#771) +- Add 'password' to hidden fields for User model +- Replace hardcoded `Builder` with classMapper reference + +## v4.1.6-alpha +- Fix missing permission check when `uri_account_settings` is not in role (#768) +- Add `getLastRow` method and `transformDropdownSelection` option to `ufCollection` +- Fix missing slug for permissions in "manage permissions" dropdown +- Add "manage permissions" to role page menu +- Factor out custom relation methods into `HasRelationships` trait on `Model` +- Add `withoutGlobalScopes` to `Syncable::sync` +- Add option to use `forceCreate` in `Syncable::sync` +- Add option to use custom key in `Syncable::sync` +- Complete redesign of `BelongsToManyThrough` - possible BC for a few people, as you now need to load the "via" models explicitly using `withVia`. This fixes a lot of issues with `BelongsToManyThrough`. +- Deprecate `BelongsToManyConstrained` +- Add `MorphToManyUnique` +- Integration tests now use an in-memory sqlite database (`test_integration`) by default + +## v4.1.5-alpha +- Spanish language support (#770) +- Show current filter in select-menu filters (#744) +- Cursor styling for ufCopy +- Transition overlay for ufTables +- Minor fix to ufTable.cleanHash +- Correctly target pager container in Tablesorter options +- Add table of users to role pages +- Fix issue with links in permission users table + +## v4.1.4-alpha +- Permissions rows get duplicated when upgrading from 4.0 to 4.1 (fix #759) +- Fix migrate:rollback not running down the migration in the correct order +- Updated type in `composer.json` for default sprinkles +- Added missing french translations & more default validation messages +- Bump Fortress version (fix #766) +- Support SQLite in Bakery setup +- Fix for PostgreSQL charset in Bakery (#745) + +## v4.1.3-alpha +- Add Italian translations +- Add `data-priority` attributes to built-in tables (#752) +- Use `this.slug` to avoid conflict with helper names (#748) +- Add block `table_search` to `table-paginated.html.twig` + +## v4.1.2-alpha +- Remove call to setFilters that was causing problems with pagination (#688) +- Update German translations and factor out some hardcoded text (#725) +- Update French translations +- Update Arabic translations (#733, #734, #735) + +## v4.1.1-alpha +- Fixed missing template in ExceptionHandler and `notFoundHandler` +- Migration rollback will throw a warning if a class is not found instead of aborting +- Temporary fix for trouble with `npm install` in Windows (#742) + +## v4.1.0-alpha +- Switch from pagination "plugin" to "widget" for Tablesorter. Allows us to update to the latest version of TS (fix #688, #715) +- Implement `WhoopsRenderer` for pretty debug pages. See (#674) +- Refactor error handling. Move responsibility for displayErrorDetails to handlers, and factor our ErrorRenderers. Addresses (#702) +- Move `composer.json` to root directory to allow installing UF via composer create-project +- Move `sprinkles.json` to app directory to make it easier to find +- Eliminate the `root` theme Sprinkle. Custom styling for the root user is now handled with some Twig logic in the `admin` Sprinkle (#726) +- Rename bundle.config.json -> asset-bundles.json (#726) +- Reorganize assets (#726) +- Heavily reorganize templates (#726) +- Move request schema from `schema/` to `schema/requests/` (#726) +- Factor out "system" classes from core Sprinkle +- Refactor overall application lifecycle; move main lifecycle into UserFrosting\System\UserFrosting +- SprinkleManager now better focused on a single responsibility +- Sprinkle initializer classes now use events to hook into application lifecycle +- Support for allowing Sprinkles to register middleware (#617) +- Automatically load Sprinkle service providers (see #636) +- Get rid of "implicit loading" for core Sprinkle - core is now just an ordinary Sprinkle like any other. +- The `sprinkles://` stream now represents a virtual filesystem for the root directory of each loaded sprinkle, rather than the `sprinkles/` directory itself. +- Separate out `localePathBuilder` from the `translator` service. Makes it easier to add/remove paths before actually loading the translations. +- Only present locale options with non-null names. +- Rebased ufTable and ufModal with new jQuery plugin template. (part of #646) +- Removed the search bar from the Dashboard layout +- Added Tablesorter pagination translation +- New Translator Facade +- New CLI tool (Bakery). +- New migration system based on bakery CLI +- Listable sprunjing +- Refactor groups and user routes (Fix #721) +- Added the `config` alert stream to save ufAlerts to the cache instead of sessions. Fix #633. The old `session` is still the default alertStream in 4.1. +- Added support for the Redis cache driver and refactored the cache config values. +- Added user and session cache. +- Common log file for db queries, auth checks, smtp, errors, and debug messages (#709). +- Use YAML as default format for request schema (#690) + +See [http://learn.userfrosting.com/upgrading/40-to-41](Upgrading 4.0.x to 4.1.x documentation) for complete list of changes and breaking changes. + +## v4.0.24-alpha +- Fixes to nginx config file, and add location block for LE acme challenge +- Fix JS errors when `#alerts-page` is not present on a page +- Fix hardcoded User class in AdminController (#753) +- Update message PASSWORD.FORGET.REQUEST_SENT (#749) + +## v4.0.23-alpha +- Set module dependency versions to ~4.0.0 instead of ^4.0.0 (since 4.1.x will introduce breaking changes) +- Fix bug in ufCollection + +## v4.0.22-alpha +- Fix issue where 'Change User Password' popup form couldn't handle specifying a new password. +- Display message when there are no results in `ufTable` + +## v4.0.21-alpha +- Implement reflow and column selector for tables (#670) +- Overhauled ufAlerts, improving efficiency, reliability, and fixed a discovered edge case that caused `render` to never complete. (part of #646) +- ufAlerts will only auto-scroll when outside the viewport (even if only partially). Can be overriden with `scrollWhenVisible: true`. (#714) +- Rebased ufCollection, and ufForm with new jQuery plugin template. (part of #646) +- Misc UI update +- Added Twig blocks +- Fix issue with duplicate query logs when using multiple databases + +## v4.0.20-alpha +- Remove pivot columns from pagination subquery in BelongsToManyThrough, to deal with MySQL's `only_full_group_by` warning + +## v4.0.19-alpha +- Explicit column names in new user permissions relations + +## v4.0.18-alpha +- Permission info page (#638) +- New custom relationships 'BelongsToManyThrough', 'BelongsToManyUnique', 'BelongsToManyConstrained', 'HasManySyncable', 'MorphManySyncable' +- Change implementation of User::permissions() to use BelongsToManyThrough +- New ufForm options: setting reqParams, encType, submittingText +- ufCollection now triggers a check for virgin rows when _any_ control is touched +- Fix issue with Sprunje when generating CSV with empty child collections (#697) +- Authorizer now correctly interprets string literals (#482) +- Authorizer now correctly interprets numeric types in access conditions. **Caution**: this causes the `equals()` callback to return true in situations where it would have (incorrectly) returned false before. For example, `equals(self.group_id,2)` would have returned false for users in group 2, because it was interpreting `2` as a string and then performing its strict comparison. It now (correctly) returns true. Notice that `equals(self.group_id,'2')`, on the other hand, will now return `false`. +- User object caches permissions loaded from DB to reduce number of queries (#612) +- Type declarations in authorization classes (#652) +- Fix issue with Twig debug (#356) +- Show disabled/unactivated badges on user info page + +## v4.0.17-alpha +- Add IIS config file (#371) +- New ufCollection now supports free text input mode +- New design and layout for user, group, and role summary boxes (also fixes #703) +- Registration page returns 404 when registration disabled (#705) + +## v4.0.16-alpha +- Add Docker configuration files +- Begin work on Bakery, the command-line debug tool +- Lock version of tablesorter due to breaking changes +- Fix bugs in GroupController and RoleController +- Fix bug in URLs for redirect-on-login +- Added UTF-8 as default mail charset + +## v4.0.15-alpha +- Prevent mixed content on demo pages +- Fixed some missing translations +- Fixed error in ufAlerts push method +- Fixed usage of hard coded path +- Fixed default OS option in migration script +- Prevents empty locale's from displaying as empty options in profile form +- Unignore .gitkeeps of directories that need to exist + +## v4.0.14-alpha +- Fix ajax.delay in ufCollection +- Fix missing translations +- Minor fix in French translation +- Fix alert margin when displayed inside a modal + +## v4.0.13-alpha +- Update to RememberMe 2.0 (https://github.com/userfrosting/UserFrosting/issues/635) +- Remove database checks, as they are apparently no longer needed (https://github.com/userfrosting/UserFrosting/issues/655) +- Bump dependencies + +## v4.0.12-alpha +- Separate out the registration and sign-in pages (https://github.com/userfrosting/UserFrosting/issues/657) **BC** +- Slightly change behavior of form validation icons +- Sprunje input validation (https://github.com/userfrosting/UserFrosting/issues/640) +- Sprunje sort/filter fields now must be explicitly listed in a whitelist (https://github.com/userfrosting/UserFrosting/issues/640) **BC** +- Errors from tablesorter now get displayed +- Support for OR expressions using `||` in Sprunje filters (https://github.com/userfrosting/UserFrosting/issues/647) + +## v4.0.11-alpha +- Fix [#663](https://github.com/userfrosting/UserFrosting/issues/663) +- Adding more Twig `blocks` +- ufAlerts now scroll to alert location, if and only if alerts are output. +- Updated Dutch locale +- Minor update in French locale +- Added comments in `.env.example` + +## v4.0.10-alpha +- Move suggestion button outta-da-way +- Add email to registration success message +- Separate out some page content into smaller blocks +- Factor out select2 options in ufCollection, into the 'dropdown' key so that any select2 option can be set + +## v4.0.9-alpha +- Oops, `exists` needs to be static + +## v4.0.8-alpha +- Autogenerate and suggestion features for usernames during account registration (partially addresses https://github.com/userfrosting/UserFrosting/issues/569) +- Restrict username characters to a-z0-9.-_ +- Require first name by default +- Throttle registration attempts +- Implement User::exists method +- keyupDelay option in ufForm +- More logging of group and role CRUD +- Implement extra:// stream +- Lots of missing translation keys + +## v4.0.7-alpha +- Separate "profile settings" from "account settings" + +## v4.0.6-alpha +- Fix throttling issue #656 +- Other miscellaneous fixes + +## v4.0.5-alpha +- Allow nulling out of throttle rules (to disable) +- Disable Google Analytics by default (but enabled in production) +- Other miscellaneous fixes + +## v4.0.4-alpha +- UfAlert style customization (See [#634](https://github.com/userfrosting/UserFrosting/issues/634)) +- Translation function can now display raw placeholder using the `|raw` filter in the placeholder name. Other Twig filters are also avaiable. Requires latest version of the [i18n](https://github.com/userfrosting/i18n) component (See [#621](https://github.com/userfrosting/UserFrosting/issues/621)). +- Fix the "Root account" message breaking the UI on smaller screens (See [#641](https://github.com/userfrosting/UserFrosting/issues/641)) - Thanks @brunomnsilva ! +- Added `DB_DRIVER` and `DB_PORT` as environment variables to allow better out-of-box database configuration support, and to provide additional protection by obscurity. +- Normalised default values to `null` for environment variables in configuration. +- Added `getCallbacks` public method to `AuthorizationManager` to enable drop-in extensions to `AuthorizationManager`. +- Fixed broken links in generated asset bundles. +- Introduced `clean` gulp task to act as a shotcut for removing all frontend vendor packages, all generated asset bundles, and copied assets. Accessible via `npm run uf-clean`. +- Merged `copy` task with `bundle-build`. +- Fixed missing translations +- Added Thai translation - Thanks @popiazaza ! + +## v4.0.3-alpha +- Add config file for nginx (https://github.com/userfrosting/UserFrosting/issues/373) +- Add Portuguese translations (thanks to @brunomnsilva!) +- Add Arabic (MSA) translations (thanks to @abdullah.seba!) +- Add Dispatcher to db service to allow registering model events. +- Specify foreign keys explicitly in all relationships. +- Use classMapper for admin Sprunjes. + +## v4.0.2-alpha +- Specify foreign key explicitly in `User::activities()` relationship. +- Database checks in installer and Authenticator now respect custom database ports. (See [#628](https://github.com/userfrosting/UserFrosting/issues/628)) +- Fixed edge case where `5%C` would appear in generated urls. +- Improved stability and added php version check in `migrations/intall.php` +- Update ClassMapper to throw exception when class is not found +- Fix minor errors in French locale +- Fix translation error on the Legal page + +## v4.0.1-alpha +- Bump min version of PHP to 5.6 +- Added German translation (See [#625](https://github.com/userfrosting/UserFrosting/issues/625)) - Thanks @X-Anonymous-Y +- Improved Gulp Build task +- Remove site-dev from example sprinkles.json +- Fix some styling issues on the Dashboard and footer +- Display group link in menu for group admins +- Keep dashboard sidebar collapsed across page load (See [#616](https://github.com/userfrosting/UserFrosting/issues/616)) +- Fixed missing translation keys inside Handlebar tables (See [#624](https://github.com/userfrosting/UserFrosting/issues/624)) +- Admin panel link style in main dropdown menu (See [#627](https://github.com/userfrosting/UserFrosting/issues/627)) +- Implement AuthGuard middleware +- Handling of redirect after login (See [#627#issuecomment-275607492](https://github.com/userfrosting/UserFrosting/issues/627#issuecomment-275607492)) +- Directly check database in installer using PDO +- Refactor installer and how version are displayed in system info panel. Added notice when a migration is available for a sprinkle +- Etc. + +## v4.0.0-alpha + +**Initial release of UserFrosting V4** + +- The [Sprinkle](https://learn.userfrosting.com/sprinkles) system, which keeps your code completely separate from the core UF codebase; +- We're upgraded from Slim 2 to Slim 3, which is significantly different; +- Completely redesigned [database structure](https://learn.userfrosting.com/database/default-tables); +- Initialization is now achieved through [services](https://learn.userfrosting.com/services), with the Pimple dependency injection container; +- [Composer](https://learn.userfrosting.com/installation/requirements/essential-tools-for-php#composer) is now a mandatory part of the installation process; +- [Bower](https://learn.userfrosting.com/sprinkles/contents#-bower-json) is now used to install third-party client-side assets (Javascript/CSS packages); +- "Groups" and "Primary Group" have been replaced by "Roles" and "Group", respectively; +- Tables no longer need to be "registered" in any kind of initialization routine. Simply set the table names directly in your data models; +- Twig templates have been [reorganized and redesigned](https://learn.userfrosting.com/templating-with-twig/sprinkle-templates); +- SB Admin has been removed and we now use the [AdminLTE](https://adminlte.io/) front-end theme; +- Client-side code has been heavily refactored into reusable [components](https://learn.userfrosting.com/client-side-code/components). + +## v0.3.1.23 +- Also fix the `occurred_at` timestamp in the `user_event` table to allow null, for newer versions of MySQL that don't allow a zero date (see #605). + +## v0.3.1.22 +- Use `nullableTimestamps` instead of `timestamps` in installer, to prevent conflict with MySQL modes 'NO_ZERO_IN_DATE' and 'NO_ZERO_DATE'. + +## v0.3.1.21 +- Use Laravel's Schema interface to create tables and default rows, instead of constructing them with SQL + +## v0.3.1.20 +- Added `pushAlert()`,`clearAlerts()` in `public/js/userfrosting.js` and updated `flashAlerts()` +- Revert changes to User::fresh() but leave comment regarding upgrading Eloquent + +## v0.3.1.19 +- Fix some minor error screen layout issues +- Make User::fresh() compatible with Eloquent\Model v5.2.40+ +- Update composer require to allow for Fortress 1.x bugfixes +- Allow database port definitions in config-userfrosting.php +- Fix fatal error when evaluateCondition is called before the router populates current route information + +## v0.3.1.18 +- Add check for logging being enabled but log file not existing yet + +## v0.3.1.17 +- Fix occasional bug when end-of-file is reached on log file before requested number of lines is reached +- Roll back database connection checking to fix installer routines (frostbitten) +- UI fixes for smaller screens (frostbitten) +- Update Gitter references to Rocket.chat +- Clarify hotfix branch procedure for contributions + +## v0.3.1.16 +- Fix comment reference to \Fortress\JqueryValidationAdaptor +- CONTRIBUTING.md - Add note about proper Pull Requests +- French language file fixes (#565) (lcharette) +- Add HTTP status codes to 404 errors and database errors (frostbitten) +- Change database errors to use BaseController instead of DatabaseController (frostbitten) + +## v0.3.1.15 +- Fix unattached submitHandler bug in Group and Auth edit interfaces (#465) +- Remove references to nonexistent `formUserView` and `formGroupView` (#478) +- Gracefully handle session destruction due to missing or disabled accounts (#510) +- Add `attributeExists` and `relationExists` for models (#520) + +## v0.3.1.14 +- Stop reading entire log files to avoid out-of-memory errors (#497) +- Deploy [league/csv](https://github.com/thephpleague/csv) to properly generate CSV files (#557) +- Fix typos in language files + +## v0.3.1.13 +- Bump dependencies +- userfrosting/fortress now has a release version + +## v0.3.1.12 +- Add sendmail support in Notification class +- Fix problem with strict comparison in Handlebars templates and inconsistent data types among different database technologies +- Override paths to font files for Bootstrap Glyphicons to support the UserFrosting directory structure +- Added missing lines of Thai language (popiazaza) +- Fix a vulnerability where users still logged in wouldn't automatically be logged out if they were disabled +- Add option for HTTPS in `.htaccess`, commented out by default +- Minor syntax fixes in `public/js/userfrosting.js`, `widget-auth.js`, `widget-groups.js`, and `widget-users.js` + +## v0.3.1.11 +- Composer can now include composer.json files from plugin folders (added "wikimedia/composer-merge-plugin" to composer) + +## v0.3.1.10 +- Select correct versions (PHP 5.x compatible) of packages in `composer.json` +- Turkish language translation +- Return `User` object created in `AccountController::register` + +## v0.3.1.9 +- Revert to loose comparison for `user_id`s because of issues with Ubuntu's PDO driver (see http://stackoverflow.com/questions/5323146/mysql-integer-field-is-returned-as-string-in-php#comment41836471_5323169) + +## v0.3.1.8 +- Finish replacing all usages of `*Loader` classes with Eloquent syntax +- Installer warning for missing `imagepng` +- Fix bug in CSV generation for user table + +## v0.3.1.7 +- Change "default theme" to "guest theme" and fix loading issues (#463). What used to be called "default theme" is now base theme, i.e. the theme to fall back to when a template file cannot be found in the current theme (user group or guest theme) +- New public template for "nyx" theme +- Remove trailing slash from configuration JS/CSS paths to make uniform with site.uri.public +- Make routes for config.js and theme.css dynamically generated from configuration variables (#461) +- Make cookie name for "remember me" use session name +- Fix potential bug in configuration user_id's for guest, master accounts + +## v0.3.1.6 +- Fix exception-handling for mail server errors +- Notify if account creation was successful, even if mail server failed. + +## v0.3.1.5 +- Add Romanian translation +- Upgrade Tablesorter and pretty URLs for searched/sorted/paginated tables +- Fix bug in default value for user `secret_token` + +## v0.3.1.4 +- .htaccess redirect trailing slash: change to only redirect GET requests +- Natural sort order in API +- Fix bug in table pagination +- Fix bug in loading user primary group properties as user properties +- Fix mailto link bug in tables +- Warn if config file missing (#445) +- Fix dutch error (#447) + +## v0.3.1.3 +- Implement CSV download feature + +## v0.3.1.2 +- Implement `no_leading_whitespace` and `no_trailing_whitespace` rules + +## v0.3.1 +- Improved initialization routine as middleware +- Implemented "remember me" for persistent sessions - see https://github.com/gbirke/rememberme +- Converted page templates to inheritance architecture, using Twig `extends` +- Start using the `.twig` extension for template files +- All content is now part of a theme, and site can be configured so that one theme is the default theme for unauthenticated users +- User session stored via `user_id`, rather than the entire User object +- Data model is now built on Eloquent, instead of in-house +- Cleaned up some of the per-page Javascript, refactoring repetitive code +- Implement server-side pagination +- Upgrade to Tablesorter v2.23.4 +- Switch from DateJS to momentjs +- Switch to jQueryValidation from FormValidation +- Implement basic interface for modifying group authorization rules +- User events - timestamps for things like sign-in, sign-up, password reset, etc are now stored in a `user_event` table +- Wrapper class Notification for sending emails, other notifications to users +- Remove username requirement for password reset. It is more likely that an attacker would know the user's username, than the user themselves. For the next version, we can try to implement some real multi-factor authentication. +- When a user creates another user, they don't need to set a password. Instead, an email is sent out to the new user, with a token allowing them to set their own password. +- Admins can manually generate a password reset request for another user, or directly change the user's password. + +## v0.3.0 +- [Autoloading with Composer](https://v3.userfrosting.com/navigating/#composer) +- [MVC Architecture](https://v3.userfrosting.com/navigating/#structure) +- [Front Controllers and the Slim Microframework](https://v3.userfrosting.com/navigating/#slim) +- [Twig - Templating](http://twig.sensiolabs.org/) +- [Theming](https://v3.userfrosting.com/components/#theming) +- [Plugins](https://v3.userfrosting.com/components/#plugins) + +## v0.2.1 +- Implemented db-driven menu system. Menu items are pulled from the database, and can be modified via plugins. +- Implemented backend templating of forms and tables via [Bootsole](https://github.com/alexweissman/bootsole). + +## v0.2.0 (butterflyknife) +- Converted all DB calls to PDO. +- Renamed "permissions" to "groups". Same concept, but using the word "group" suggests that it can be used for more than just access control. +- Implemented "primary group" membership for users. A user can belong to multiple groups, but only one of those will be their primary group. +- Implemented DB-driven home pages for groups. Upon login, a user will be redirected to the `home_page` for their primary group. +- Implemented templated menus. Every group has a corresponding menu template in `models/menu-templates`. Upon login, the menu for a user's primary group is automatically loaded and rendered. +- Implemented function-level user authorization. Whenever a function in `secure_functions` is called, the `user_action_permits` table is checked to see whether or not that user has access to the function (the `action` column), conditional on the boolean functions specified in the `permits` column. +- Organized pages into four categories: account pages, API pages, form pages, and public pages. Public pages reside in the root directory and can be accessed by anyone. Account pages are in the `account` directory and are only accessible after logging in. API pages are in the `api` directory, and consist of all the pages that process or fetch data from the DB and interact with the frontend via AJAX/JSON. They are accessible by any logged in user, but will only perform a function if the user is authorized. Form pages are in the `forms` directory, and consist of pages that generate forms (for creating/updating users, groups, etc.) +- Converted registration page to AJAX. +- Improved installer with site configuration. + +## v0.1.7 +- Page scrolls back to top after AJAX submit. +- "Website url" is automatically suffixed with "/" if necessary. +- Fixed bad link to forgot_password.php. +- Began implementing action authorization scheme. + +## v0.1.6 +- Implemented CSRF token checking for creating and updating users +- Moved much of the nuts and bolts for generating the user-create and user-update forms to the server side, so as to streamline rendering process and require fewer requests by the client (see load_form_user.php) +- Improved responsive layout for rendering nicely on mobile devices + +## v0.1.5 +- More improvements to error-handling/rendering +- HTTPS/SSL compatible +- Fixed bug with different table name prefixes +- Improvements to CSRF tokens + +## v0.1.4 +- Updated password hashing from md5 to modern bcrypt (more secure) - thanks to contributor @r3wt +- Included better functions for sanitizing user input, validating user ip, generating csrf (cross-site request forgery) tokens - thanks to contributor @r3wt + +## v0.1.3 +- Root account (user id = 1) : created upon installation, cannot be deleted or disabled. +- Special color scheme for when logged in as root user. +- Installer now guides user through creation of root account +- Moved common JS and CSS includes to "includes.php" + +## v0.1.2 +- Improved error and exception handling +- Added 404 error page +- Standardized JSON interface for backend scripts +- Front-end should now be able to catch virtually any backend error and take an appropriate action (instead of white screen of death) diff --git a/main/LICENSE.md b/main/LICENSE.md new file mode 100755 index 0000000..476c3ca --- /dev/null +++ b/main/LICENSE.md @@ -0,0 +1,7 @@ +Copyright (c) 2017 by Alexander Weissman (https://alexanderweissman.com) + +UserFrosting is 100% free and open-source. + +Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the 'Software'), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions: The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED 'AS IS', WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. diff --git a/main/README.md b/main/README.md new file mode 100755 index 0000000..d136dc8 --- /dev/null +++ b/main/README.md @@ -0,0 +1,124 @@ +# UserFrosting 4.1 + +[https://www.userfrosting.com](https://www.userfrosting.com) + +[![Build Status](https://travis-ci.org/userfrosting/UserFrosting.svg?branch=master)](https://travis-ci.org/userfrosting/UserFrosting) +[![Join the chat at https://chat.userfrosting.com/channel/support](https://demo.rocket.chat/images/join-chat.svg)](https://chat.userfrosting.com/channel/support) +[![Backers on Open Collective](https://opencollective.com/userfrosting/backers/badge.svg)](#backers) [![Sponsors on Open Collective](https://opencollective.com/userfrosting/sponsors/badge.svg)](#sponsors) + + + + + +If you simply want to show that you like this project, or want to remember it for later, you should **star**, not **fork**, this repository. Forking is only for when you are ready to create your own copy of the code to work on. + +## By [Alex Weissman](https://alexanderweissman.com) + +Copyright (c) 2018, free to use in personal and commercial software as per the [license](LICENSE.md). + +UserFrosting is a secure, modern user management system written in PHP and built on top of the [Slim Microframework](http://www.slimframework.com/), [Twig](http://twig.sensiolabs.org/) templating engine, and [Eloquent](https://laravel.com/docs/5.4/eloquent#introduction) ORM. + +## Features + +### User login screen +![User login script](screenshots/login.png) + +### User management page +![PHP user management script](screenshots/users.png) + +### Permissions management page +![UserFrosting permissions management](screenshots/permissions.png) + +## [Demo](https://demo.userfrosting.com) + +## Installation + +Please see our [installation guide](https://learn.userfrosting.com/installation). + +## Troubleshooting + +If you are having trouble installing UserFrosting, please [join us in chat](https://chat.userfrosting.com) or try our [forums](https://forums.userfrosting.com). + +If you are generally confused about the structure and layout of the code, or it doesn't look like the kind of PHP code that you're used to, please [start from the beginning](https://learn.userfrosting.com/background). + +## Mission Objectives + +UserFrosting seeks to balance modern programming principles, like DRY and MVC, with a shallow learning curve for new developers. Our goals are to: + +- Create a fully-functioning user management script that can be set up in just a few minutes +- Make it easy for users to quickly adapt the code for their needs +- Introduce novice developers to best practices such as separation of concerns and DRY programming +- Introduce novice developers to modern constructs such as front-end controllers, RESTful URLs, namespacing, and object-oriented modeling +- Build on existing, widely used server- and client-side components +- Clean, consistent, and well-documented code + +## Documentation + +### [Learning UserFrosting](https://learn.userfrosting.com) + +### [API documentation](http://api.userfrosting.com) + +### [Change log](CHANGELOG.md) + +## Running tests + +Run `php bakery test` from the root project directory. Any tests included in `sprinkles/*/tests` will be run. + +## Development Team + +### Alexander Weissman + +Alex is the founder and co-owner of two companies, one that does [math tutoring at Indiana University](https://bloomingtontutors.com) in Bloomington, IN and another company that does [math tutoring at UMD](https://collegeparktutors.com) in College Park, MD. He is a PhD student in the School of Informatics and Computing at Indiana University. + +### Louis Charette + +Louis's a civil engineer in Montréal, Québec who also has a passion for coding. He is one of the main contributors for SimpsonsCity.com and likes to share his knowledge by helping others the same way he was helped when he first started coding. + +### Jordan Mele + +Jordan's a developer at Mayvin Training and a student studying Computer Science at the University of Wollongong. His passion is creating software-based solutions to overcomplicated problems, without taking control away from the user. He's also Australian. + +### Sarah Baghdadi + +Sarah is UserFrosting's UX specialist and frontend designer. In addition to her work on the UF application itself, she is responsible for the amazing design of https://www.userfrosting.com and https://learn.userfrosting.com. + +### Srinivas Nukala + +Srinivas's a web applications architect, with a passion for open source technologies. He is experienced in building SaaS (software as a service) web applications and enjoys working on open source projects and contributing to the community. He has a Masters in Computer Science from Pune University, India. + +## Contributing + +This project exists thanks to all the people who contribute. If you're interested in contributing to the UserFrosting codebase, please see our [contributing guidelines](.github/CONTRIBUTING.md) as well as our [style guidelines](STYLE-GUIDE.md). + + + +### Thanks to our translators! + +- Louis Charette (@lcharette) - French +- Karuhut Komol (@popiazaza) - Thai +- Pietro Marangon (@Pe46dro) - Italian +- Abdullah Seba (@abdullahseba) - Arabic +- Bruno Silva (@brunomnsilva) - Portuguese +- @BruceGui - Chinese +- @kevinrombach - German +- @rafa31gz - Spanish +- @splitt3r - German +- @X-Anonymous-Y - German +- Dmitriy (@rendername) - Russian +- Amin Akbari (@aminakbari) - Farsi +- Dumblledore - Turkish + +## Supporting UserFrosting + +### Backers + +Backers help us continue to develop UserFrosting by pledging a regular monthly contribution of $5 or more. [[Become a backer](https://opencollective.com/userfrosting#contribute)] + + + +#### Sponsors + +Support this project by becoming a sponsor. Sponsors have contributed a total of $500 or more to UserFrosting (either as an ongoing backer or one-time contributions). Your logo will show up here with a link to your website. [[Become a sponsor](https://opencollective.com/userfrosting#sponsor)] + +[![USOR Games](sponsors/usor.png)](https://usorgames.com) +[![Next Generation Internet](sponsors/nextgi.png)](https://nextgi.com) diff --git a/main/STYLE-GUIDE.md b/main/STYLE-GUIDE.md new file mode 100755 index 0000000..94c10b2 --- /dev/null +++ b/main/STYLE-GUIDE.md @@ -0,0 +1,44 @@ +# Style guide for contributing to UserFrosting + +## PHP + +All PHP contributions must adhere to [PSR-1](http://www.php-fig.org/psr/psr-1/) and [PSR-2](http://www.php-fig.org/psr/psr-2/) specifications. + +In addition: + +### Documentation + +- All documentation blocks must adhere to the [PHPDoc](https://phpdoc.org/) format and syntax. +- All PHP files MUST contain the following documentation block immediately after the opening ` + +RewriteEngine On + +## Begin - Security +# Block all direct access to files and folders beginning with a dot +RewriteRule (^\.|/\.) - [F] +# Block access to specific files in the root folder +RewriteRule ^(LICENSE.txt|composer.lock|composer.json|\.htaccess|\.env)$ error [F] +## End - Security + + diff --git a/main/app/cache/.gitkeep b/main/app/cache/.gitkeep new file mode 100755 index 0000000..e69de29 diff --git a/main/app/defines.php b/main/app/defines.php new file mode 100755 index 0000000..d129424 --- /dev/null +++ b/main/app/defines.php @@ -0,0 +1,52 @@ + This version only works with UserFrosting 4.1.x ! + +# Help and Contributing + +If you need help using this sprinkle or found any bug, feels free to open an issue or submit a pull request. You can also find me on the [UserFrosting Chat](https://chat.userfrosting.com/) most of the time for direct support. + +Buy Me a Coffee at ko-fi.com + +# Installation + +Edit UserFrosting `app/sprinkles.json` file and add the following to the `require` list : `"lcharette/uf_configmanager": "^2.0.0"`. Also add `FormGenerator` and `ConfigManager` to the `base` list. For example: + +``` +{ + "require": { + "lcharette/uf_configmanager": "^2.0.0" + }, + "base": [ + "core", + "account", + "admin", + "FormGenerator", + "ConfigManager" + ] +} +``` + +Run `composer update` then `php bakery bake` to install the sprinkle. + +## Permissions +The migration will automatically add the `update_site_config` permission to the `Site Administrator` role. To give access to the config UI to another user, simply add the `update_site_config` permission slug to that user role. + +## Add link to the menu +The configuration UI is bound to the the `/settings` route. Simply add a link to this route where you want it. The checkAccess make it so it will appear only for users having the appropriate permission. For example, you can add the following to the sidebar menu : + +``` +{% if checkAccess('update_site_config') %} +
  • + {{ translate("SITE.CONFIG.MANAGER") }} +
  • +{% endif %} +``` + +## Adding custom config + +! TODO + +> *NOTE* Only `.json` are accepted. `Yaml` schemas are cannot be used for now. + +# Licence + +By [Louis Charette](https://github.com/lcharette). Copyright (c) 2017, free to use in personal and commercial software as per the MIT license. \ No newline at end of file diff --git a/main/app/sprinkles/ConfigManager/asset-bundles.json b/main/app/sprinkles/ConfigManager/asset-bundles.json new file mode 100755 index 0000000..7550df9 --- /dev/null +++ b/main/app/sprinkles/ConfigManager/asset-bundles.json @@ -0,0 +1,16 @@ +{ + "bundle": { + "js/ConfigManager" : { + "scripts": [ + "js/ConfigManager.js" + ], + "options": { + "result": { + "type": { + "scripts": "plain" + } + } + } + } + } +} \ No newline at end of file diff --git a/main/app/sprinkles/ConfigManager/assets/js/ConfigManager.js b/main/app/sprinkles/ConfigManager/assets/js/ConfigManager.js new file mode 100755 index 0000000..2a298f5 --- /dev/null +++ b/main/app/sprinkles/ConfigManager/assets/js/ConfigManager.js @@ -0,0 +1,92 @@ +/*! + * UF Config Manager - Config Manager Widget + * + * @link https://github.com/lcharette/UF_ConfigManager + * @copyright Copyright (c) 2016 Louis Charette + * @license https://github.com/lcharette/UF_ConfigManager/blob/master/LICENSE (MIT License) + */ + +(function( $ ){ + + 'use strict'; + + var options = {}; + + var methods = { + init : function(optionsArg) { + + // Setup options + options = $.extend( options, $.fn.ConfigManager.defaultOptions, optionsArg ); + + // To use this inside sub-functions + var elements = this; + + // Get the currently selected panel from the url anchor and switch to it + var hash = window.location.hash.substr(1); + if (hash != undefined && hash !== "") { + $(elements).hide(); + $("#"+hash).show(); + + // Change the menu + $(options.menu).find("li").removeClass('active'); + $(options.menu).find('a[href="#'+hash+'"]').parent().addClass("active"); + } + + // Set the menu + $(options.menu).find("li > a").click(function () { + + // Change the menu first + $(options.menu).find("li").removeClass('active'); + $(this).parent().addClass("active"); + + // Change the displayed forms next + $(elements).hide(); + $("#"+$(this).data('target')).show(); + }); + + // For each element the plugin is called on + this.each(function() { + + // To use this inside sub-functions + var formPanel = this; + + // ufForm instance. Don't need FormGeneator now + $(formPanel).find("form").ufForm({ + validators: options.validators[ $(formPanel).attr('id') ], + msgTarget: $(formPanel).find("form .form-alerts") + }).on("submitSuccess.ufForm", function() { + // Forward to settings page on success + window.location.reload(true); + }).on("submitError.ufForm", function() { + $(formPanel).find("form .form-alerts").show(); + }); + + }); + return; + } + }; + + /* + * Main plugin function + */ + $.fn.ConfigManager = function(methodOrOptions) { + if ( methods[methodOrOptions] ) { + return methods[ methodOrOptions ].apply( this, Array.prototype.slice.call( arguments, 1 )); + } else if ( typeof methodOrOptions === 'object' || ! methodOrOptions ) { + // Default to "init" + return methods.init.apply( this, arguments ); + } else { + $.error( 'Method ' + methodOrOptions + ' does not exist on jQuery.ConfigManager' ); + } + }; + + /* + * Default plugin options + */ + $.fn.ConfigManager.defaultOptions = { + menu : $(".configMenu"), + validators: {} + }; + + +})( jQuery ); \ No newline at end of file diff --git a/main/app/sprinkles/ConfigManager/composer.json b/main/app/sprinkles/ConfigManager/composer.json new file mode 100755 index 0000000..d524836 --- /dev/null +++ b/main/app/sprinkles/ConfigManager/composer.json @@ -0,0 +1,26 @@ +{ + "name": "lcharette/uf_configmanager", + "type": "userfrosting-sprinkle", + "description": "This Sprinkle provides a UI for core and custom site settings", + "keywords": ["UserFrosting", "Config", "Settings", "Admin Panel"], + "homepage": "https://github.com/lcharette/UF_ConfigManager", + "license" : "MIT", + "authors" : [ + { + "name": "Louis Charette", + "homepage": "https://github.com/lcharette" + } + ], + "require": { + "php": ">=5.6", + "lcharette/uf_formgenerator": "^2.2.10" + }, + "autoload": { + "psr-4": { + "UserFrosting\\Sprinkle\\ConfigManager\\": "src/" + } + }, + "extra": { + "installer-name": "ConfigManager" + } +} diff --git a/main/app/sprinkles/ConfigManager/locale/en_US/AdminLTE.php b/main/app/sprinkles/ConfigManager/locale/en_US/AdminLTE.php new file mode 100755 index 0000000..bd1d032 --- /dev/null +++ b/main/app/sprinkles/ConfigManager/locale/en_US/AdminLTE.php @@ -0,0 +1,27 @@ + [ + "SETTINGS" => [ + "@TRANSLATION" => "AdminLTE Settings", + + "DESC" => "Settings for the AdminLTE Theme", + + "SKIN" => "Skin color" + ], + + "SKIN" => [ + "BLUE" => "Blue", + "BLACK" => "Black", + "PURPLE" => "Purple", + "GREEN" => "Green", + "RED" => "Red", + "YELLOW" => "Yellow", + "BLUE_LIGHT" => "Blue Light", + "BLACK_LIGHT" => "Black Light", + "PURPLE_LIGHT" => "Purple Light", + "GREEN_LIGHT" => "Green Light", + "RED_LIGHT" => "Red Light", + "YELLOW_LIGHT" => "Yellow Light" + ] + ] +]; \ No newline at end of file diff --git a/main/app/sprinkles/ConfigManager/locale/en_US/ConfigManager.php b/main/app/sprinkles/ConfigManager/locale/en_US/ConfigManager.php new file mode 100755 index 0000000..b82f099 --- /dev/null +++ b/main/app/sprinkles/ConfigManager/locale/en_US/ConfigManager.php @@ -0,0 +1,27 @@ + [ + "CONFIG" => [ + "@TRANSLATION" => "UserFrosting Settings", + + "DESC" => "Core settings of UserFrosting. See the config file for more configuration options", + + "MANAGER" => "Configuration manager", + + "PAGEDESC" => "This pages allows to edit the global site configuration variables stored in the database", + + "SAVED" => "Changes saved successfully !" + ], + "TITLE" => [ + "@TRANSLATION" => "Site title", + "REQUIRED" => "The site title is required" + ], + "REGISTRATION" => [ + "ENABLED" => "Enabled site registration", + "REQUIRE_EMAIL_VERIFICATION" => "Require email verification when registering" + ] + ], + "SETTINGS" => [ + "DISPLAY_ERROR_DETAILS" => "Display error details" + ] +]; \ No newline at end of file diff --git a/main/app/sprinkles/ConfigManager/locale/fr_FR/AdminLTE.php b/main/app/sprinkles/ConfigManager/locale/fr_FR/AdminLTE.php new file mode 100755 index 0000000..39b4614 --- /dev/null +++ b/main/app/sprinkles/ConfigManager/locale/fr_FR/AdminLTE.php @@ -0,0 +1,27 @@ + [ + "SETTINGS" => [ + "@TRANSLATION" => "Paramètres d'AdminLTE", + + "DESC" => "Paramètres du thème AdminLTE", + + "SKIN" => "Couleur du thème" + ], + + "SKIN" => [ + "BLUE" => "Bleu", + "BLACK" => "Noir", + "PURPLE" => "Violet", + "GREEN" => "Vert", + "RED" => "Rouge", + "YELLOW" => "Jaune", + "BLUE_LIGHT" => "Bleu (Light)", + "BLACK_LIGHT" => "Noir (Light)", + "PURPLE_LIGHT" => "Violet (Light)", + "GREEN_LIGHT" => "Vert (Light)", + "RED_LIGHT" => "Rouge (Light)", + "YELLOW_LIGHT" => "Jaune (Light)" + ] + ] +]; \ No newline at end of file diff --git a/main/app/sprinkles/ConfigManager/locale/fr_FR/ConfigManager.php b/main/app/sprinkles/ConfigManager/locale/fr_FR/ConfigManager.php new file mode 100755 index 0000000..3f9acff --- /dev/null +++ b/main/app/sprinkles/ConfigManager/locale/fr_FR/ConfigManager.php @@ -0,0 +1,27 @@ + [ + "CONFIG" => [ + "@TRANSLATION" => "Paramètres de UserFrosting", + + "DESC" => "Paramètres principaux de UserFrosting. Voir le fichier config pour plus d'options", + + "MANAGER" => "Gestionnaire des paramètres", + + "PAGEDESC" => "Cette page permet de modifier les paramètres globaux du site enregistrés dans la base de données", + + "SAVED" => "Changements sauvegardés avec succès !" + ], + "TITLE" => [ + "@TRANSLATION" => "Titre du site", + "REQUIRED" => "Le titre du site est requis" + ], + "REGISTRATION" => [ + "ENABLED" => "Activer l'inscription", + "REQUIRE_EMAIL_VERIFICATION" => "Exiger une vérification par e-mail lors de l'inscription" + ] + ], + "SETTINGS" => [ + "DISPLAY_ERROR_DETAILS" => "Afficher le détails des erreurs" + ] +]; \ No newline at end of file diff --git a/main/app/sprinkles/ConfigManager/routes/ConfigManager.php b/main/app/sprinkles/ConfigManager/routes/ConfigManager.php new file mode 100755 index 0000000..f17f589 --- /dev/null +++ b/main/app/sprinkles/ConfigManager/routes/ConfigManager.php @@ -0,0 +1,16 @@ +group('/settings', function () { + $this->get('', 'UserFrosting\Sprinkle\ConfigManager\Controller\ConfigManagerController:displayMain') + ->setName('ConfigManager'); + + $this->post('/{schema}', 'UserFrosting\Sprinkle\ConfigManager\Controller\ConfigManagerController:update') + ->setName('ConfigManager.save'); +}); \ No newline at end of file diff --git a/main/app/sprinkles/ConfigManager/schema/config/AdminLTE.json b/main/app/sprinkles/ConfigManager/schema/config/AdminLTE.json new file mode 100755 index 0000000..04e92da --- /dev/null +++ b/main/app/sprinkles/ConfigManager/schema/config/AdminLTE.json @@ -0,0 +1,32 @@ +{ + "name" : "ADMINLTE.SETTINGS", + "desc" : "ADMINLTE.SETTINGS.DESC", + + "config": { + "site.AdminLTE.skin" : { + "validators" : { + "required" : {} + }, + "cached" : true, + "form" : { + "type" : "select", + "label" : "ADMINLTE.SETTINGS.SKIN", + "icon" : "", + "options" : { + "blue" : "ADMINLTE.SKIN.BLUE", + "black" : "ADMINLTE.SKIN.BLACK", + "purple" : "ADMINLTE.SKIN.PURPLE", + "green" : "ADMINLTE.SKIN.GREEN", + "red" : "ADMINLTE.SKIN.RED", + "yellow" : "ADMINLTE.SKIN.YELLOW", + "blue-light" : "ADMINLTE.SKIN.BLUE_LIGHT", + "black-light" : "ADMINLTE.SKIN.BLACK_LIGHT", + "purple-light" : "ADMINLTE.SKIN.PURPLE_LIGHT", + "green-light" : "ADMINLTE.SKIN.GREEN_LIGHT", + "red-light" : "ADMINLTE.SKIN.RED_LIGHT", + "yellow-light" : "ADMINLTE.SKIN.YELLOW_LIGHT" + } + } + } + } +} \ No newline at end of file diff --git a/main/app/sprinkles/ConfigManager/schema/config/site.json b/main/app/sprinkles/ConfigManager/schema/config/site.json new file mode 100755 index 0000000..2c9f4f0 --- /dev/null +++ b/main/app/sprinkles/ConfigManager/schema/config/site.json @@ -0,0 +1,40 @@ +{ + "name" : "SITE.CONFIG", + "desc" : "SITE.CONFIG.DESC", + + "config": { + "site.title" : { + "validators" : { + "required" : { + "message" : "SITE.TITLE.REQUIRED" + } + }, + "form" : { + "type" : "text", + "label" : "SITE.TITLE", + "icon" : "fa-comment" + } + }, + "site.registration.enabled" : { + "validators" : {}, + "form" : { + "type" : "checkbox", + "label" : "SITE.REGISTRATION.ENABLED" + } + }, + "site.registration.require_email_verification" : { + "validators" : {}, + "form" : { + "type" : "checkbox", + "label" : "SITE.REGISTRATION.REQUIRE_EMAIL_VERIFICATION" + } + }, + "settings.displayErrorDetails" : { + "validators" : {}, + "form" : { + "type" : "checkbox", + "label" : "SETTINGS.DISPLAY_ERROR_DETAILS" + } + } + } +} \ No newline at end of file diff --git a/main/app/sprinkles/ConfigManager/src/ConfigManager.php b/main/app/sprinkles/ConfigManager/src/ConfigManager.php new file mode 100755 index 0000000..c29cee9 --- /dev/null +++ b/main/app/sprinkles/ConfigManager/src/ConfigManager.php @@ -0,0 +1,40 @@ + ['onAddGlobalMiddleware', 0] + ]; + } + + /** + * Add middleware. + */ + public function onAddGlobalMiddleware(Event $event) + { + $app = $event->getApp(); + $app->add($this->ci->configManager); + } +} diff --git a/main/app/sprinkles/ConfigManager/src/Controller/ConfigManagerController.php b/main/app/sprinkles/ConfigManager/src/Controller/ConfigManagerController.php new file mode 100755 index 0000000..0b5aac0 --- /dev/null +++ b/main/app/sprinkles/ConfigManager/src/Controller/ConfigManagerController.php @@ -0,0 +1,173 @@ +ci = $ci; + $this->manager = new ConfigManager($ci); + } + + /** + * mainList function. + * Used to display a list of all schema with their form + * + * @access public + * @param mixed $request + * @param mixed $response + * @param mixed $args + * @return void + */ + public function displayMain($request, $response, $args){ + + // Access-controlled resource + if (!$this->ci->authorizer->checkAccess($this->ci->currentUser, 'update_site_config')) { + throw new ForbiddenException(); + } + + // Get all the config schemas + $schemas = $this->manager->getAllShemas(); + + // Parse each of them to get it's content + foreach ($schemas as $i => $schemaData) { + + // Set the schemam, the validator and the form + $schema = new RequestSchemaRepository($schemaData['config']); + $validator = new JqueryValidationAdapter($schema, $this->ci->translator); + + // Create the form + $config = $this->ci->config; + $form = new Form($schema, $config); + + // The field names dot syntaxt won't make it across the HTTP POST request. + // Wrap them in a nice `data` array + $form->setFormNamespace("data"); + + // Twig doesn't need the raw thing + unset($schemas[$i]['config']); + + // Add the field and validator so Twig can play with them + $schemas[$i]["fields"] = $form->generate(); + $schemas[$i]["validators"] = $validator->rules('json', true); + + // Add the save url for that schema + $schemas[$i]["formAction"] = $this->ci->router->pathFor('ConfigManager.save', ['schema' => $schemaData['filename']]); + } + + // Time to render the page ! + $this->ci->view->render($response, 'pages/ConfigManager.html.twig', [ + "schemas" => $schemas, + ]); + + } + + /** + * update function. + * Processes the request to save the settings to the db + * + * @access public + * @param mixed $request + * @param mixed $response + * @param mixed $args + * @return void + */ + public function update($request, $response, $args){ + + // Get the alert message stream + $ms = $this->ci->alerts; + + // Access-controlled resource + if (!$this->ci->authorizer->checkAccess($this->ci->currentUser, 'update_site_config')) { + throw new ForbiddenException(); + } + + // Request POST data + $post = $request->getParsedBody(); + + // So we first get the shcema data + $loader = new YamlFileLoader("schema://config/".$args['schema'].".json"); + $schemaData = $loader->load(); + + // We can't pass the file directly to RequestSchema because it's a custom one + // So we create a new empty RequestSchemaRepository and feed it the `config` part of our custom schema + $schema = new RequestSchemaRepository($schemaData['config']); + + // Transform the data + $transformer = new RequestDataTransformer($schema); + $data = $transformer->transform($post['data']); + + // We change the dot notation of our elements to a multidimensionnal array + // This is required for the fields (but not for the schema) because the validator doesn't use the + // dot notation the same way. Sending dot notation field name to the validator will fail. + $dataArray = array(); + foreach ($data as $key => $value) { + array_set($dataArray, $key, $value); + } + + // We validate the data array against the schema + $validator = new ServerSideValidator($schema, $this->ci->translator); + if (!$validator->validate($dataArray)) { + $ms->addValidationErrors($validator); + return $response->withStatus(400); + } + + // The data is now validaded. Instead or switching back the array to dot notation, + // we can use the `$data` that's still intact. The validator doesn't change the data + // Next, update each config + foreach ($data as $key => $value) { + + // We need to access the $schemaData to find if we need to cache this one + $cached = (isset($schemaData['config'][$key]['cached'])) ? $schemaData['config'][$key]['cached'] : true; + + // Set the config using the manager + $this->manager->set($key, $value, $cached); + } + + //Success message! + $ms->addMessageTranslated("success", "SITE.CONFIG.SAVED"); + return $response->withJson([], 200, JSON_PRETTY_PRINT); + } +} diff --git a/main/app/sprinkles/ConfigManager/src/Database/Migrations/v100/SettingsTable.php b/main/app/sprinkles/ConfigManager/src/Database/Migrations/v100/SettingsTable.php new file mode 100755 index 0000000..182dbfb --- /dev/null +++ b/main/app/sprinkles/ConfigManager/src/Database/Migrations/v100/SettingsTable.php @@ -0,0 +1,48 @@ +schema->hasTable('settings')) { + $this->schema->create('settings', function (Blueprint $table) { + $table->increments('id'); + $table->string('key'); + $table->string('value')->nullable(); + $table->boolean('cached')->default(1); + $table->timestamps(); + + $table->engine = 'InnoDB'; + $table->collation = 'utf8_unicode_ci'; + $table->charset = 'utf8'; + }); + } + } + + /** + * {@inheritDoc} + */ + public function down() + { + $this->schema->drop('settings'); + } +} \ No newline at end of file diff --git a/main/app/sprinkles/ConfigManager/src/Database/Migrations/v101/SettingsPermissions.php b/main/app/sprinkles/ConfigManager/src/Database/Migrations/v101/SettingsPermissions.php new file mode 100755 index 0000000..c3928d9 --- /dev/null +++ b/main/app/sprinkles/ConfigManager/src/Database/Migrations/v101/SettingsPermissions.php @@ -0,0 +1,70 @@ +first(); + if ($permissionExist) { + $this->io->warning("\nPermission slug `update_site_config` already exist. Skipping..."); + return; + } + + // Add default permissions + $permission = new Permission([ + 'slug' => 'update_site_config', + 'name' => 'Update site configuration', + 'conditions' => 'always()', + 'description' => 'Edit site configuration from the UI' + ]); + $permission->save(); + + $roleSiteAdmin = Role::where('slug', 'site-admin')->first(); + if ($roleSiteAdmin) { + $roleSiteAdmin->permissions()->attach([ + $permission->id + ]); + } + } + + /** + * {@inheritDoc} + */ + public function down() + { + $permissions = Permission::where('slug', 'update_site_config')->get(); + foreach ($permissions as $permission) { + $permission->delete(); + } + } +} \ No newline at end of file diff --git a/main/app/sprinkles/ConfigManager/src/Database/Models/Config.php b/main/app/sprinkles/ConfigManager/src/Database/Models/Config.php new file mode 100755 index 0000000..53a85ca --- /dev/null +++ b/main/app/sprinkles/ConfigManager/src/Database/Models/Config.php @@ -0,0 +1,70 @@ +db; + return new ConfigManager($c); + }; + } +} diff --git a/main/app/sprinkles/ConfigManager/src/Util/ConfigManager.php b/main/app/sprinkles/ConfigManager/src/Util/ConfigManager.php new file mode 100755 index 0000000..e935487 --- /dev/null +++ b/main/app/sprinkles/ConfigManager/src/Util/ConfigManager.php @@ -0,0 +1,243 @@ +ci = $ci; + } + + /** + * __invoke function. + * Invoke the ConfigManager middleware, merging the db config with the file based one + * + * @access public + * @param \Psr\Http\Message\ServerRequestInterface $request PSR7 request + * @param \Psr\Http\Message\ResponseInterface $response PSR7 response + * @param callable $next Next middleware + * @return \Psr\Http\Message\ResponseInterface + */ + public function __invoke($request, $response, $next) + { + $this->ci->config->mergeItems(null, $this->fetch()); + return $next($request, $response); + } + + /** + * fetch function. + * Fetch all the config from the db and uses the + * cache container to store most of thoses setting in the cache system + * + * @access public + * @return void + */ + public function fetch() { + + $cache = $this->ci->cache; + + // Case n° 1 we don't have cached content. We load everything + // Case n° 2 we have cached content, pull that and load the non chanched things to it + if (($cached_settings = $cache->get('UF_config')) === null) + { + $settingsCollection = Config::all(); + $settings = $this->collectionToArray($settingsCollection); + + // Save in cache. The settings that are not cached are not included + $cache->forever('UF_config', $this->collectionToArray($settingsCollection, false)); + } + else + { + // We have the cached values, we need to grab the non cached ones + $settingsCollection = Config::where('cached', 0); + $settings = array_merge_recursive($cached_settings, $this->collectionToArray($settingsCollection)); + } + + return $settings; + } + + /** + * delete function. + * Removes a configuration option + * + * @access public + * @param string $key The setting's name + * @return bool Success + */ + public function delete($key) { + + // Get the desired key + if (!$setting = Config::where('key', $key)->first()) { + return false; + } + + // Delete time + $setting->delete(); + + // Remove from current laod + unset($this->ci->config[$key]); + + // Delete cache + if ($setting->cached) + { + $this->ci->cache->forget('UF_config'); + } + + return true; + } + + /** + * set function. + * Sets a setting's value + * + * @access public + * @param string $key The setting's name + * @param string $value The new value + * @param bool $cached (default: true) Whether this variable should be cached or if it + * changes too frequently to be efficiently cached. + * @return bool True if the value was changed, false otherwise + */ + public function set($key, $value, $cached = true) { + return $this->set_atomic($key, false, $value, $cached); + } + + /** + * set_atomic function. + * Sets a setting's value only if the old_value matches the + * current value or the setting does not exist yet. + * + * @access public + * @param string $key The setting's name + * @param string $old_value Current configuration value or false to ignore + * the old value + * @param string $new_value The new value + * @param bool $cached (default: true) Whether this variable should be cached or if it + * changes too frequently to be efficiently cached. + * @return bool True if the value was changed, false otherwise + */ + public function set_atomic($key, $old_value, $new_value, $cached = true) { + + // Get the desired key + $setting = Config::where('key', $key)->first(); + + if ($setting) { + + if ($old_value === false || $setting->value == $old_value) { + + $setting->value = $new_value; + $setting->save(); + + } else { + return false; + } + + } else { + $setting = new Config([ + 'key' => $key, + 'value' => $new_value, + 'cached' => $cached + ]); + $setting->save(); + } + + if ($cached) + { + $this->ci->cache->forget('UF_config'); + } + + $this->ci->config[$key] = $new_value; + return true; + } + + /** + * getAllShemas function. + * Get all the config schemas available + * + * @access public + * @return void + */ + public function getAllShemas() { + + $configSchemas = []; + + $loader = new YamlFileLoader([]); + + // Get all the location where we can find config schemas + $paths = array_reverse($this->ci->locator->findResources('schema://config', true, false)); + + // For every location... + foreach ($paths as $path) { + + // Get a list of all the schemas file + $files_with_path = glob($path . "/*.json"); + + // Load every found files + foreach ($files_with_path as $file) { + + // Load the file content + $schema = $loader->loadFile($file); + + // Get file name + $filename = basename($file, ".json"); + + //inject file name + $schema['filename'] = $filename; + + // Add to list + $configSchemas[$filename] = $schema; + } + } + + return $configSchemas; + } + + + /** + * collectionToArray function. + * This function Expand the db dot notation single level array + * to a multi-dimensional array + * + * @access private + * @param Collection $Collection Eloquent collection + * @return array + */ + private function collectionToArray($Collection, $include_noncached = true) { + $settings_array = array(); + foreach ($Collection as $setting) { + if ($include_noncached || $setting->cached) { + array_set($settings_array, $setting->key, $setting->value); + } + } + return $settings_array; + } +} \ No newline at end of file diff --git a/main/app/sprinkles/ConfigManager/templates/pages/ConfigManager.html.twig b/main/app/sprinkles/ConfigManager/templates/pages/ConfigManager.html.twig new file mode 100755 index 0000000..ebfb251 --- /dev/null +++ b/main/app/sprinkles/ConfigManager/templates/pages/ConfigManager.html.twig @@ -0,0 +1,64 @@ +{% extends "pages/abstract/dashboard.html.twig" %} + +{% block page_title %}{{ translate('SITE.CONFIG.MANAGER') }}{% endblock %} +{% block page_description %}{{ translate('SITE.CONFIG.PAGEDESC') }}{% endblock %} + +{% block body_matter %} +
    +
    +
    +
    +

    Menu

    +
    +
    + +
    +
    +
    +
    + {% for schema in schemas %} + + {% endfor %} +
    +
    +{% endblock %} +{% block scripts_page %} + + + + + {{ assets.js('js/ConfigManager') | raw }} + {{ assets.js('js/form-widgets') | raw }} + +{% endblock %} diff --git a/main/app/sprinkles/FormGenerator/.gitignore b/main/app/sprinkles/FormGenerator/.gitignore new file mode 100755 index 0000000..3afbe61 --- /dev/null +++ b/main/app/sprinkles/FormGenerator/.gitignore @@ -0,0 +1,3 @@ + +.DS_Store +assets/vendor/ \ No newline at end of file diff --git a/main/app/sprinkles/FormGenerator/CHANGELOG.md b/main/app/sprinkles/FormGenerator/CHANGELOG.md new file mode 100755 index 0000000..0975d22 --- /dev/null +++ b/main/app/sprinkles/FormGenerator/CHANGELOG.md @@ -0,0 +1,78 @@ +# Change Log + +## 2.2.10 +- Added support for Repository + +## 2.2.9 +- Fix issue when setting data that is a collection +- `formSuccess` and `confirmSuccess` events now include the request data as a second argument + +## 2.2.8 +- Fix icon in textarea macro + +## 2.2.7 +- Added `modal-large` template file. + +## 2.2.6 +- Fix issue with `binary` checkbox tests. +- Fix Text input style when no icon is added + +## 2.2.5 +- Added `binary` option for checkbox to disable UF binary checkbox system (bool; default true). + +## 2.2.4 +- Add necessary HTML to disable submit and cancel button in modal form. + +## 2.2.3 +- New `$form->setOptions` function to set options of a select element. Shortcut for using `setInputArgument` and `setValue`. + +## 2.2.2 +- Fix issue with error alert no displaying on confirmation dialog + +## 2.2.1 +- Initialize ufAlert if not already done +- Autofocus first form field when modal is displayed + +## 2.2.0 +- Refactored the javascript plugin +- Added new events +- Added new `redirectAfterSuccess` boolean option + +## 2.1.2 +- Fix warning with select macro + +## 2.1.1 +- Fix issue with the select macro +- Renamed macro templates with the `*.html.twig` extension + +## 2.1.0 +- Completely refactored how form fields are parsed, including how default value are defined. Each input type now defines it's own class for defining default values and transforming some input. +- Twig templates updated to reflect the new parser. +- Twig macros changed from `*.generate(name, value)` to `*.generate(input)`. +- **`Bool` type changed to `checkbox`**. +- Removed the `number` Twig template (Will use the text input one). +- Added unit tests. +- Support for any attributes in the schema. For example, if you need to add a data attribute to a field, your schema would be: +``` +"myField" : { + "form" : { + "type" : "text", + "label" : "My Field", + "icon" : "fa-pencil", + "data-something" : "blah" + } +} +``` + +## 2.0.0 +- Updated for UserFrosting v4.1.x + +The custom `RequestSchema` have been removed. Instead of building the form directly on the schema using `$schema->initForm()`, you now create a new Form using `$form = new Form($schema)` and go on from there. Handling complex schema can now be done using the new loader system from UF 4.1. + +`$schema->generateForm();` has also been changed to `$form->generate();`. + +## 1.0.1 +- Bug fixes + +## 1.0.0 +- Initial release diff --git a/main/app/sprinkles/FormGenerator/LICENSE b/main/app/sprinkles/FormGenerator/LICENSE new file mode 100755 index 0000000..09386f7 --- /dev/null +++ b/main/app/sprinkles/FormGenerator/LICENSE @@ -0,0 +1,21 @@ +MIT License + +Copyright (c) 2016 Louis Charette + +Permission is hereby granted, free of charge, to any person obtaining a copy +of this software and associated documentation files (the "Software"), to deal +in the Software without restriction, including without limitation the rights +to use, copy, modify, merge, publish, distribute, sublicense, and/or sell +copies of the Software, and to permit persons to whom the Software is +furnished to do so, subject to the following conditions: + +The above copyright notice and this permission notice shall be included in all +copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE +AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, +OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE +SOFTWARE. diff --git a/main/app/sprinkles/FormGenerator/README.md b/main/app/sprinkles/FormGenerator/README.md new file mode 100755 index 0000000..2924ede --- /dev/null +++ b/main/app/sprinkles/FormGenerator/README.md @@ -0,0 +1,352 @@ +# Form Generator Sprinkle for [UserFrosting 4](https://www.userfrosting.com) +This Sprinkle provides helper classes, Twig template and JavaScript plugins to generate HTML forms, modals and confirm modal bases on UserFrosting/[validation schemas](https://learn.userfrosting.com/routes-and-controllers/client-input/validation). + +> This version only works with UserFrosting 4.1.x ! + +# Help and Contributing + +If you need help using this sprinkle or found any bug, feels free to open an issue or submit a pull request. You can also find me on the [UserFrosting Chat](https://chat.userfrosting.com/) most of the time for direct support. + +Buy Me a Coffee at ko-fi.com + +# Installation +Edit UserFrosting `app/sprinkles.json` file and add the following to the `require` list : `"lcharette/uf_formgenerator": "^2.0.0"`. Also add `FormGenerator` to the `base` list. For example: + +``` +{ + "require": { + "lcharette/uf_formgenerator": "^2.0.0" + }, + "base": [ + "core", + "account", + "admin", + "FormGenerator" + ] +} +``` + +Run `composer update` then `php bakery bake` to install the sprinkle. + +# Features and usage +Before starting with _FormGenerator_, you should read the main UserFrosting guide to familiarize yourself with _validation schemas_: (https://learn.userfrosting.com/routes-and-controllers/client-input/validation). + +## Form generation +### Defining the fields in the schema +This sprinkle uses the `schemas` used by UserFrosting to validate form data to build form. To achieve this, a new `form` key is simply added to the fields found in a `schema` file. + +For example, here's a simple `schema` used to validate a form used to create a `project`. The form will contain a `name`, `description` and `status` fields. + +``` +{ + "name" : { + "validators" : { + "length" : { + "min" : 1, + "max" : 100 + }, + "required" : { + "message" : "PROJECT.VALIDATE.REQUIRED_NAME" + } + } + }, + "description" : { + "validators" : {} + }, + "status" : { + "validators" : { + "member_of" : { + "values" : [ + "0", "1" + ] + }, + "required" : { + "message" : "PROJECT.VALIDATE.STATUS" + } + } + } +} +``` +> Note: FormGenerator works with json and YAML schemas. + +At this point, with typical UserFrosting setup, you would be going into your controller and Twig files to manually create your HTML form. This can be easy if you have a two or three fields, but can be a pain with a dozen fields and more. This is where FormGenerator steps in with the use of a new `form` attribute. Let's add it to our `project` form : + +``` +{ + "name" : { + "validators" : { + "length" : { + "min" : 1, + "max" : 100 + }, + "required" : { + "message" : "VALIDATE.REQUIRED_NAME" + } + }, + "form" : { + "type" : "text", + "label" : "NAME", + "icon" : "fa-flag", + "placeholder" : "NAME" + } + }, + "description" : { + "validators" : {}, + "form" : { + "type" : "textarea", + "label" : "DESCRIPTION", + "icon" : "fa-pencil", + "placeholder" : "DESCRIPTION", + "rows" : 5 + } + }, + "status" : { + "validators" : { + "member_of" : { + "values" : [ + "0", "1" + ] + }, + "required" : { + "message" : "VALIDATE.STATUS" + } + }, + "form" : { + "type" : "select", + "label" : "STATUS", + "options" : { + "0" : "Active", + "1" : "Disabled" + } + } + } +} +``` + +Let's look closer at the `name` field : + +``` +"form" : { + "type" : "text", + "label" : "PROJECT.NAME", + "icon" : "fa-flag", + "placeholder" : "PROJECT.NAME" +} +``` + +Here you can see that we define the `type`, `label`, `icon` and `placeholder` value for this `name` field. You can define any standard [form attributes](http://www.w3schools.com/html/html_form_attributes.asp), plus the `icon`, `label` and `default` attributes. `data-*` attributes can also be defined in your schema if you need them. For the `select` element, a special `options` attribute containing an array of `key : value` can be used to define the dropdown options. The select options (as any other attributes) can also be set in PHP (see further below). + +And of course, the values of the `label` and `placeholder` attributes can be defined using _translation keys_. + +Currently, FormGenerator supports the following form elements : +- text (and any input supported by the HTML5 standard : number, tel, password, etc.) +- textarea +- select +- checkbox +- hidden +- alert (Display a static alert box in the form) + +### The controller part +Once your fields defined in the `schema` json or yaml file, you need to load that schema in your controller. + +First thing to do is add FormGenerator's `Form` class to your "use" list : +`use UserFrosting\Sprinkle\FormGenerator\Form;` + +Next, where you load the schema and setup the `validator`, you simply add the new Form creation: +``` +// Load validator rules +$schema = new RequestSchema("schema://project.json"); +$validator = new JqueryValidationAdapter($schema, $this->ci->translator); + +// Create the form +$form = new Form($schema, $project); +``` + +In this example, `$project` can contain the default (or current value) of the fields. A data collection fetched from the database with eloquent can also be passed directly. That second argument can also be omited to create an empty form. + +Last thing to do is send the fields to Twig. In the list of retuned variables to the template, add the `fields` variable: +``` +$this->ci->view->render($response, "pages/myPage.html.twig", [ + "fields" => $form->generate(), + "validators" => $validator->rules('json', true) +]); + +``` + +### The Twig template part + +Now it's time to display the form in `myPage.html.twig` ! + +``` +
    + {% include "forms/csrf.html.twig" %} +
    +
    +
    + {% include 'FormGenerator/FormGenerator.html.twig' %} +
    +
    +
    + +
    +
    +``` + +That's it! No need to list all the field manually. The ones defined in the `fields` variable will be displayed by `FormGenerator/FormGenerator.html.twig`. Note that this will only load the fields, not the form itself. The `
    ` tag and `submit` button needs to be added manually. + +## Modal form +What if you want to show a form in a modal window? Well, FormGenerator makes it even easier! It's basically three steps: +1. Setup your form schema (described above) +2. Setup the form in your controller +3. Call the modal from your template + +### Setup the form in your controller +With your schema in hand, it's time to create a controller and route to load your modal. The controller code will be like any basic UserFrosting modal, plus the `$form` part above and one changes in the `render` part. For example : + +``` +$this->ci->view->render($response, "FormGenerator/modal.html.twig", [ + "box_id" => $get['box_id'], + "box_title" => "PROJECT.CREATE", + "submit_button" => "CREATE", + "form_action" => '/project/create', + "fields" => $form->generate(), + "validators" => $validator->rules('json', true) +]); +``` + +As you can see, instead of rendering your own Twig template, you simply have to specify FormGenerator's modal template. This template requires the following variables: +1. `box_id`: This should always be `$get['box_id']`. This is used by the JavaScript code to actually display the modal. +2. `box_title`: The title of the modal. +3. `submit_button`: The label of the submit button. Optional. Default to `SUBMIT` (localized). +4. `form_action`: The route where the form will be sent +5. `fields`: The fields. Should always be `$form->generate()` +6. `validators`: Client side validators + +### Call the modal from your template +So at this point you have a controller that displays the modal at a `/path/to/controller` route. Time to show that modal. Again, two steps: + +First, define a link or a button that will call the modal when clicked. For example : +``` + +``` + +The important part here is the `data-formUrl` attribute. This is the route that will load your form. `js-displayForm` is used here to bind the button to the action. + +Second, load the FormGenerator JavaScript widget. Add this to your Twig file: +``` +{% block scripts_page %} + {{ assets.js('js/FormGenerator') | raw }} +{% endblock %} +``` + +By default, the `formGenerator` plugin will bind a **form modal** to every element with the `js-displayForm` class. + +## Modal confirmation + +One side features of FormGenerator is the ability to add a confirmation modal to your pages with simple HTML5 attributes. The process is similar to adding a modal form, without the need to create any controller or route. + +Let's look at a delete button / confirmation for our `project` : +``` + Delete +``` +(Note that content of data attributes can be translation keys) + +If not aready done, make sure the FormGenerator assets are included in your template. +``` +{% block scripts_page %} + {{ assets.js('js/FormGenerator') | raw }} +{% endblock %} +``` + +By default, the `formGenerator` plugin will bind a **confirmation modal** to every element with the `js-displayConfirm` class. + +## Advance usage + +### Defining attributes in PHP + +#### setInputArgument + +Form field input attributes can also be added or edited from PHP. This can be usefull when dynamically defining a Select input options. To do this, simply use the `setInputArgument($inputName, $property, $data)` method. For example, to add a list to a `clients` select : + +``` +// Get clients from the db model +$clients = Clients::all(); + +$form = new Form($schema); +$form->setInputArgument('clients', 'options', $clients); +``` + +#### setData + +If you want to set the form values once the form instance is created, you can use the `setData($data)` method: + +``` +$form = new Form($schema); +$form->setData($clients, $project); +``` + +#### setValue + +Similar to the `setData` method, you can set a specific input value using the `setValue($inputName, $value)` method : + +``` +$currentClient = ... + +$form = new Form($schema, $project); +$form->setValue('clients', $currentClient); +``` + +#### setFormNamespace + +When dealing with multiple form on the same page or a dynamic number of input (you can use the new `Loader` system in 4.1 to build dynamic schemas!), it can be useful to wrap form elements in an array using the `setFormNamespace($namespace)` method. This can also your the input names [to contains dot syntaxt](http://stackoverflow.com/a/20365198/445757). + +For example, `$form->setFormNamespace("data");` will transform all the input names from `` to ``. + +### Javascript Plugin + +By default, the `formGenerator` plugin will bind a **form modal** to every element with the `js-displayForm` class and will bind a **confirmation modal** to every element with the `js-displayConfirm` class. You can + +#### Options +The following options are available: + +Just pass an object with those + - `mainAlertElement` (jQuery element). The element on the main page where the main alerts will be displayed. Default to `$('#alerts-page')`. + - `redirectAfterSuccess` (bool). If set to true, the page will reload when the form submission or confirmation is succesful. Default to `true`. + +Example: +``` +$(".project-edit-button").formGenerator({redirectAfterSuccess: false}); +``` + +#### Events +You can listen for some events returned by FormGenerator. Those events can be used to apply some actions when the modal is displayed or the form is successfully sent. For example, this is can be used with `redirectAfterSuccess` on `false` to refresh the data on the page when the form is submitted successfully. + +- `formSuccess.formGenerator` +- `displayForm.formGenerator` +- `displayConfirmation.formGenerator` +- `confirmSuccess.formGenerator` +- `error.formGenerator` + +Example: +``` +$(".project-edit-button").on("formSuccess.formGenerator", function () { + // Refresh data +}); +``` + +# Working example + +See the [UF_FormGeneratorExample](https://github.com/lcharette/UF_FormGeneratorExample) repo for an example of the FormGenerator full code. + +# Running tests + +FormGenerator comes with some unit tests. Before submitting a new Pull Request, you need to make sure all tests are a go. With the sprinkle added to your UserFrosting installation, simply execute the `php bakery test` command to run the tests. + +# Licence + +By [Louis Charette](https://github.com/lcharette). Copyright (c) 2017, free to use in personal and commercial software as per the MIT license. diff --git a/main/app/sprinkles/FormGenerator/asset-bundles.json b/main/app/sprinkles/FormGenerator/asset-bundles.json new file mode 100755 index 0000000..b763d55 --- /dev/null +++ b/main/app/sprinkles/FormGenerator/asset-bundles.json @@ -0,0 +1,17 @@ +{ + "bundle": { + "js/FormGenerator": { + "scripts": [ + "vendor/bootstrap3-typeahead/bootstrap3-typeahead.js", + "js/widget-formGenerator.js" + ], + "options": { + "result": { + "type": { + "scripts": "plain" + } + } + } + } + } +} diff --git a/main/app/sprinkles/FormGenerator/assets/js/widget-formGenerator.js b/main/app/sprinkles/FormGenerator/assets/js/widget-formGenerator.js new file mode 100755 index 0000000..52743f6 --- /dev/null +++ b/main/app/sprinkles/FormGenerator/assets/js/widget-formGenerator.js @@ -0,0 +1,348 @@ +/*! + * FormGenerator Plugin + * + * JQuery plugin for the UserFrosting FormGenerator Sprinkle + * Based on UserFrosting v3 + * + * @package UF_FormGenerator + * @author Louis Charette + * @link https://github.com/lcharette/UF_FormGenerator + * @license MIT + */ + +;(function($, window, document, undefined) { + "use strict"; + + // Define plugin name and defaults. + var pluginName = "formGenerator", + defaults = { + DEBUG : false, + mainAlertElement : $('#alerts-page'), + redirectAfterSuccess : true, + autofocusModalElement : true + }; + + // Constructor + function Plugin (element, options) { + this.elements = element; + this.$elements = $(this.elements); + this.settings = $.extend(true, {}, defaults, options); + this._defaults = defaults; + this._name = pluginName; + + // Detect changes to element attributes + this.$elements.attrchange({ callback: function (event) { this.elements = event.target; }.bind(this) }); + + // Initialise ufAlerts + if (!this.settings.mainAlertElement.data('ufAlerts')) { + this.settings.mainAlertElement.ufAlerts(); + } + + return this; + } + + // Functions + $.extend(Plugin.prototype, { + /** + * Bind the display action for a form to the button + */ + display: function() { + this.$elements.on('click', $.proxy(this._fetchForm, this)); + return this.$elements; + }, + /** + * Bind the confirm action to the button + */ + confirm: function() { + this.$elements.on('click', $.proxy(this._fetchConfirmModal, this)); + return this.$elements; + }, + /** + * Fetch the form HTML + */ + _fetchForm: function(event) { + + // Get the button element + var button = event.currentTarget; + + // Get the box_id. Define one if none is defined + var box_id = $(button).data('target'); + if (box_id == undefined) { + box_id = "formGeneratorModal"; + } + + // Delete any existing instance of the form with the same name + if($('#' + box_id).length) { + $('#' + box_id).remove(); + } + + // Prepare the ajax payload + var payload = $.extend({ + box_id: box_id} + , button.dataset); + + // Fetch and render the form + $.ajax({ + type: "GET", + url: $(button).data('formurl'), + data: payload, + cache: false + }) + .done($.proxy(this._displayForm, this, box_id, button)) + .fail($.proxy(this._displayFailure, this, button)); + }, + /** + * Displays the form modal and set up ufForm + */ + _displayForm: function(box_id, button, data) { + + // Trigger pre-display event + $(button).trigger("displayForm." + this._name); + + // Append the form as a modal dialog to the body + $( "body" ).append(data); + $('#' + box_id).modal('show'); + + // Set focus on first element + if (this.settings.autofocusModalElement) { + $('#' + box_id).on('shown.bs.modal', function () { + $(this).find(".modal-body").find(':input:enabled:visible:first').focus(); + }); + } + + // Setup ufAlerts + var boxMsgTarget = $("#"+box_id+" #form-alerts"); + + // Show the alert. We could have info alert coming in + if (!boxMsgTarget.data('ufAlerts')) { + boxMsgTarget.ufAlerts(); + } + boxMsgTarget.ufAlerts('clear').ufAlerts('fetch').ufAlerts('render'); + + // Setup the loaded form with ufForm + $('#' + box_id).find("form").ufForm({ + validators: validators, + msgTarget: $("#"+box_id+" #form-alerts") + }) + .on("submitSuccess.ufForm", $.proxy(this._formPostSuccess, this, box_id, button)) + .on("submitError.ufForm", $.proxy(this._displayFormFaillure, this, box_id, button)); + }, + /** + * Action done when a form is successful + */ + _formPostSuccess: function(box_id, button, event, data) { + + // Trigger success event + $(button).trigger("formSuccess." + this._name, data); + + // Refresh page or close modal + if (this.settings.redirectAfterSuccess) { + window.location.reload(true); + } else { + $('#' + box_id).modal('hide'); + this.settings.mainAlertElement.ufAlerts('clear').ufAlerts('fetch').ufAlerts('render'); + } + }, + /** + * Fetch confirmation modal + */ + _fetchConfirmModal: function(event) { + + // Get the button element + var button = event.currentTarget; + + // Get the box_id. Define one if none is defined + var box_id = $(button).data('target'); + if (box_id == undefined) { + box_id = "formGeneratorModal"; + } + + // Delete any existing instance of the form with the same name + if($('#' + box_id).length) { + $('#' + box_id).remove(); + } + + // Prepare the ajax payload + var payload = $.extend({ + box_id: box_id, + box_title: $(button).data('confirmTitle') ? $(button).data('confirmTitle') : null, + confirm_message: $(button).data('confirmMessage') ? $(button).data('confirmMessage') : null, + confirm_warning: $(button).data('confirmWarning') ? $(button).data('confirmWarning') : null, + confirm_button: $(button).data('confirmButton') ? $(button).data('confirmButton') : null, + cancel_button: $(button).data('cancelButton') ? $(button).data('cancelButton') : null + }, button.dataset); + + // Fetch and render the form + $.ajax({ + type: "GET", + url: $(button).data('formurl') ? $(button).data('formurl') : site['uri']['public'] + "/forms/confirm", + data: payload, + cache: false + }) + .done($.proxy(this._displayConfirmation, this, box_id, button)) + .fail($.proxy(this._displayFailure, this, button)); + }, + /** + * Display confirmation modal + */ + _displayConfirmation: function(box_id, button, data) { + + // Trigger pre-display event + $(button).trigger("displayConfirmation." + this._name); + + // Append the form as a modal dialog to the body + $( "body" ).append(data); + $('#' + box_id).modal('show'); + + $('#' + box_id + ' .js-confirm').on('click', $.proxy(this._sendConfirmation, this, box_id, button)); + }, + /** + * Send confirmation query + */ + _sendConfirmation: function(box_id, button) { + + // Prepare payload + var url = $(button).data('postUrl'); + var method = ($(button).data('postMethod')) ? $(button).data('postMethod') : "POST"; + var data = { + bData: button.dataset, + csrf_name: $('#' + box_id).find("input[name='csrf_name']").val(), + csrf_value: $('#' + box_id).find("input[name='csrf_value']").val() + }; + + // Send ajax + $.ajax({ + type: method, + url: url, + data: data + }) + .done($.proxy(this._confirmationSuccess, this, box_id, button)) + .fail($.proxy(this._displayConfirmationFaillure, this, box_id, button)); + }, + /** + * Action done when a confirmation request is successful + */ + _confirmationSuccess: function(box_id, button, data) { + + // Trigger success event + $(button).trigger("confirmSuccess." + this._name, data); + + // Refresh page or close modal + if (this.settings.redirectAfterSuccess) { + + // Redirect if result contains intrusctions to + if (data.redirect) { + window.location.replace(data.redirect); + } else { + window.location.reload(true); + } + } else { + $('#' + box_id).modal('hide'); + this.settings.mainAlertElement.ufAlerts('clear').ufAlerts('fetch').ufAlerts('render'); + } + }, + /** + * Failure callback for ajax requests. Displays the error in the main alertElement + */ + _displayFailure: function(button, response) { + $(button).trigger("error." + this._name); + if ((typeof site !== "undefined") && site.debug.ajax && response.responseText) { + document.write(response.responseText); + document.close(); + } else { + if (this.settings.DEBUG) { + $.error("Error (" + response.status + "): " + response.responseText ); + } + this.settings.mainAlertElement.ufAlerts('clear').ufAlerts('fetch').ufAlerts('render'); + } + }, + /** + * Faillure callback for ajax requests to be displayed in a modal form + */ + _displayFormFaillure: function(box_id, button) { + $(button).trigger("error." + this._name); + $("#"+box_id+" #form-alerts").show(); + }, + /** + * Faillure callback for ajax requests to be displayed in a confirmation form + */ + _displayConfirmationFaillure: function(box_id, button) { + $(button).trigger("error." + this._name); + + // Setup ufAlerts + var boxMsgTarget = $("#"+box_id+" #confirmation-alerts"); + + // Show the alert. We could have info alert coming in + if (!boxMsgTarget.data('ufAlerts')) { + boxMsgTarget.ufAlerts(); + } + boxMsgTarget.ufAlerts('clear').ufAlerts('fetch').ufAlerts('render'); + }, + /** + * Completely destroy the ufAlerts plugin on the element. + */ + destroy: function() { + // Unbind any bound events + this.$elements.off('.' + this._name); + + // Grab jQuery wrapped element before plugin destruction + var $elements = this.$elements; + + // Remove plugin from element + this.$elements.removeData(this._name); + + return $elements; + } + }); + + // Handles instantiation and access to non-private methods. + $.fn[pluginName] = function(methodOrOptions) { + + // If the plugin is called on a non existing element, return nothing + if (this.length == 0) { + return this; + } + + // Grab plugin instance + var instance = $(this).data(pluginName); + + // If undefined or object, uses the default `display` method. + if (methodOrOptions === undefined || typeof methodOrOptions === 'object') { + var method = "display"; + var options = methodOrOptions; + } + // Otherwise ensure first parameter is a valid string + else if (typeof methodOrOptions === 'string') { + // Ensure not a private function + if (methodOrOptions.indexOf('_') !== 0) { + var method = methodOrOptions; + var options = Array.prototype.slice.call(arguments, 1)[0]; + } + else { + $.error( 'Method ' + methodOrOptions + ' is private!' ); + } + } + else { + $.error( 'Method ' + methodOrOptions + ' is invalid.' ); + } + + // Only initalise if not previously done. + if (!instance) { + $(this).data(pluginName, new Plugin(this, options)); + instance = $(this).data(pluginName); + } + + // Make sure method exist + if (typeof instance[method] === 'function') { + // Run the required method + return instance[method](options); + } else { + $.error( 'Method ' + method + ' does not exist.' ); + } + }; + + // Apply on default selector + $(".js-displayForm").formGenerator(); + $(".js-displayConfirm").formGenerator('confirm'); + +})(jQuery, window, document); \ No newline at end of file diff --git a/main/app/sprinkles/FormGenerator/bower.json b/main/app/sprinkles/FormGenerator/bower.json new file mode 100755 index 0000000..723b5cc --- /dev/null +++ b/main/app/sprinkles/FormGenerator/bower.json @@ -0,0 +1,30 @@ +{ + "name": "formgenerator-assets", + "version": "0.0.1", + "homepage": "https://github.com/lcharette/UF_FormGenerator", + "authors": [ + "lcharette" + ], + "moduleType": [ + "node" + ], + "license": "MIT", + "ignore": [ + "**/.*", + "node_modules", + "bower_components", + "assets/vendor", + "examples", + "demo-resources", + "demo", + "test", + "tests" + ], + "dependencies": { + "bootstrap3-typeahead": "~3.1.0" + }, + "resolutions": { + "jquery": ">= 2.2.4", + "bootstrap": "3.x" + } +} diff --git a/main/app/sprinkles/FormGenerator/composer.json b/main/app/sprinkles/FormGenerator/composer.json new file mode 100755 index 0000000..e37a372 --- /dev/null +++ b/main/app/sprinkles/FormGenerator/composer.json @@ -0,0 +1,25 @@ +{ + "name": "lcharette/uf_formgenerator", + "type": "userfrosting-sprinkle", + "description": "Form generator for UserFrosting V4", + "keywords": ["Form", "generator", "userfrosting"], + "homepage": "https://github.com/lcharette/UF_FormGenerator", + "license" : "MIT", + "authors" : [ + { + "name": "Louis Charette", + "homepage": "https://github.com/lcharette" + } + ], + "require": { + "php": ">=5.6" + }, + "autoload": { + "psr-4": { + "UserFrosting\\Sprinkle\\FormGenerator\\": "src/" + } + }, + "extra": { + "installer-name": "FormGenerator" + } +} diff --git a/main/app/sprinkles/FormGenerator/locale/en_US/FormGenerator.php b/main/app/sprinkles/FormGenerator/locale/en_US/FormGenerator.php new file mode 100755 index 0000000..41d7bfa --- /dev/null +++ b/main/app/sprinkles/FormGenerator/locale/en_US/FormGenerator.php @@ -0,0 +1,14 @@ + [ + "@TRANSLATION" => "Confirm action", + + "MESSAGE" => "Are you sure you want to do this?", + + "WARNING" => "This action cannot be undone.", + + "YES" => "Yes, do it" + ] +]; \ No newline at end of file diff --git a/main/app/sprinkles/FormGenerator/locale/fr_FR/FormGenerator.php b/main/app/sprinkles/FormGenerator/locale/fr_FR/FormGenerator.php new file mode 100755 index 0000000..9ede59c --- /dev/null +++ b/main/app/sprinkles/FormGenerator/locale/fr_FR/FormGenerator.php @@ -0,0 +1,14 @@ + [ + "@TRANSLATION" => "Confirmer l'action", + + "MESSAGE" => "Êtes-vous certain de vouloir faire ceci?", + + "WARNING" => "Cette action ne peut pas être annulée.", + + "YES" => "Oui" + ] +]; \ No newline at end of file diff --git a/main/app/sprinkles/FormGenerator/routes/FormGenerator.php b/main/app/sprinkles/FormGenerator/routes/FormGenerator.php new file mode 100755 index 0000000..0b7ea51 --- /dev/null +++ b/main/app/sprinkles/FormGenerator/routes/FormGenerator.php @@ -0,0 +1,12 @@ +get('/forms/confirm','UserFrosting\Sprinkle\FormGenerator\Controller\FormGeneratorController:confirm'); \ No newline at end of file diff --git a/main/app/sprinkles/FormGenerator/src/Controller/FormGeneratorController.php b/main/app/sprinkles/FormGenerator/src/Controller/FormGeneratorController.php new file mode 100755 index 0000000..e731011 --- /dev/null +++ b/main/app/sprinkles/FormGenerator/src/Controller/FormGeneratorController.php @@ -0,0 +1,26 @@ +ci->view->render($response, 'FormGenerator/confirm.html.twig', $request->getQueryParams()); + } +} diff --git a/main/app/sprinkles/FormGenerator/src/Element/Alert.php b/main/app/sprinkles/FormGenerator/src/Element/Alert.php new file mode 100755 index 0000000..f848b5c --- /dev/null +++ b/main/app/sprinkles/FormGenerator/src/Element/Alert.php @@ -0,0 +1,33 @@ +element = array_merge([ + "class" => "alert-danger", + "icon" => "fa-ban", + "value" => $this->value, + "name" => $this->name + ], $this->element); + } +} diff --git a/main/app/sprinkles/FormGenerator/src/Element/BaseInput.php b/main/app/sprinkles/FormGenerator/src/Element/BaseInput.php new file mode 100755 index 0000000..d892001 --- /dev/null +++ b/main/app/sprinkles/FormGenerator/src/Element/BaseInput.php @@ -0,0 +1,111 @@ +name = $name; + $this->element = $element; + $this->value = $value; + } + + /** + * parse function. + * + * Return the parsed input attributes + * @access public + * @return void + */ + public function parse() + { + $this->applyTransformations(); + return $this->element; + } + + /** + * translateArgValue function. + * + * Translate the value of passed argument using the Translator Facade + * @access public + * @param String $argument + * @return void + */ + public function translateArgValue($argument) { + if (isset($this->element[$argument])) { + $this->element[$argument] = Translator::translate($this->element[$argument]); + } + } + + /** + * getValue function. + * + * Return the value of the current input element. If not value is set in + * `$this->value`, return the default value (from the schema data), if any. + * @access public + * @return string The input current value + */ + public function getValue() { + if (isset($this->value) && $this->value !== null) { + return $this->value; + } else if (isset($this->element['default'])) { + return $this->element['default']; + } else { + return ""; + } + } + + /** + * applyTransformations function. + * + * Add defaut attributes to the current input element. Also transform + * attributes values passed from the schema + * @access protected + * @abstract + * @return void + */ + abstract protected function applyTransformations(); +} diff --git a/main/app/sprinkles/FormGenerator/src/Element/Checkbox.php b/main/app/sprinkles/FormGenerator/src/Element/Checkbox.php new file mode 100755 index 0000000..59e6eaf --- /dev/null +++ b/main/app/sprinkles/FormGenerator/src/Element/Checkbox.php @@ -0,0 +1,38 @@ +element = array_merge([ + "class" => "js-icheck", + "name" => $this->name, + "id" => "field_" . $this->name, + "binary" => true + ], $this->element); + + // We add the check status instead of the value + if ($this->element["binary"] !== false && $this->getValue() == 1) { + $this->element["checked"] = "checked"; + } + } +} diff --git a/main/app/sprinkles/FormGenerator/src/Element/Hidden.php b/main/app/sprinkles/FormGenerator/src/Element/Hidden.php new file mode 100755 index 0000000..08c22f7 --- /dev/null +++ b/main/app/sprinkles/FormGenerator/src/Element/Hidden.php @@ -0,0 +1,32 @@ +element = array_merge([ + "value" => $this->getValue(), + "name" => $this->name, + "id" => "field_" . $this->name + ], $this->element); + } +} diff --git a/main/app/sprinkles/FormGenerator/src/Element/InputInterface.php b/main/app/sprinkles/FormGenerator/src/Element/InputInterface.php new file mode 100755 index 0000000..7405109 --- /dev/null +++ b/main/app/sprinkles/FormGenerator/src/Element/InputInterface.php @@ -0,0 +1,19 @@ +element = array_merge([ + "class" => "form-control js-select2", + "value" => $this->getValue(), + "name" => $this->name, + "id" => "field_" . $this->name + ], $this->element); + + // Placeholder is required to be in `data-*` for select 2 + // Plus we translate the placeholder + if (isset($this->element["placeholder"])) { + $this->element["data-placeholder"] = $this->element["placeholder"]; + unset($this->element["placeholder"]); + $this->translateArgValue('data-placeholder'); + } + } +} diff --git a/main/app/sprinkles/FormGenerator/src/Element/Text.php b/main/app/sprinkles/FormGenerator/src/Element/Text.php new file mode 100755 index 0000000..b936fe2 --- /dev/null +++ b/main/app/sprinkles/FormGenerator/src/Element/Text.php @@ -0,0 +1,37 @@ +element = array_merge([ + "autocomplete" => "off", + "class" => "form-control", + "value" => $this->getValue(), + "name" => $this->name, + "id" => "field_" . $this->name + ], $this->element); + + // Translate placeholder + $this->translateArgValue('placeholder'); + } +} diff --git a/main/app/sprinkles/FormGenerator/src/Element/Textarea.php b/main/app/sprinkles/FormGenerator/src/Element/Textarea.php new file mode 100755 index 0000000..bec3a6c --- /dev/null +++ b/main/app/sprinkles/FormGenerator/src/Element/Textarea.php @@ -0,0 +1,38 @@ +element = array_merge([ + "autocomplete" => "off", + "class" => "form-control", + "value" => $this->getValue(), + "name" => $this->name, + "rows" => 3, + "id" => "field_" . $this->name + ], $this->element); + + // Translate placeholder + $this->translateArgValue('placeholder'); + } +} diff --git a/main/app/sprinkles/FormGenerator/src/Form.php b/main/app/sprinkles/FormGenerator/src/Form.php new file mode 100755 index 0000000..e845e3e --- /dev/null +++ b/main/app/sprinkles/FormGenerator/src/Form.php @@ -0,0 +1,188 @@ +setSchema($schema); + $this->setData($data); + } + + /** + * Set the form current values + * + * @param array|object $data The form values + */ + public function setData($data) + { + if ($data instanceof Collection || $data instanceof Model) { + $this->data = $data->toArray(); + } else if (is_array($data) || $data instanceof Repository) { + $this->data = $data; + } else { + throw new \InvalidArgumentException("Data must be an array, a Collection, a Model or a Repository"); + } + } + + /** + * Set the schema for this validator. + * + * @param RequestSchemaInterface $schema A RequestSchemaInterface object, containing the form definition. + */ + public function setSchema(RequestSchemaInterface $schema) + { + $this->schema = $schema; + } + + /** + * Use to define the value of a form input when `setData` is already set + * + * @param mixed $inputName + * @param mixed $value + * @return void + */ + public function setValue($inputName, $value) + { + $this->data[$inputName] = $value; + } + + /** + * Function used to overwrite the input argument from a schema file + * Can also be used to overwrite an argument hardcoded in the Twig file. + * Use `setCustomFormData` to set any other tag. + * + * @param string $inputName The input name where the argument will be added + * @param string $property The argument name. Example "data-color" + * @param string $data The value of the argument + * @return void + */ + public function setInputArgument($inputName, $property, $data) + { + if ($this->schema->has($inputName)) { + // Get the element and force set the property + $element = $this->schema->get($inputName); + $element['form'][$property] = $data; + + // Push back the modifyed element in the schema + $this->schema->set($inputName, $element); + } + } + + /** + * Function used to set options of a select element. Shortcut for using + * `setInputArgument` and `setValue`. + * + * @param string $inputName The select name to add options to + * @param array $data An array of `value => label` options + * @param string $selected The selected key + * @return void + */ + public function setOptions($inputName, $data = [], $selected = null) + { + // Set opdations + $this->setInputArgument($inputName, 'options', $data); + + // Set the value + if (!is_null($selected)) { + $this->setValue($inputName, $selected); + } + } + + /** + * Function to set the form namespace. + * Use the form namespace to wrap the fields name in a top level array. + * Useful when using multiple schemas at once or if the names are using dot syntaxt. + * See : http://stackoverflow.com/a/20365198/445757 + * + * @param string $namespace + * @return void + */ + public function setFormNamespace($namespace) + { + $this->formNamespace = $namespace; + } + + /** + * Generate an array contining all nececerry value to generate a form + * with Twig. + * + * @return array The form fields data + */ + public function generate() + { + $form = collect([]); + + // Loop all the the fields in the schema + foreach ($this->schema->all() as $name => $input) { + + // Skip the one that don't have a `form` definition + if (isset($input['form'])) { + + // Get the value from the data + $value = isset($this->data[$name]) ? $this->data[$name] : null; + + // Add the namespace to the name if it's defined + $name = ($this->formNamespace != "") ? $this->formNamespace."[".$name."]" : $name; + + // Get the element class and make sure it exist + $type = (isset($input['form']['type'])) ? $input['form']['type'] : "text"; + $type = "UserFrosting\\Sprinkle\\FormGenerator\\Element\\" . Str::studly($type); + + // If class doesn't esist, default to Text element + if (!class_exists($type)) { + $type = "UserFrosting\\Sprinkle\\FormGenerator\\Element\\Text"; + } + + // Create a new instance + $element = new $type($name, $input['form'], $value); + + // Push data to `$form` + $form->put($name, $element->parse()); + } + } + + return $form->toArray(); + } +} diff --git a/main/app/sprinkles/FormGenerator/templates/FormGenerator/FormGenerator.html.twig b/main/app/sprinkles/FormGenerator/templates/FormGenerator/FormGenerator.html.twig new file mode 100755 index 0000000..c902064 --- /dev/null +++ b/main/app/sprinkles/FormGenerator/templates/FormGenerator/FormGenerator.html.twig @@ -0,0 +1,33 @@ +{% import "FormGenerator/macros/checkbox.html.twig" as checkbox %} +{% import "FormGenerator/macros/textarea.html.twig" as textarea %} +{% import "FormGenerator/macros/select.html.twig" as select %} +{% import "FormGenerator/macros/text.html.twig" as text %} +{% import "FormGenerator/macros/hidden.html.twig" as hidden %} +{% import "FormGenerator/macros/alert.html.twig" as alert %} + +{% for name,input in fields %} + {% if input.type == "hidden" %} + {{ hidden.generate(input) }} + {% elseif input.type == "alert" %} + {{ alert.generate(input) }} + {% else %} + {% if not input.hidden %} +
    + {% if formLayout == 'horizontal' %}
    + {% endif %} + {% endif %} +{% endfor %} \ No newline at end of file diff --git a/main/app/sprinkles/FormGenerator/templates/FormGenerator/confirm.html.twig b/main/app/sprinkles/FormGenerator/templates/FormGenerator/confirm.html.twig new file mode 100755 index 0000000..96d0072 --- /dev/null +++ b/main/app/sprinkles/FormGenerator/templates/FormGenerator/confirm.html.twig @@ -0,0 +1,28 @@ + \ No newline at end of file diff --git a/main/app/sprinkles/FormGenerator/templates/FormGenerator/macros/alert.html.twig b/main/app/sprinkles/FormGenerator/templates/FormGenerator/macros/alert.html.twig new file mode 100755 index 0000000..21636f0 --- /dev/null +++ b/main/app/sprinkles/FormGenerator/templates/FormGenerator/macros/alert.html.twig @@ -0,0 +1,5 @@ +{% macro generate(input) %} +
    + {% if input.icon %} {% endif %}{{ translate(input.value) }} +
    +{% endmacro %} \ No newline at end of file diff --git a/main/app/sprinkles/FormGenerator/templates/FormGenerator/macros/checkbox.html.twig b/main/app/sprinkles/FormGenerator/templates/FormGenerator/macros/checkbox.html.twig new file mode 100755 index 0000000..9065651 --- /dev/null +++ b/main/app/sprinkles/FormGenerator/templates/FormGenerator/macros/checkbox.html.twig @@ -0,0 +1,5 @@ +{% macro generate(input) %} +
    + +
    +{% endmacro %} \ No newline at end of file diff --git a/main/app/sprinkles/FormGenerator/templates/FormGenerator/macros/hidden.html.twig b/main/app/sprinkles/FormGenerator/templates/FormGenerator/macros/hidden.html.twig new file mode 100755 index 0000000..a30fb7a --- /dev/null +++ b/main/app/sprinkles/FormGenerator/templates/FormGenerator/macros/hidden.html.twig @@ -0,0 +1,3 @@ +{% macro generate(input) %} + +{% endmacro %} \ No newline at end of file diff --git a/main/app/sprinkles/FormGenerator/templates/FormGenerator/macros/select.html.twig b/main/app/sprinkles/FormGenerator/templates/FormGenerator/macros/select.html.twig new file mode 100755 index 0000000..256c3f2 --- /dev/null +++ b/main/app/sprinkles/FormGenerator/templates/FormGenerator/macros/select.html.twig @@ -0,0 +1,5 @@ +{% macro generate(input) %} + + {% for option, label in input.options %}{% endfor %} + +{% endmacro %} \ No newline at end of file diff --git a/main/app/sprinkles/FormGenerator/templates/FormGenerator/macros/text.html.twig b/main/app/sprinkles/FormGenerator/templates/FormGenerator/macros/text.html.twig new file mode 100755 index 0000000..30beb5b --- /dev/null +++ b/main/app/sprinkles/FormGenerator/templates/FormGenerator/macros/text.html.twig @@ -0,0 +1,10 @@ +{% macro generate(input) %} + {% if input.icon %} +
    + + {% else %} +
    + {% endif %} + +
    +{% endmacro %} diff --git a/main/app/sprinkles/FormGenerator/templates/FormGenerator/macros/textarea.html.twig b/main/app/sprinkles/FormGenerator/templates/FormGenerator/macros/textarea.html.twig new file mode 100755 index 0000000..ce5e2e7 --- /dev/null +++ b/main/app/sprinkles/FormGenerator/templates/FormGenerator/macros/textarea.html.twig @@ -0,0 +1,5 @@ +{% macro generate(input) %} + {% if input.icon %}
    {% endif %} + {{input.value}} + {% if input.icon %}
    {% endif %} +{% endmacro %} \ No newline at end of file diff --git a/main/app/sprinkles/FormGenerator/templates/FormGenerator/modal-large.html.twig b/main/app/sprinkles/FormGenerator/templates/FormGenerator/modal-large.html.twig new file mode 100755 index 0000000..b2de3f9 --- /dev/null +++ b/main/app/sprinkles/FormGenerator/templates/FormGenerator/modal-large.html.twig @@ -0,0 +1,3 @@ +{% extends "FormGenerator/modal.html.twig" %} + +{% block modal_size %}modal-lg{% endblock %} diff --git a/main/app/sprinkles/FormGenerator/templates/FormGenerator/modal.html.twig b/main/app/sprinkles/FormGenerator/templates/FormGenerator/modal.html.twig new file mode 100755 index 0000000..3c66201 --- /dev/null +++ b/main/app/sprinkles/FormGenerator/templates/FormGenerator/modal.html.twig @@ -0,0 +1,62 @@ +{% block modal %} + +{% endblock %} + +{% block scripts_page %} + + +{% endblock %} \ No newline at end of file diff --git a/main/app/sprinkles/FormGenerator/templates/FormGenerator/typehead.html.twig b/main/app/sprinkles/FormGenerator/templates/FormGenerator/typehead.html.twig new file mode 100755 index 0000000..150427a --- /dev/null +++ b/main/app/sprinkles/FormGenerator/templates/FormGenerator/typehead.html.twig @@ -0,0 +1,15 @@ +{% extends "FormGenerator/modal.html.twig" %} + +{% block scripts_page %} +{{ parent() }} + + +{% endblock %} \ No newline at end of file diff --git a/main/app/sprinkles/FormGenerator/tests/Unit/FormGeneratorTest.php b/main/app/sprinkles/FormGenerator/tests/Unit/FormGeneratorTest.php new file mode 100755 index 0000000..066e98d --- /dev/null +++ b/main/app/sprinkles/FormGenerator/tests/Unit/FormGeneratorTest.php @@ -0,0 +1,408 @@ +basePath = __DIR__ . '/data'; + } + + /** + * Test the base `Test` element class works on it's own + */ + public function testTextFormElement() + { + // Get Schema + $loader = new YamlFileLoader($this->basePath . '/good.json'); + $schema = new RequestSchemaRepository($loader->load()); + + // Get TextInput from the `name` element of the schema + $inputSchema = $schema["name"]["form"]; + $textInput = new \UserFrosting\Sprinkle\FormGenerator\Element\Text("name", $inputSchema); + + // Test instanceof $textInput + $this->assertInstanceof(InputInterface::class, $textInput); + + // Parse the input + $text = $textInput->parse(); + + // Test the parsing + $expected = [ + "type" => "text", + "label" => "Project Name", + "icon" => "fa-flag", + "autocomplete" => "off", + "class" => "form-control", + "placeholder" => "Project Name", + 'name' => 'name', + 'id' => 'field_name', + 'value' => '' + ]; + + // We test the generated result + $this->assertEquals($expected, $text); + } + + /** + * This test make sure the `Text` element works correctly when a current + * value is passed to the constructor. Should return the same as the + * previous test, but with the `value` setup instead of empty + */ + public function testTextFormElementWithData() + { + // Get Schema + $loader = new YamlFileLoader($this->basePath . '/good.json'); + $schema = new RequestSchemaRepository($loader->load()); + + // Get TextInput from the `name` element of the schema + $inputSchema = $schema["name"]["form"]; + $textInput = new \UserFrosting\Sprinkle\FormGenerator\Element\Text("name", $inputSchema, "The Bar project"); + + // Test instanceof $textInput + $this->assertInstanceof(InputInterface::class, $textInput); + + // Parse the input + $text = $textInput->parse(); + + // Test the parsing + $expected = [ + "type" => "text", + "label" => "Project Name", + "icon" => "fa-flag", + "autocomplete" => "off", + "class" => "form-control", + "placeholder" => "Project Name", + 'name' => 'name', + 'id' => 'field_name', + 'value' => 'The Bar project' + ]; + + // We test the generated result + $this->assertEquals($expected, $text); + } + + /** + * This test is the same as the one before, but we test the `owener` field with some data + * This make sure the `default` schema field will work correctly when empty data is passed + */ + public function testTextFormElementWithEmptyData() + { + // Get Schema + $loader = new YamlFileLoader($this->basePath . '/good.json'); + $schema = new RequestSchemaRepository($loader->load()); + + // Get TextInput from the `name` element of the schema + $inputSchema = $schema["owner"]["form"]; + $textInput = new \UserFrosting\Sprinkle\FormGenerator\Element\Text("owner", $inputSchema, ""); + + // Test instanceof $textInput + $this->assertInstanceof(InputInterface::class, $textInput); + + // Parse the input + $text = $textInput->parse(); + + // Test the parsing + $expected = [ + 'label' => 'Project Owner', + 'autocomplete' => 'off', + 'class' => 'form-control', + 'value' => '', //Shoudn't be a value here ! "" is overwritting "Foo" + 'name' => 'owner', + 'id' => 'owner', + 'type' => 'text', + 'icon' => 'fa-user', + 'placeholder' => 'Project Owner', + 'default' => 'Foo' + ]; + + // We test the generated result + $this->assertEquals($expected, $text); + } + + /** + * Test the Form Class. + * Run the test with no current values (empty form) + */ + public function testForm() + { + // Get Schema + $loader = new YamlFileLoader($this->basePath . '/good.json'); + $schema = new RequestSchemaRepository($loader->load()); + + // Generate the form + $form = new Form($schema); + + // Test to make sure the class creation is fine + $this->assertInstanceof(Form::class, $form); + + // Test the form generation + $generatedForm = $form->generate(); + $this->assertInternalType("array", $generatedForm); + + // Test one of the form input + $expected = [ + 'number' => [ + 'label' => 'Project Number', + 'autocomplete' => 'off', + 'class' => 'form-control', + 'value' => '', + 'name' => 'number', + 'id' => 'field_number', + 'type' => 'number', + 'icon' => 'fa-edit', + 'placeholder' => 'Project Number' + ], + 'owner' => [ + 'label' => 'Project Owner', + 'autocomplete' => 'off', + 'class' => 'form-control', + 'value' => 'Foo', + 'name' => 'owner', + 'id' => 'owner', + 'type' => 'text', + 'icon' => 'fa-user', + 'placeholder' => 'Project Owner', + 'default' => 'Foo' + ], + 'name' => [ + 'label' => 'Project Name', + 'autocomplete' => 'off', + 'class' => 'form-control', + 'value' => '', + 'name' => 'name', + 'id' => 'field_name', + 'type' => 'text', + 'icon' => 'fa-flag', + 'placeholder' => 'Project Name' + ], + 'description' => [ + 'label' => 'Project Description', + 'autocomplete' => 'off', + 'class' => 'form-control', + 'value' => '', + 'name' => 'description', + 'rows' => 5, + 'id' => 'field_description', + 'type' => 'textarea', + 'icon' => 'fa-pencil', + 'placeholder' => 'Project Description' + ], + 'status' => [ + 'label' => 'Project Status', + 'class' => 'form-control js-select2', + 'value' => '', + 'name' => 'status', + 'id' => 'field_status', + 'type' => 'select', + 'options' => [ + 0 => 'Closed', + 1 => 'Open' + ] + ], + 'active' => [ + 'label' => 'Active', + 'class' => 'js-icheck', + 'name' => 'active', + 'id' => 'field_active', + 'type' => 'checkbox', + 'binary' => true + ], + 'hidden' => [ + 'value' => 'Something', + 'name' => 'hidden', + 'id' => 'field_hidden', + 'type' => 'hidden' + ], + 'alert' => [ + 'class' => 'alert-success', + 'icon' => 'fa-check', + 'value' => 'You\'re awesome!', + 'name' => 'alert', + 'type' => 'alert' + ] + ]; + + // We test the generated result + $this->assertEquals($expected, $generatedForm); + } + + /** + * Test the Form Clas with values to make sure filled form works correctly + */ + public function testFormWithData() + { + // Get Schema + $loader = new YamlFileLoader($this->basePath . '/good.json'); + $schema = new RequestSchemaRepository($loader->load()); + + // The data + $data = [ + "name" => "Bar project", + "owner" => "", + "description" => "The bar project is less awesome, but at least it's open.", + "status" => 1, + "hiddenString" => "The Bar secret code is...", + "completion" => 12, + "active" => true + ]; + + // Generate the form + $form = new Form($schema, $data); + + // Test to make sure the class creation is fine + $this->assertInstanceof(Form::class, $form); + + // Test the form generation + $generatedForm = $form->generate(); + $this->assertInternalType("array", $generatedForm); + + // Test one of the form input + $expected = [ + 'number' => [ + 'label' => 'Project Number', + 'autocomplete' => 'off', + 'class' => 'form-control', + 'value' => '', + 'name' => 'number', + 'id' => 'field_number', + 'type' => 'number', + 'icon' => 'fa-edit', + 'placeholder' => 'Project Number' + ], + 'name' => [ + 'label' => 'Project Name', + 'autocomplete' => 'off', + 'class' => 'form-control', + 'value' => 'Bar project', //Value here ! + 'name' => 'name', + 'id' => 'field_name', + 'type' => 'text', + 'icon' => 'fa-flag', + 'placeholder' => 'Project Name' + ], + 'owner' => [ + 'label' => 'Project Owner', + 'autocomplete' => 'off', + 'class' => 'form-control', + 'value' => '', //Shoudn't be a value here ! "" is overwritting "Foo" + 'name' => 'owner', + 'id' => 'owner', + 'type' => 'text', + 'icon' => 'fa-user', + 'placeholder' => 'Project Owner', + 'default' => 'Foo' + ], + 'description' => [ + 'label' => 'Project Description', + 'autocomplete' => 'off', + 'class' => 'form-control', + 'value' => 'The bar project is less awesome, but at least it\'s open.', //Value here ! + 'name' => 'description', + 'rows' => 5, + 'id' => 'field_description', + 'type' => 'textarea', + 'icon' => 'fa-pencil', + 'placeholder' => 'Project Description' + ], + 'status' => [ + 'label' => 'Project Status', + 'class' => 'form-control js-select2', + 'value' => 1, //Value here ! + 'name' => 'status', + 'id' => 'field_status', + 'type' => 'select', + 'options' => [ + 0 => 'Closed', + 1 => 'Open' + ] + ], + 'active' => [ + 'label' => 'Active', + 'class' => 'js-icheck', + 'name' => 'active', + 'id' => 'field_active', + 'type' => 'checkbox', + 'checked' => 'checked', //Value here ! + 'binary' => true + ], + 'hidden' => [ + 'value' => 'Something', + 'name' => 'hidden', + 'id' => 'field_hidden', + 'type' => 'hidden' + ], + 'alert' => [ + 'class' => 'alert-success', + 'icon' => 'fa-check', + 'value' => 'You\'re awesome!', + 'name' => 'alert', + 'type' => 'alert' + ] + ]; + + // We test the generated result + $this->assertEquals($expected, $generatedForm); + } + + /** + * Test a non existant input type. It's supposed to not find the class and + * default back to the `Text` element class. + */ + public function testUndefinedFormElement() + { + // Get Schema + $loader = new YamlFileLoader($this->basePath . '/bad.json'); + $schema = new RequestSchemaRepository($loader->load()); + + // Generate the form + $form = new Form($schema); + + // Test to make sure the class creation is fine + $this->assertInstanceof(Form::class, $form); + + // Test the form generation + $generatedForm = $form->generate(); + $this->assertInternalType("array", $generatedForm); + + // Test one of the form input + $expected = [ + "type" => "foo", + "autocomplete" => "off", + "class" => "form-control", + 'name' => 'myField', + 'id' => 'field_myField', + 'value' => '' + ]; + + // We test the generated result + $this->assertEquals($expected, $generatedForm['myField']); + } +} diff --git a/main/app/sprinkles/FormGenerator/tests/Unit/data/bad.json b/main/app/sprinkles/FormGenerator/tests/Unit/data/bad.json new file mode 100755 index 0000000..683add2 --- /dev/null +++ b/main/app/sprinkles/FormGenerator/tests/Unit/data/bad.json @@ -0,0 +1,12 @@ +{ + "myField" : { + "form" : { + "type" : "foo" + } + }, + "myOtherField" : { + "form" : { + "value" : "Bar" + } + } +} diff --git a/main/app/sprinkles/FormGenerator/tests/Unit/data/good.json b/main/app/sprinkles/FormGenerator/tests/Unit/data/good.json new file mode 100755 index 0000000..61d5233 --- /dev/null +++ b/main/app/sprinkles/FormGenerator/tests/Unit/data/good.json @@ -0,0 +1,67 @@ +{ + "number" : { + "form" : { + "type" : "number", + "label" : "Project Number", + "icon" : "fa-edit", + "placeholder" : "Project Number" + } + }, + "name" : { + "form" : { + "type" : "text", + "label" : "Project Name", + "icon" : "fa-flag", + "placeholder" : "Project Name" + } + }, + "owner" : { + "form" : { + "type" : "text", + "label" : "Project Owner", + "icon" : "fa-user", + "id" : "owner", + "placeholder" : "Project Owner", + "default" : "Foo" + } + }, + "description" : { + "form" : { + "type" : "textarea", + "label" : "Project Description", + "icon" : "fa-pencil", + "placeholder" : "Project Description", + "rows" : 5 + } + }, + "status" : { + "form" : { + "type" : "select", + "label" : "Project Status", + "options" : { + "0" : "Closed", + "1" : "Open" + } + } + }, + "active" : { + "form" : { + "type" : "checkbox", + "label" : "Active" + } + }, + "hidden" : { + "form" : { + "type" : "hidden", + "value" : "Something" + } + }, + "alert" : { + "form" : { + "type" : "alert", + "class" : "alert-success", + "icon" : "fa-check", + "value" : "You're awesome!" + } + } +} diff --git a/main/app/sprinkles/account/asset-bundles.json b/main/app/sprinkles/account/asset-bundles.json new file mode 100755 index 0000000..77ee559 --- /dev/null +++ b/main/app/sprinkles/account/asset-bundles.json @@ -0,0 +1,79 @@ +{ + "bundle": { + "js/pages/account-settings": { + "scripts": [ + "userfrosting/js/pages/account-settings.js" + ], + "options": { + "result": { + "type": { + "scripts": "plain" + } + } + } + }, + "js/pages/forgot-password": { + "scripts": [ + "userfrosting/js/pages/forgot-password.js" + ], + "options": { + "result": { + "type": { + "scripts": "plain" + } + } + } + }, + "js/pages/resend-verification": { + "scripts": [ + "userfrosting/js/pages/resend-verification.js" + ], + "options": { + "result": { + "type": { + "scripts": "plain" + } + } + } + }, + "js/pages/set-or-reset-password": { + "scripts": [ + "userfrosting/js/pages/set-or-reset-password.js" + ], + "options": { + "result": { + "type": { + "scripts": "plain" + } + } + } + }, + "js/pages/register": { + "scripts": [ + "vendor/speakingurl/speakingurl.min.js", + "userfrosting/js/uf-captcha.js", + "userfrosting/js/pages/register.js" + ], + "options": { + "result": { + "type": { + "scripts": "plain" + } + } + } + }, + "js/pages/sign-in": { + "scripts": [ + "vendor/urijs/src/URI.js", + "userfrosting/js/pages/sign-in.js" + ], + "options": { + "result": { + "type": { + "scripts": "plain" + } + } + } + } + } +} \ No newline at end of file diff --git a/main/app/sprinkles/account/assets/userfrosting/js/pages/account-settings.js b/main/app/sprinkles/account/assets/userfrosting/js/pages/account-settings.js new file mode 100755 index 0000000..8d8d2e7 --- /dev/null +++ b/main/app/sprinkles/account/assets/userfrosting/js/pages/account-settings.js @@ -0,0 +1,29 @@ +/** + * Page-specific Javascript file. Should generally be included as a separate asset bundle in your page template. + * example: {{ assets.js('js/pages/sign-in-or-register') | raw }} + * + * This script depends on validation rules specified in pages/partials/page.js.twig. + * + * Target page: account/settings + */ +$(document).ready(function() { + + // Apply select2 to locale field + $('.js-select2').select2(); + + $("#account-settings").ufForm({ + validators: page.validators.account_settings, + msgTarget: $("#alerts-page") + }).on("submitSuccess.ufForm", function() { + // Reload the page on success + window.location.reload(); + }); + + $("#profile-settings").ufForm({ + validators: page.validators.profile_settings, + msgTarget: $("#alerts-page") + }).on("submitSuccess.ufForm", function() { + // Reload the page on success + window.location.reload(); + }); +}); diff --git a/main/app/sprinkles/account/assets/userfrosting/js/pages/forgot-password.js b/main/app/sprinkles/account/assets/userfrosting/js/pages/forgot-password.js new file mode 100755 index 0000000..3f24311 --- /dev/null +++ b/main/app/sprinkles/account/assets/userfrosting/js/pages/forgot-password.js @@ -0,0 +1,19 @@ +/** + * Page-specific Javascript file. Should generally be included as a separate asset bundle in your page template. + * example: {{ assets.js('js/pages/sign-in-or-register') | raw }} + * + * This script depends on validation rules specified in pages/partials/page.js.twig. + * + * Target page: account/forgot-password + */ +$(document).ready(function() { + + // TODO: Process form + $("#request-password-reset").ufForm({ + validators: page.validators.forgot_password, + msgTarget: $("#alerts-page") + }).on("submitSuccess.ufForm", function() { + // Forward to login page on success + window.location.replace(site.uri.public + "/account/sign-in"); + }); +}); diff --git a/main/app/sprinkles/account/assets/userfrosting/js/pages/register.js b/main/app/sprinkles/account/assets/userfrosting/js/pages/register.js new file mode 100755 index 0000000..d855bb9 --- /dev/null +++ b/main/app/sprinkles/account/assets/userfrosting/js/pages/register.js @@ -0,0 +1,94 @@ +/** + * Page-specific Javascript file. Should generally be included as a separate asset bundle in your page template. + * example: {{ assets.js('js/pages/sign-in-or-register') | raw }} + * + * This script depends on validation rules specified in pages/partials/page.js.twig. + * + * Target page: account/register + */ +$(document).ready(function() { + // TOS modal + $(this).find('.js-show-tos').click(function() { + $("body").ufModal({ + sourceUrl: site.uri.public + "/modals/account/tos", + msgTarget: $("#alerts-page") + }); + }); + + // Auto-generate username when name is filled in + var autoGenerate = true; + $("#register").find('input[name=first_name], input[name=last_name]').on('input change', function() { + if (!autoGenerate) { + return; + } + + var form = $("#register"); + + var firstName = form.find('input[name=first_name]').val().trim(); + var lastName = form.find('input[name=last_name]').val().trim(); + + if (!firstName && !lastName) { + return; + } + + var userName = getSlug(firstName + ' ' + lastName, { + separator: '.' + }); + // Set slug + form.find('input[name=user_name]').val(userName); + }); + + // Autovalidate username field on a delay + var timer; + $("#register").find('input[name=first_name], input[name=last_name], input[name=user_name]').on('input change', function() { + clearTimeout(timer); // Clear the timer so we don't end up with dupes. + timer = setTimeout(function() { // assign timer a new timeout + $("#register").find('input[name=user_name]').valid(); + }, 500); + }); + + // Enable/disable username suggestions in registration page + $("#register").find('#form-register-username-suggest').on('click', function(e) { + e.preventDefault(); + var form = $("#register"); + $.getJSON(site.uri.public + '/account/suggest-username') + .done(function (data) { + // Set suggestion + form.find('input[name=user_name]').val(data.user_name); + }); + }); + + // Turn off autogenerate when someone enters stuff manually in user_name + $("#register").find('input[name=user_name]').on('input', function() { + autoGenerate = false; + }); + + // Add remote rule for checking usernames on the fly + var registrationValidators = $.extend( + true, // deep extend + page.validators.register, + { + rules: { + user_name: { + remote: { + url: site.uri.public + '/account/check-username', + dataType: 'text' + } + } + } + } + ); + + // Handles form submission + $("#register").ufForm({ + validators: registrationValidators, + msgTarget: $("#alerts-page"), + keyupDelay: 500 + }).on("submitSuccess.ufForm", function() { + // Reload to clear form and show alerts + window.location.reload(); + }).on("submitError.ufForm", function() { + // Reload captcha + $("#captcha").captcha(); + }); +}); diff --git a/main/app/sprinkles/account/assets/userfrosting/js/pages/resend-verification.js b/main/app/sprinkles/account/assets/userfrosting/js/pages/resend-verification.js new file mode 100755 index 0000000..5c3eaf8 --- /dev/null +++ b/main/app/sprinkles/account/assets/userfrosting/js/pages/resend-verification.js @@ -0,0 +1,19 @@ +/** + * Page-specific Javascript file. Should generally be included as a separate asset bundle in your page template. + * example: {{ assets.js('js/pages/sign-in-or-register') | raw }} + * + * This script depends on validation rules specified in pages/partials/page.js.twig. + * + * Target page: account/resend-verification + */ +$(document).ready(function() { + + // TODO: Process form + $("#request-verification-email").ufForm({ + validators: page.validators.resend_verification, + msgTarget: $("#alerts-page") + }).on("submitSuccess.ufForm", function() { + // Forward to login page on success + window.location.replace(site.uri.public + "/account/sign-in"); + }); +}); diff --git a/main/app/sprinkles/account/assets/userfrosting/js/pages/set-or-reset-password.js b/main/app/sprinkles/account/assets/userfrosting/js/pages/set-or-reset-password.js new file mode 100755 index 0000000..39cfd16 --- /dev/null +++ b/main/app/sprinkles/account/assets/userfrosting/js/pages/set-or-reset-password.js @@ -0,0 +1,19 @@ +/** + * Page-specific Javascript file. Should generally be included as a separate asset bundle in your page template. + * example: {{ assets.js('js/pages/sign-in-or-register') | raw }} + * + * This script depends on validation rules specified in pages/partials/page.js.twig. + * + * Target pages: account/set-password, account/reset-password + */ +$(document).ready(function() { + + $("#set-or-reset-password").ufForm({ + validators: page.validators.set_password, + msgTarget: $("#alerts-page") + }).on("submitSuccess.ufForm", function() { + // Forward to home page on success + // TODO: forward to landing/last page + window.location.replace(site.uri.public + "/account/sign-in"); + }); +}); diff --git a/main/app/sprinkles/account/assets/userfrosting/js/pages/sign-in.js b/main/app/sprinkles/account/assets/userfrosting/js/pages/sign-in.js new file mode 100755 index 0000000..40a8628 --- /dev/null +++ b/main/app/sprinkles/account/assets/userfrosting/js/pages/sign-in.js @@ -0,0 +1,39 @@ +/** + * Page-specific Javascript file. Should generally be included as a separate asset bundle in your page template. + * example: {{ assets.js('js/pages/sign-in-or-register') | raw }} + * + * This script depends on validation rules specified in pages/partials/page.js.twig. + * + * Target page: account/sign-in + */ +$(document).ready(function() { + /** + * If there is a redirect parameter in the query string, redirect to that page. + * Otherwise, if there is a UF-Redirect header, redirect to that page. + * Otherwise, redirect to the home page. + */ + function redirectOnLogin(jqXHR) { + var components = URI.parse(window.location.href); + var query = URI.parseQuery(components['query']); + + if (query && query['redirect']) { + // Strip leading slashes from redirect strings + var redirectString = site.uri.public + '/' + query['redirect'].replace(/^\/+/, ""); + // Strip excess trailing slashes for clean URLs. e.g. if redirect=%2F + redirectString = redirectString.replace(/\/+$/, "/"); + // Redirect + window.location.replace(redirectString); + } else if (jqXHR.getResponseHeader('UF-Redirect')) { + window.location.replace(jqXHR.getResponseHeader('UF-Redirect')); + } else { + window.location.replace(site.uri.public); + } + } + + $("#sign-in").ufForm({ + validators: page.validators.login, + msgTarget: $("#alerts-page") + }).on("submitSuccess.ufForm", function(event, data, textStatus, jqXHR) { + redirectOnLogin(jqXHR); + }); +}); diff --git a/main/app/sprinkles/account/bower.json b/main/app/sprinkles/account/bower.json new file mode 100755 index 0000000..8e7ef39 --- /dev/null +++ b/main/app/sprinkles/account/bower.json @@ -0,0 +1,28 @@ +{ + "name": "userfrosting-sprinkle-account", + "description": "Authentication and account management module for UserFrosting.", + "homepage": "https://github.com/userfrosting", + "license": "MIT", + "authors": [ + { + "name": "Alexander Weissman", + "homepage": "https://alexanderweissman.com" + }, + "ssnukala" + ], + "dependencies": {}, + "moduleType": [ + "node" + ], + "ignore": [ + "**/.*", + "node_modules", + "bower_components", + "assets/vendor", + "examples", + "demo-resources", + "demo", + "test", + "tests" + ] +} diff --git a/main/app/sprinkles/account/composer.json b/main/app/sprinkles/account/composer.json new file mode 100755 index 0000000..fa2e178 --- /dev/null +++ b/main/app/sprinkles/account/composer.json @@ -0,0 +1,24 @@ +{ + "name": "userfrosting/sprinkle-account", + "type": "userfrosting-sprinkle", + "description": "Authentication and account management module for UserFrosting.", + "keywords": ["php user management", "usercake", "bootstrap"], + "homepage": "https://github.com/userfrosting/UserFrosting", + "license" : "MIT", + "authors" : [ + { + "name": "Alexander Weissman", + "homepage": "https://alexanderweissman.com" + } + ], + "require": { + "birke/rememberme" : "^2.0", + "nikic/php-parser" : "^1", + "php": ">=5.6" + }, + "autoload": { + "psr-4": { + "UserFrosting\\Sprinkle\\Account\\": "src/" + } + } +} diff --git a/main/app/sprinkles/account/config/default.php b/main/app/sprinkles/account/config/default.php new file mode 100755 index 0000000..e154643 --- /dev/null +++ b/main/app/sprinkles/account/config/default.php @@ -0,0 +1,79 @@ + [ + 'auth' => false + ], + // configuration for the 'password reset' feature + 'password_reset' => [ + 'algorithm' => 'sha512', + 'timeouts' => [ + 'create' => 86400, + 'reset' => 10800 + ] + ], + // See https://github.com/gbirke/rememberme for an explanation of these settings + 'remember_me' => [ + 'cookie' => [ + 'name' => 'rememberme' + ], + 'expire_time' => 604800, + 'session' => [ + 'path' => '/' + ], + 'table' => [ + 'tableName' => 'persistences', + 'credentialColumn' => 'user_id', + 'tokenColumn' => 'token', + 'persistentTokenColumn' => 'persistent_token', + 'expiresColumn' => 'expires_at' + ] + ], + 'reserved_user_ids' => [ + 'guest' => -1, + 'master' => 1 + ], + 'session' => [ + // The keys used in the session to store info about authenticated users + 'keys' => [ + 'current_user_id' => 'account.current_user_id', // the key to use for storing the authenticated user's id + 'captcha' => 'account.captcha' // Key used to store a captcha hash during captcha verification + ] + ], + // "Site" settings that are automatically passed to Twig + 'site' => [ + 'login' => [ + 'enable_email' => true + ], + 'registration' => [ + 'enabled' => true, + 'captcha' => true, + 'require_email_verification' => true, + 'user_defaults' => [ + 'locale' => 'en_US', + 'group' => 'terran', + // Default roles for newly registered users + 'roles' => [ + 'user' => true + ] + ] + ] + ], + 'throttles' => [ + 'check_username_request' => null, + 'password_reset_request' => null, + 'registration_attempt' => null, + 'sign_in_attempt' => null, + 'verification_request' => null + ], + // configuration for the 'email verification' feature + 'verification' => [ + 'algorithm' => 'sha512', + 'timeout' => 10800 + ] + ]; diff --git a/main/app/sprinkles/account/config/production.php b/main/app/sprinkles/account/config/production.php new file mode 100755 index 0000000..b7c3288 --- /dev/null +++ b/main/app/sprinkles/account/config/production.php @@ -0,0 +1,67 @@ + [ + 'check_username_request' => [ + 'method' => 'ip', + 'interval' => 3600, + 'delays' => [ + 40 => 1000 + ] + ], + 'password_reset_request' => [ + 'method' => 'ip', + 'interval' => 3600, + 'delays' => [ + 2 => 5, + 3 => 10, + 4 => 20, + 5 => 40, + 6 => 80, + 7 => 600 + ] + ], + 'registration_attempt' => [ + 'method' => 'ip', + 'interval' => 3600, + 'delays' => [ + 2 => 5, + 3 => 10, + 4 => 20, + 5 => 40, + 6 => 80, + 7 => 600 + ] + ], + 'sign_in_attempt' => [ + 'method' => 'ip', + 'interval' => 3600, + 'delays' => [ + 4 => 5, + 5 => 10, + 6 => 20, + 7 => 40, + 8 => 80, + 9 => 600 + ] + ], + 'verification_request' => [ + 'method' => 'ip', + 'interval' => 3600, + 'delays' => [ + 2 => 5, + 3 => 10, + 4 => 20, + 5 => 40, + 6 => 80, + 7 => 600 + ] + ] + ] + ]; diff --git a/main/app/sprinkles/account/factories/Permissions.php b/main/app/sprinkles/account/factories/Permissions.php new file mode 100755 index 0000000..591f5fd --- /dev/null +++ b/main/app/sprinkles/account/factories/Permissions.php @@ -0,0 +1,19 @@ +define('UserFrosting\Sprinkle\Account\Database\Models\Permission')->setDefinitions([ + 'slug' => Faker::word(), + 'name' => Faker::word(), + 'description' => Faker::paragraph(), + 'conditions' => Faker::word() +]); diff --git a/main/app/sprinkles/account/factories/Roles.php b/main/app/sprinkles/account/factories/Roles.php new file mode 100755 index 0000000..cdbb5a3 --- /dev/null +++ b/main/app/sprinkles/account/factories/Roles.php @@ -0,0 +1,18 @@ +define('UserFrosting\Sprinkle\Account\Database\Models\Role')->setDefinitions([ + 'slug' => Faker::unique()->word(), + 'name' => Faker::word(), + 'description' => Faker::paragraph() +]); diff --git a/main/app/sprinkles/account/factories/Users.php b/main/app/sprinkles/account/factories/Users.php new file mode 100755 index 0000000..7390c44 --- /dev/null +++ b/main/app/sprinkles/account/factories/Users.php @@ -0,0 +1,23 @@ +define('UserFrosting\Sprinkle\Account\Database\Models\User')->setDefinitions([ + 'user_name' => Faker::unique()->firstNameMale(), + 'first_name' => Faker::firstNameMale(), + 'last_name' => Faker::firstNameMale(), + 'email' => Faker::unique()->email(), + 'locale' => 'en_US', + 'flag_verified' => 1, + 'flag_enabled' => 1, + 'password' => Faker::password() +]); diff --git a/main/app/sprinkles/account/locale/ar/messages.php b/main/app/sprinkles/account/locale/ar/messages.php new file mode 100755 index 0000000..7203904 --- /dev/null +++ b/main/app/sprinkles/account/locale/ar/messages.php @@ -0,0 +1,176 @@ + [ + "@TRANSLATION" => "الحساب", + + "ACCESS_DENIED" => "يبدو أنك لا تملك صلاحية للقيام بذلك", + + "DISABLED" => "هذا الحساب معطل يمكنك الاتصال بنا للحصول على مزيد من المعلومات", + + "EMAIL_UPDATED" => "تم تجديد البريد الإلكتروني بالحساب", + + "INVALID" => "هذا الحساب غير موجود قد تم حذفه يمكنك الاتصا بنا للحصول على مزيد من المعلومات", + + "MASTER_NOT_EXISTS" => "لا يمكنك تسجيل حساب جديد حتى تم إنشاء الحساب الرئيسي", + "MY" => "حسابي", + + "SESSION_COMPROMISED" => [ + "@TRANSLATION" => "تم اختراق جلسنك يجب عليك الخروج على كافة الأجهزة، ثم تسجيل الدخول مرة أخرى والتأكد من أن المعلومات الخاصة بك لم يعبث بها", + "TITLE" => "من الممكن أن حسابك قد اخترق", + "TEXT" => "ربما استخدم شخص معلومات التسجيل الدخول للدخول إلى هذه الصفحة. لسلامتك، تم انتهاء جميع الجلسات يرجا التسجيل مرة اخرى وتحقق من حسابك بسبب النشاط الغريب قد ترغب في تغيير كلمة المرور" + ], + + "SESSION_EXPIRED" => "انتهت جلستك تستطيع تسجيل الدخول مرة أخرى", + + "SETTINGS" => [ + "@TRANSLATION" => "إعدادات الحساب", + "DESCRIPTION" => "غير إعدادات حسابك، بما في ذلك البريد الإلكتروني، واسم وكلمة المرور +", + "UPDATED" => "تم تجديد إعدادات الحساب" + ], + + "TOOLS" => "أدوات الحساب", + + "UNVERIFIED" => "لم يتم التحقق من حسابك بعد افحص في رسائل البريد الإلكتروني و ملف البريد المزعج للحصول على تعليمات تفعيل الحساب", + + "VERIFICATION" => [ + "NEW_LINK_SENT" => "لقد أرسلنا رابط جديدا لتحقق عبر البريد الإلكتروني إلى {{email}} افحص في رسائل البريد الإلكتروني و ملف البريد المزعج", + "RESEND" => "إعادة ارسال بريد التحقق", + "COMPLETE" => "لقد تم التحقق من حسابك بنجاح يمكنك الآن تسجيل الدخول", + "EMAIL" => "ادخل عنوان البريد الإلكتروني الذي استخدمته للتسجيل، و سوف نرسل البريد الإلكتروني لتحقق مرة أخرى", + "PAGE" => "إعادة إرسال البريد الإلكتروني التحقق من حسابك الجديد", + "SEND" => "ارسل رابط للتحقق عبر البريد الالكتروني", + "TOKEN_NOT_FOUND" => "رمز التحقق غير موجود أو تم تحقق الحساب من قبل", + ] + ], + + "EMAIL" => [ + "INVALID" => "لا يوجد حساب ل {{email}}", + "IN_USE" => "البريد الإلكتروني {{email}} قيد الاستخدام" + ], + + "FIRST_NAME" => "الاسم الاول", + + "HEADER_MESSAGE_ROOT" => "تسجيل الدخول باسم المستخدم ROOT", + + "LAST_NAME" => "اسم العائلة", + + "LOCALEACCOUNT" => "اللغة التي تستخدم لحسابك", + + "LOGIN" => [ + "@TRANSLATION" => "تسجيل الدخول", + + "ALREADY_COMPLETE" => "انت بالفعل داخل", + "SOCIAL" => "أو الدخول مع", + "REQUIRED" => "عذرا، يجب عليك تسجيل الدخول للوصول إلى هذا المكان" + ], + + "LOGOUT" => "تسجيل الخروج", + + "NAME" => "اسم", + + "PAGE" => [ + "LOGIN" => [ + "DESCRIPTION" => "سجل الدخول إلى حسابك في {{site_name}} أو سجيل للحصول على حساب جديد", + "SUBTITLE" => "التسجيل مجانا أو قم بتسجيل الدخول باستخدام حساب موجود", + "TITLE" => "هيا نبدأ", + ] + ], + + "PASSWORD" => [ + "@TRANSLATION" => "كلمه المرور", + + "BETWEEN" => "ما بين {{min}}-{{max}} حروف", + + "CONFIRM" => "تأكيد كلمة المرور", + "CONFIRM_CURRENT" => "تأكيد كلمه المرور الحالي", + "CONFIRM_NEW" => "تأكيد كلمة المرور الجديدة", + "CONFIRM_NEW_EXPLAIN" => "إعادة إدخال كلمة المرور الجديدة", + "CONFIRM_NEW_HELP" => "لازم إذا كان المطلوب اختيار كلمة مرور جديدة", + "CURRENT" => "كلمة المرور الحالية", + "CURRENT_EXPLAIN" => "يجب عليك تأكيد كلمة المرور الحالية لإجراء التغييرات", + + "FORGOTTEN" => "كلمه المرور منسية", + "FORGET" => [ + "@TRANSLATION" => "لقد نسيت كلمة المرور", + + "COULD_NOT_UPDATE" => "لا يمكن تحديث كلمة المرور", + "EMAIL" => "ادخل عنوان البريد الإلكتروني الذي استخدمته للتسجيل وسوف نرسل تعليمات لإعادة تعيين كلمة المرور", + "EMAIL_SEND" => "أرسل رابط تعيين كلمة المرور عبر البريد الالكتروني", + "INVALID" => "لم يتم العثور على إعادة تعيين كلمة المرور، أو انتهت صلاحية رابط حاول إعادة تقديم طلبك", + "PAGE" => "الحصول على رابط لإعادة تعيين كلمة المرور", + "REQUEST_CANNED" => "إلغاء طلب كلمة المرور", + "REQUEST_SENT" => "إذا تطابق البريد الإلكتروني {{email}} حسابا في نظامنا، فسيتم إرسال رابط إعادة تعيين كلمة المرور إلى {{email}}." + ], + + "RESET" => [ + "@TRANSLATION" => "إعادة تعيين كلمة المرور", + "CHOOSE" => "اختيار كلمة مرور جديدة للتواصل", + "PAGE" => "اختيار كلمة مرور جديدة لحسابك", + "SEND" => "تعيين كلمة المرور الجديدة وتسجيل الدخول" + ], + + "HASH_FAILED" => "فشلت التجزئة كلمة المرور يرجى الاتصال بمسؤول الموقع", + "INVALID" => "كلمة مرور الحالية لا تتطابق مع ما لدينا", + "NEW" => "كلمة مرور الجديدة", + "NOTHING_TO_UPDATE" => "لا يمكنك تحديث مع نفس كلمة مرور", + "UPDATED" => "جدد كلمة مرور", + + "CREATE" => [ + "@TRANSLATION" => "إنشاء كلمة مرور", + "PAGE" => "اختر كلمة مرور لحسابك الجديد", + "SET" => "تعيين كلمة المرور وتسجيل الدخول" + ] + ], + + "REGISTER" => "تسجيل", + "REGISTER_ME" => "سجلني", + "SIGN_IN_HERE" => "هل لديك حساب؟ تسجيل الدخول هنا", + + "REGISTRATION" => [ + "BROKEN" => "نحن آسفون، هناك مشكلة مع عملية تسجيل الحساب يرجى الاتصال بنا مباشرة للحصول على المساعدة", + "COMPLETE_TYPE1" => "لقد سجلت بنجاح يمكنك الآن تسجيل الدخول", + "COMPLETE_TYPE2" => "لقد سجلت بنجاح سوف تتلقى قريبا رسالة التحقق تحتوي على رابط لتفعيل حسابك لن تكون قادرا على تسجيل الدخول حتى الانتهاء من هذه الخطوة", + "DISABLED" => "عذرا، لقد تم تعطيل تسجيل اي حساب", + "LOGOUT" => "لا يمكنك التسجيل للحصول على حساب أثناء تسجيل الدخول", + "WELCOME" => "التسجيل سريع وبسيط" + ], + + "RATE_LIMIT_EXCEEDED" => "تم تجاوز الحد عددا لهذا الإجراء يجب الانتظار {{delay}} ثواني قبل القيام بمحاولة أخرى", + "REMEMBER_ME" => "تذكرنى", + "REMEMBER_ME_ON_COMPUTER" => "تذكرني على هذا الحاسوب (غير مستحسن للحواسب العامة)", + + "SIGNIN" => "تسجيل الدخول", + "SIGNIN_OR_REGISTER" => "تسجيل الدخول أو التسجيل", + "SIGNUP" => "تسجيل", + + "TOS" => "الأحكام والشروط", + "TOS_AGREEMENT" => "من خلال تسجيل حساب جديد في {{site_title}}, انت تقبل الأحكام والشروط", + "TOS_FOR" => "الأحكام والشروط ل {{title}}", + + "USERNAME" => [ + "@TRANSLATION" => "اسم المستخدم", + + "CHOOSE" => "اختيار اسم مستخدم فريد", + "INVALID" => "اسم المستخدم غير صالح", + "IN_USE" => "اسم المستخدم {{user_name}} قيد الاستخدام" + ], + + "USER_ID_INVALID" => "عدم وجود هوية المستخدم المطلوب", + "USER_OR_EMAIL_INVALID" => "اسم المستخدم أو عنوان البريد الإلكتروني غير صالح", + "USER_OR_PASS_INVALID" => "اسم المستخدم أو كلمة المرور غير صالحة", + + "WELCOME" => "مرحبا بعودتك, {{first_name}}" +]; diff --git a/main/app/sprinkles/account/locale/ar/validate.php b/main/app/sprinkles/account/locale/ar/validate.php new file mode 100755 index 0000000..37693fb --- /dev/null +++ b/main/app/sprinkles/account/locale/ar/validate.php @@ -0,0 +1,18 @@ + [ + "PASSWORD_MISMATCH" => "يجب أن تكون كلمة المرور وكلمة المرور التأكيدية نفس" + ] +]; diff --git a/main/app/sprinkles/account/locale/de_DE/messages.php b/main/app/sprinkles/account/locale/de_DE/messages.php new file mode 100755 index 0000000..b331552 --- /dev/null +++ b/main/app/sprinkles/account/locale/de_DE/messages.php @@ -0,0 +1,188 @@ + [ + "@TRANSLATION" => "Konto", + + "ACCESS_DENIED" => "Hmm, sieht aus als hätten Sie keine Berechtigung, um dies zu tun.", + + "DISABLED" => "Dieses Konto wurde deaktiviert. Bitte Kontaktieren Sie uns für weitere Informationen.", + + "EMAIL_UPDATED" => "E-Mail-Adresse aktualisiert.", + + "INVALID" => "Dieses Konto existiert nicht. Es wurde möglicherweise gelöscht. Bitte kontaktieren Sie uns für weitere Informationen.", + + "MASTER_NOT_EXISTS" => "Sie können kein neues Konto anlegen solange kein Root-Konto angelegt wurde!", + "MY" => "Mein Konto", + + "SESSION_COMPROMISED" => [ + "@TRANSLATION" => "Ihre Sitzung wurde beeinträchtigt. Sie sollten sich auf allen Geräten abmelden, sich dann wieder anmelden und sicherstellen, dass Ihre Daten nicht manipuliert wurden.", + "TITLE" => "Ihr Konto wurde möglicherweise beeinträchtigt", + "TEXT" => "Möglicherweise ist es jemandem gelungen, Ihren Zugang zu dieser Seite zu übernehmen. Aus Sicherheitsgründen wurden Sie überall abgemeldet. Bitte melden Sie sich neu an und untersuchen Sie das Konto nach verdächtigen Aktivitäten. Außerdem sollten Sie Ihr Passwort ändern." + ], + "SESSION_EXPIRED" => "Ihre Sitzung ist abgelaufen. Bitte melden Sie sich erneut an.", + + "SETTINGS" => [ + "@TRANSLATION" => "Kontoeinstellungen", + "DESCRIPTION" => "Aktualisieren Sie Ihre Kontoeinstellungen, einschließlich E-Mail, Name und Passwort.", + "UPDATED" => "Kontoeinstellungen aktualisiert" + ], + + "TOOLS" => "Konto-Werkzeuge", + + "UNVERIFIED" => "Ihr Konto wurde noch nicht bestätigt. Überprüfen Sie Ihr E-Mails/Spam-Ordner für die Konto-Aktivierungsanleitung.", + + "VERIFICATION" => [ + "NEW_LINK_SENT" => "Wir haben einen neuen Bestätigungslink an {{email}} gesendet. Überprüfen Sie Ihr E-Mail/Spam-Ordner oder versuchen Sie es später noch einmal.", + "RESEND" => "Bestätigungsmail erneut senden", + "COMPLETE" => "Sie haben Ihr Konto erfolgreich Verifiziert. Sie können sich jetzt anmelden.", + "EMAIL" => "Bitte geben Sie die E-Mail-Adresse ein, mit der Sie sich registriert haben, Überprüfen Sie Ihr E-Mails/Spam-Ordner für die Bestätigungs-E-Mail.", + "PAGE" => "Senden Sie die Bestätigungs-E-Mail erneut für Ihr neues Konto.", + "SEND" => "Bestätigungslink erneut per E-Mail zusenden", + "TOKEN_NOT_FOUND" => "Verifizierungstoken existiert nicht / Konto wurde bereits verifiziert" + ] + ], + + "EMAIL" => [ + "INVALID" => "Es gibt kein Konto für {{email}}.", + "IN_USE" => "Die E-Mail Adresse {{email}} wird bereits verwendet.", + "VERIFICATION_REQUIRED" => "E-Mail (Bestätigung benötigt - Benutzen Sie eine echte E-Mail Adresse!)" + ], + + "EMAIL_OR_USERNAME" => "Benutzername oder E-mail Adresse", + + "FIRST_NAME" => "Vorname", + + "HEADER_MESSAGE_ROOT" => "Sie sind als Root-Benutzer angemeldet.", + + "LAST_NAME" => "Nachname", + + "LOCALE" => [ + "ACCOUNT" => "Die Sprache und das Gebietsschema für Ihr Konto", + "INVALID" => "{{locale}} ist kein gültiges Gebietsschema." + ], + + "LOGIN" => [ + "@TRANSLATION" => "Anmelden", + "ALREADY_COMPLETE" => "Sie sind bereits eingeloggt!", + "SOCIAL" => "Oder loggen Sie sich ein mit", + "REQUIRED" => "Sorry, Sie müssen angemeldet sein. Um auf diese Ressource zugreifen zu können." + ], + + "LOGOUT" => "Ausloggen", + + "NAME" => "Name", + + "NAME_AND_EMAIL" => "Name und E-Mail", + + "PAGE" => [ + "LOGIN" => [ + "DESCRIPTION" => "Melden Sie sich in Ihr {{site_name}} Konto an oder registrieren Sie sich für ein neues Konto.", + "SUBTITLE" => "Registrieren Sie sich kostenlos oder melden Sie sich mit einem bestehenden Konto an.", + "TITLE" => "Lass uns anfangen!" + ] + ], + + "PASSWORD" => [ + "@TRANSLATION" => "Passwort", + + "BETWEEN" => "Zwischen {{min}}-{{max}} Zeichen", + + "CONFIRM" => "Bestätige das Passwort", + "CONFIRM_CURRENT" => "Bitte bestätige dein jetziges Passwort", + "CONFIRM_NEW" => "Neues Passwort bestätigen", + "CONFIRM_NEW_EXPLAIN" => "Geben Sie Ihr neues Passwort erneut ein", + "CONFIRM_NEW_HELP" => "Erforderlich, wenn Sie ein neues Passwort wählen", + "CREATE" => [ + "@TRANSLATION" => "Passwort setzen", + "PAGE" => "Setzen Sie ein Passwort für den Account.", + "SET" => "Passwort setzen und anmelden" + ], + "CURRENT" => "Aktuelles Passwort", + "CURRENT_EXPLAIN" => "Sie müssen Ihr aktuelles Passwort bestätigen, um Änderungen vorzunehmen", + + "FORGOTTEN" => "Passwort vergessen", + "FORGET" => [ + "@TRANSLATION" => "Ich habe mein Passwort vergessen", + + "COULD_NOT_UPDATE" => "Das Passwort konnte nicht aktualisiert werden.", + "EMAIL" => "Bitte geben Sie die E-Mail-Adresse ein, mit der Sie sich registriert haben. Ein Link mit der Anweisungen zum Zurücksetzen Ihres Passworts wird Ihnen per E-Mail zugeschickt.", + "EMAIL_SEND" => "Neue Passwort zurücksetzen E-Mail senden", + "INVALID" => "Diese Anforderung zum Zurücksetzen des Passworts wurde nicht gefunden oder ist abgelaufen.Bitte versuchen Sie Ihre Anfrage erneut einzureichen.", + "PAGE" => "Holen Sie sich einen Link, um Ihr Passwort zurückzusetzen.", + "REQUEST_CANNED" => "Verlorene Passwortanforderung abgebrochen.", + "REQUEST_SENT" => "Wenn die E-Mail {{email}} mit einem Account in unserem System übereinstimmt, wird ein Passwort-Reset-Link an {{email}} gesendet." + ], + + "HASH_FAILED" => "Passwort Hashing fehlgeschlagen. Bitte kontaktieren Sie einen Administrator.", + "INVALID" => "Das aktuelle Passwort stimmt nicht mit dem Datensatz überein", + "NEW" => "Neues Passwort", + "NOTHING_TO_UPDATE" => "Sie können nicht das gleiche Passwort zum Aktualisieren verwenden", + + "RESET" => [ + "@TRANSLATION" => "Passwort zurücksetzen", + "CHOOSE" => "Bitte wählen Sie ein neues Passwort, um fortzufahren.", + "PAGE" => "Wählen Sie ein neues Passwort für Ihr Konto.", + "SEND" => "Neues Passwort festlegen und anmelden" + ], + + "UPDATED" => "Konto Passwort aktualisiert" + ], + + "PROFILE" => [ + "SETTINGS" => "Profileinstellungen", + "UPDATED" => "Profileinstellungen aktualisiert" + ], + + "RATE_LIMIT_EXCEEDED" => "Die grenze für diese Maßnahme wurde überschritten. Sie müssen weitere {{delay}} Sekunden warten, bevor Sie einen weiteren Versuch machen dürfen.", + + "REGISTER" => "Registrieren", + "REGISTER_ME" => "Melden Sie mich an", + "REGISTRATION" => [ + "BROKEN" => "Es tut uns leid, es gibt ein Problem mit unserer Registrierung. Bitte kontaktieren Sie uns direkt für Hilfe.", + "COMPLETE_TYPE1" => "Sie haben sich erfolgreich registriert. Sie können sich jetzt anmelden.", + "COMPLETE_TYPE2" => "Sie haben sich erfolgreich registriert. Sie erhalten in Kürze eine Bestätigungs-E-Mail mit einem Link zur Aktivierung Ihres Kontos. Sie können sich nicht anmelden, bis Sie diesen Schritt abgeschlossen haben.", + "DISABLED" => "Es tut uns leid, Die Registrierung des Kontos ist deaktiviert.", + "LOGOUT" => "Es tut uns leid, Sie können kein neues Konto registrieren, während Sie angemeldet sind. Bitte melden Sie sich zuerst ab.", + "WELCOME" => "Die Registrierung ist schnell und einfach." + ], + "REMEMBER_ME" => "Erinnere dich an mich!", + "REMEMBER_ME_ON_COMPUTER" => "Erinnere dich an mich auf diesem Computer (nicht für öffentliche Computer empfohlen)", + + "SIGN_IN_HERE" => "Sie haben bereits einen Account? Melden Sie sich hier an.", + "SIGNIN" => "Anmelden", + "SIGNIN_OR_REGISTER" => "Anmelden oder registrieren", + "SIGNUP" => "Anmelden", + + "TOS" => "Geschäftsbedingungen", + "TOS_AGREEMENT" => "Durch die Registrierung eines Kontos auf {{site_title}} akzeptieren Sie die Bedingungen .", + "TOS_FOR" => "Allgemeine Geschäftsbedingungen für {{title}}", + + "USERNAME" => [ + "@TRANSLATION" => "Benutzername", + + "CHOOSE" => "Wählen Sie einen eindeutigen Benutzernamen", + "INVALID" => "Ungültiger Benutzername", + "IN_USE" => "Benutzername {{user_name}} wird bereits verwendet.", + "NOT_AVAILABLE" => "Benutzername {{user_name}} ist nicht verfügbar. Wähle einen anderen Namen, der klicken Sie auf 'vorschlagen'." + ], + + "USER_ID_INVALID" => "Die angeforderte Benutzer-ID existiert nicht.", + "USER_OR_EMAIL_INVALID" => "Benutzername oder E-Mail-Adresse ist ungültig.", + "USER_OR_PASS_INVALID" => "Benutzername oder Passwort ist ungültig.", + + "WELCOME" => "Willkommen zurück, {{first_name}}" +]; diff --git a/main/app/sprinkles/account/locale/de_DE/validate.php b/main/app/sprinkles/account/locale/de_DE/validate.php new file mode 100755 index 0000000..30cf98b --- /dev/null +++ b/main/app/sprinkles/account/locale/de_DE/validate.php @@ -0,0 +1,21 @@ + [ + "PASSWORD_MISMATCH" => "Ihr Passwort und das Bestätigungspasswort müssen übereinstimmen.", + "USERNAME" => "Benutzernamen dürfen nur aus Kleinbuchstaben, Zahlen, '.', '-' und '_' bestehen." + ] +]; diff --git a/main/app/sprinkles/account/locale/en_US/messages.php b/main/app/sprinkles/account/locale/en_US/messages.php new file mode 100755 index 0000000..17d7582 --- /dev/null +++ b/main/app/sprinkles/account/locale/en_US/messages.php @@ -0,0 +1,183 @@ + [ + "@TRANSLATION" => "Account", + + "ACCESS_DENIED" => "Hmm, looks like you don't have permission to do that.", + + "DISABLED" => "This account has been disabled. Please contact us for more information.", + + "EMAIL_UPDATED" => "Account email updated", + + "INVALID" => "This account does not exist. It may have been deleted. Please contact us for more information.", + + "MASTER_NOT_EXISTS" => "You cannot register an account until the master account has been created!", + "MY" => "My Account", + + "SESSION_COMPROMISED" => [ + "@TRANSLATION" => "Your session has been compromised. You should log out on all devices, then log back in and make sure that your data has not been tampered with.", + "TITLE" => "Your account may have been compromised", + "TEXT" => "Someone may have used your login information to acccess this page. For your safety, all sessions were logged out. Please log in and check your account for suspicious activity. You may also wish to change your password." + ], + "SESSION_EXPIRED" => "Your session has expired. Please sign in again.", + + "SETTINGS" => [ + "@TRANSLATION" => "Account settings", + "DESCRIPTION" => "Update your account settings, including email, name, and password.", + "UPDATED" => "Account settings updated" + ], + + "TOOLS" => "Account tools", + + "UNVERIFIED" => "Your account has not yet been verified. Check your emails / spam folder for account activation instructions.", + + "VERIFICATION" => [ + "NEW_LINK_SENT" => "We have emailed a new verification link to {{email}}. Please check your inbox and spam folders for this email.", + "RESEND" => "Resend verification email", + "COMPLETE" => "You have successfully verified your account. You can now login.", + "EMAIL" => "Please enter the email address you used to sign up, and your verification email will be resent.", + "PAGE" => "Resend the verification email for your new account.", + "SEND" => "Email the verification link for my account", + "TOKEN_NOT_FOUND" => "Verification token does not exist / Account is already verified", + ] + ], + + "EMAIL" => [ + "INVALID" => "There is no account for {{email}}.", + "IN_USE" => "Email {{email}} is already in use.", + "VERIFICATION_REQUIRED" => "Email (verification required - use a real address!)" + ], + + "EMAIL_OR_USERNAME" => "Username or email address", + + "FIRST_NAME" => "First name", + + "HEADER_MESSAGE_ROOT" => "YOU ARE SIGNED IN AS THE ROOT USER", + + "LAST_NAME" => "Last name", + "LOCALE" => [ + "ACCOUNT" => "The language and locale to use for your account", + "INVALID" => "{{locale}} is not a valid locale." + ], + "LOGIN" => [ + "@TRANSLATION" => "Login", + "ALREADY_COMPLETE" => "You are already logged in!", + "SOCIAL" => "Or login with", + "REQUIRED" => "Sorry, you must be logged in to access this resource." + ], + "LOGOUT" => "Logout", + + "NAME" => "Name", + + "NAME_AND_EMAIL" => "Name and email", + + "PAGE" => [ + "LOGIN" => [ + "DESCRIPTION" => "Sign in to your {{site_name}} account, or register for a new account.", + "SUBTITLE" => "Register for free, or sign in with an existing account.", + "TITLE" => "Let's get started!", + ] + ], + + "PASSWORD" => [ + "@TRANSLATION" => "Password", + + "BETWEEN" => "Between {{min}}-{{max}} characters", + + "CONFIRM" => "Confirm password", + "CONFIRM_CURRENT" => "Please confirm your current password", + "CONFIRM_NEW" => "Confirm New Password", + "CONFIRM_NEW_EXPLAIN" => "Re-enter your new password", + "CONFIRM_NEW_HELP" => "Required only if selecting a new password", + "CREATE" => [ + "@TRANSLATION" => "Create Password", + "PAGE" => "Choose a password for your new account.", + "SET" => "Set Password and Sign In" + ], + "CURRENT" => "Current Password", + "CURRENT_EXPLAIN" => "You must confirm your current password to make changes", + + "FORGOTTEN" => "Forgotten Password", + "FORGET" => [ + "@TRANSLATION" => "I forgot my password", + + "COULD_NOT_UPDATE" => "Couldn't update password.", + "EMAIL" => "Please enter the email address you used to sign up. A link with instructions to reset your password will be emailed to you.", + "EMAIL_SEND" => "Email Password Reset Link", + "INVALID" => "This password reset request could not be found, or has expired. Please try resubmitting your request.", + "PAGE" => "Get a link to reset your password.", + "REQUEST_CANNED" => "Lost password request cancelled.", + "REQUEST_SENT" => "If the email {{email}} matches an account in our system, a password reset link will be sent to {{email}}." + ], + + "HASH_FAILED" => "Password hashing failed. Please contact a site administrator.", + "INVALID" => "Current password doesn't match the one we have on record", + "NEW" => "New Password", + "NOTHING_TO_UPDATE" => "You cannot update with the same password", + + "RESET" => [ + "@TRANSLATION" => "Reset Password", + "CHOOSE" => "Please choose a new password to continue.", + "PAGE" => "Choose a new password for your account.", + "SEND" => "Set New Password and Sign In" + ], + + "UPDATED" => "Account password updated" + ], + + "PROFILE" => [ + "SETTINGS" => "Profile settings", + "UPDATED" => "Profile settings updated" + ], + + "RATE_LIMIT_EXCEEDED" => "The rate limit for this action has been exceeded. You must wait another {{delay}} seconds before you will be allowed to make another attempt.", + + "REGISTER" => "Register", + "REGISTER_ME" => "Sign me up", + "REGISTRATION" => [ + "BROKEN" => "We're sorry, there is a problem with our account registration process. Please contact us directly for assistance.", + "COMPLETE_TYPE1" => "You have successfully registered. You can now sign in.", + "COMPLETE_TYPE2" => "You have successfully registered. A link to activate your account has been sent to {{email}}. You will not be able to sign in until you complete this step.", + "DISABLED" => "We're sorry, account registration has been disabled.", + "LOGOUT" => "I'm sorry, you cannot register for an account while logged in. Please log out first.", + "WELCOME" => "Registration is fast and simple." + ], + "REMEMBER_ME" => "Keep me signed in", + "REMEMBER_ME_ON_COMPUTER" => "Remember me on this computer (not recommended for public computers)", + + "SIGN_IN_HERE" => "Already have an account? Sign in here.", + "SIGNIN" => "Sign in", + "SIGNIN_OR_REGISTER" => "Sign in or register", + "SIGNUP" => "Sign Up", + + "TOS" => "Terms and Conditions", + "TOS_AGREEMENT" => "By registering an account with {{site_title}}, you accept the terms and conditions.", + "TOS_FOR" => "Terms and Conditions for {{title}}", + + "USERNAME" => [ + "@TRANSLATION" => "Username", + + "CHOOSE" => "Choose a unique username", + "INVALID" => "Invalid username", + "IN_USE" => "Username {{user_name}} is already in use.", + "NOT_AVAILABLE" => "Username {{user_name}} is not available. Choose a different name, or click 'suggest'." + ], + + "USER_ID_INVALID" => "The requested user id does not exist.", + "USER_OR_EMAIL_INVALID" => "Username or email address is invalid.", + "USER_OR_PASS_INVALID" => "User not found or password is invalid.", + + "WELCOME" => "Welcome back, {{first_name}}" +]; diff --git a/main/app/sprinkles/account/locale/en_US/validate.php b/main/app/sprinkles/account/locale/en_US/validate.php new file mode 100755 index 0000000..00c0aef --- /dev/null +++ b/main/app/sprinkles/account/locale/en_US/validate.php @@ -0,0 +1,19 @@ + [ + "PASSWORD_MISMATCH" => "Your password and confirmation password must match.", + "USERNAME" => "Username may consist only of lowercase letters, numbers, '.', '-', and '_'." + ] +]; diff --git a/main/app/sprinkles/account/locale/es_ES/messages.php b/main/app/sprinkles/account/locale/es_ES/messages.php new file mode 100755 index 0000000..aa8b8ed --- /dev/null +++ b/main/app/sprinkles/account/locale/es_ES/messages.php @@ -0,0 +1,189 @@ + [ + "@TRANSLATION" => "Perfil", + + "ACCESS_DENIED" => "Hmm, parece que no tienes permiso para hacer eso.", + + "DISABLED" => "Esta cuenta se ha inhabilitado. Por favor contáctenos para más información.", + + "EMAIL_UPDATED" => "Correo electrónico de la cuenta actualizado", + + "INVALID" => "Esta cuenta no existe. Puede haber sido eliminado. Por favor contáctenos para más información.", + + "MASTER_NOT_EXISTS" => "No puede registrar una cuenta hasta que se haya creado la cuenta principal.", + "MY" => "Mi Perfil", + + "SESSION_COMPROMISED" => [ + "@TRANSLATION" => "Su sesión ha sido comprometida. Debe desconectarse de todos los dispositivos y, a continuación, volver a iniciar sesión y asegurarse de que sus datos no han sido manipulados.", + "TITLE" => "Es posible que su cuenta se haya visto comprometida.", + "TEXT" => "Alguien puede haber utilizado su información de acceso para acceder a esta página. Para su seguridad, todas las sesiones se cerraron. ingrese y compruebe si su actividad es sospechosa en su cuenta. También puede cambiar su contraseña." + ], + "SESSION_EXPIRED" => "Su sesión ha caducado. Inicie sesión nuevamente.", + + "SETTINGS" => [ + "@TRANSLATION" => "Configuraciones de la cuenta", + "DESCRIPTION" => "Actualice la configuración de su cuenta, incluido el correo electrónico, el nombre y la contraseña.", + "UPDATED" => "Configuración de la cuenta actualizada" + ], + + "TOOLS" => "Herramientas de la cuenta", + + "UNVERIFIED" => "Tu cuenta aún no se ha verificado. Revise sus correos electrónicos / carpeta de spam para obtener instrucciones sobre la activación de la cuenta.", + + "VERIFICATION" => [ + "NEW_LINK_SENT" => "Hemos enviado por correo electrónico un nuevo enlace de verificación a {{email}}. Comprueba tu bandeja de entrada y las carpetas de spam para este correo electrónico.", + "RESEND" => "Reenviar correo electrónico de verificación", + "COMPLETE" => "Ha verificado correctamente su cuenta. Ahora puede iniciar sesión.", + "EMAIL" => "Ingrese la dirección de correo electrónico que utilizó para registrarse y su correo electrónico de verificación será enviado de nuevo.", + "PAGE" => "Vuelva a enviar el correo electrónico de verificación de su nueva cuenta.", + "SEND" => "Reenviar correo de verificación", + "TOKEN_NOT_FOUND" => "El token de verificación no existe / La cuenta ya está verificada", + ] + ], + + "EMAIL" => [ + "INVALID" => "No hay cuenta para {{email}} .", + "IN_USE" => "El correo electrónico {{email}} ya está en uso.", + "VERIFICATION_REQUIRED" => "Correo electrónico (se requiere verificación - ¡use una dirección real!)" + ], + + "EMAIL_OR_USERNAME" => "Nombre de usuario o dirección de correo electrónico", + + "FIRST_NAME" => "Nombre", + + "HEADER_MESSAGE_ROOT" => "USTED HA INGRESADO COMO USUARIO ROOT", + + "LAST_NAME" => "Apellidos", + + "LOCALE" => [ + "ACCOUNT" => "El idioma y la configuración regional para utilizar en su cuenta", + "INVALID" => "{{locale}} no es un idioma válido." + ], + + "LOGIN" => [ + "@TRANSLATION" => "Acceder", + "ALREADY_COMPLETE" => "¡Ya se ha autentificado!", + "SOCIAL" => "O ingrese con", + "REQUIRED" => "Lo sentimos, debes iniciar sesión para acceder a este recurso." + ], + + "LOGOUT" => "Cerrar sesión", + + "NAME" => "Nombre", + + "NAME_AND_EMAIL" => "Nombre y correo electrónico", + + "PAGE" => [ + "LOGIN" => [ + "DESCRIPTION" => "Inicie sesión en su cuenta de {{site_name}} o regístrese para obtener una nueva cuenta.", + "SUBTITLE" => "Regístrese gratis o inicie sesión con una cuenta existente.", + "TITLE" => "¡Empecemos!", + ] + ], + + "PASSWORD" => [ + "@TRANSLATION" => "Contraseña", + + "BETWEEN" => "Entre {{min}} - {{max}} (recomendado 12)", + + "CONFIRM" => "Confirmar contraseña", + "CONFIRM_CURRENT" => "Por favor, confirma tu contraseña actual", + "CONFIRM_NEW" => "Confirmar nueva contraseña", + "CONFIRM_NEW_EXPLAIN" => "Vuelve a ingresar tu nueva contraseña", + "CONFIRM_NEW_HELP" => "Sólo se requiere si se selecciona una nueva contraseña", + "CREATE" => [ + "@TRANSLATION" => "Crear contraseña", + "PAGE" => "Elija una contraseña para su nueva cuenta.", + "SET" => "Establecer contraseña e iniciar sesión" + ], + "CURRENT" => "Contraseña actual", + "CURRENT_EXPLAIN" => "Debe confirmar su contraseña actual para realizar cambios", + + "FORGOTTEN" => "Contraseña olvidada", + "FORGET" => [ + "@TRANSLATION" => "Olvidé mi contraseña", + + "COULD_NOT_UPDATE" => "No se pudo actualizar la contraseña.", + "EMAIL" => "Introduce la dirección de correo electrónico que utilizaste para registrarte. Se le enviará por correo electrónico un enlace con las instrucciones para restablecer su contraseña.", + "EMAIL_SEND" => "Contraseña de correo electrónico Restablecer enlace", + "INVALID" => "No se pudo encontrar esta solicitud de restablecimiento de contraseña o ha caducado. Intenta volver a enviar tu solicitud .", + "PAGE" => "Obtenga un enlace para restablecer su contraseña.", + "REQUEST_CANNED" => "Se ha cancelado la solicitud de contraseña perdida.", + "REQUEST_SENT" => "Se ha enviado un enlace de restablecimiento de contraseña a {{email}} ." + ], + + "RESET" => [ + "@TRANSLATION" => "Restablecer la contraseña", + "CHOOSE" => "Por favor, elija una nueva contraseña para continuar.", + "PAGE" => "Elige una nueva contraseña para tu cuenta.", + "SEND" => "Establecer nueva contraseña e iniciar sesión" + ], + + "HASH_FAILED" => "El hash de la contraseña ha fallado. Póngase en contacto con un administrador del sitio.", + "INVALID" => "La contraseña actual no coincide con la que tenemos registrada", + "NEW" => "Nueva contraseña", + "NOTHING_TO_UPDATE" => "No se puede actualizar con la misma contraseña", + "UPDATED" => "Contraseña de la cuenta actualizada" + ], + + "PROFILE" => [ + "SETTINGS" => "Configuración de perfil", + "UPDATED" => "Configuración del perfil actualizada" + ], + + "RATE_LIMIT_EXCEEDED" => "Se ha superado el límite de velocidad para esta acción. Debe esperar otro {{delay}} segundos antes de que se le permita hacer otro intento.", + + "REGISTER" => "Registro", + "REGISTER_ME" => "Inscríbeme", + "REGISTRATION" => [ + "BROKEN" => "Lo sentimos, hay un problema con nuestro proceso de registro de cuenta. Póngase en contacto con nosotros directamente para obtener ayuda.", + "COMPLETE_TYPE1" => "Se ha registrado exitosamente. Ahora puede iniciar sesión.", + "COMPLETE_TYPE2" => "Se ha registrado exitosamente. Se ha enviado un enlace para activar tu cuenta a {{email}} . No podrá iniciar sesión hasta que complete este paso.", + "DISABLED" => "Lo sentimos, el registro de cuenta se ha deshabilitado.", + "LOGOUT" => "Lo siento, no puede registrarse para una cuenta mientras está conectado. Por favor, cierra la sesión primero.", + "WELCOME" => "El registro es rápido y sencillo." + ], + + "REMEMBER_ME" => "¡Recuérdame!", + "REMEMBER_ME_ON_COMPUTER" => "Recuérdeme en este ordenador (no se recomienda para ordenadores públicos)", + + "SIGNIN" => "Iniciar sesión", + "SIGNIN_OR_REGISTER" => "Ingresa o Registro", + "SIGNUP" => "Regístrate", + "SUGGEST" => "Sugerencia", + "HAVE_ACCOUNT" => "¿Ya tienes una cuenta?", + "SIGN_IN_HERE"=> "¿Ya tienes una cuenta? Acceda aquí. ", + + + "TOS" => "Términos y Condiciones", + "TOS_AGREEMENT" => "Al registrar una cuenta con {{site_title}}, acepta los términos y condiciones .", + "TOS_FOR" => "Términos y condiciones para {{title}}", + + "USERNAME" => [ + "@TRANSLATION" => "Nombre de usuario", + + "CHOOSE" => "Elige un nombre de usuario único", + "INVALID" => "Nombre de usuario no válido", + "IN_USE" => "El nombre de usuario {{user_name}} ya está en uso.", + "NOT_AVAILABLE" => "El nombre de usuario {{user_name}} no está disponible. Elija otro nombre o haga clic en \"sugerir\"." + ], + + "USER_ID_INVALID" => "El ID de usuario solicitado no existe.", + "USER_OR_EMAIL_INVALID" => "El nombre de usuario o la dirección de correo electrónico no son válidos.", + "USER_OR_PASS_INVALID" => "Usuario no encontrado o la contraseña no es válida.", + + "WELCOME" => "Bienvenido de nuevo, {{first_name}}" +]; diff --git a/main/app/sprinkles/account/locale/es_ES/validate.php b/main/app/sprinkles/account/locale/es_ES/validate.php new file mode 100755 index 0000000..c8ea0a4 --- /dev/null +++ b/main/app/sprinkles/account/locale/es_ES/validate.php @@ -0,0 +1,19 @@ + [ + "PASSWORD_MISMATCH" => "Su contraseña y contraseña de confirmación deben coincidir.", + "USERNAME" => "El nombre de usuario puede consistir sólo en letras minúsculas, números, '.', '-' y '_'." + ] +]; diff --git a/main/app/sprinkles/account/locale/fa/messages.php b/main/app/sprinkles/account/locale/fa/messages.php new file mode 100755 index 0000000..22623ba --- /dev/null +++ b/main/app/sprinkles/account/locale/fa/messages.php @@ -0,0 +1,178 @@ + [ + "@TRANSLATION" => "حساب", + + "ACCESS_DENIED" => "به نظر می آید که شما اجازه انجام این کار را ندارید", + + "DISABLED" => "این حساب کاربری غیر فعال شده است. برای اطلاعات بیشتر، لطفا با ما تماس برقرار کنید.", + + "EMAIL_UPDATED" => "آدرس پست الکترونیکی حساب، به روز رسانی شد", + + "INVALID" => "این اکانت موجود نیست. ممکن است که حذف شده باشد. برای اطلاعات بیشتر، لطفا با ما تماس برقرار کنید.", + + "MASTER_NOT_EXISTS" => "تا زمانی که حساب اصلی ساخته نشده است نمیتوانید حساب کاربری جدیدی بسازید.", + "MY" => "حساب من", + + "SESSION_COMPROMISED" => "ممکن است سژن شما مورد حمله واقع شده باشد. بهتر است با همه دستگاه های خود از وب سایت خارج شوید و دوباره وارد شوید. همچنین توجه بفرمایید که اطلاعات حسابتان، مورد حمله واقع نشده باشد. ", + "SESSION_COMPROMISED_TITLE" => "ممکن است که اکانت شما مورد حمله واقع شده باشد", + "SESSION_EXPIRED" => "سژن شما به پایان رسیده است. لطفا دوباره وارد شوید.", + + "SETTINGS" => [ + "@TRANSLATION" => "تنظیمات حساب", + "DESCRIPTION" => "اطلاعات حسابتان یعنی پست الکترونیکی،نام و گذرواژه خود را به روز رسانی کنید", + "UPDATED" => "تنظیمات حساب به روز رسانی شد" + ], + + "TOOLS" => "ابزار حساب", + + "UNVERIFIED" => "شما هنوز آدرس پست الکترونیکی خود را فعال نکرده اید. برای فعال سازی لطفا ایمیل خود را چک کنید.", + + "VERIFICATION" => [ + "NEW_LINK_SENT" => "لینک فعال سازی برای ایمیل {{email}} ارسال شد. لطفا ایمیل خود را چک کنید.", + "RESEND" => "ارسال دوباره ایمیل فعال سازی", + "COMPLETE" => "شما پست الکترونیکی خود را با موفقیت فعال سازی کردید. حالا می توانید وارد شوید.", + "EMAIL" => "لطفا آدرس پست الکترونیکی که با آن ثبت نام کردید وارد کنید تا ایمیل فعال سازی دوباره برایتان ارسال شود.", + "PAGE" => "ارسال دوباره ایمیل فعال سازی برای حساب جدید شما", + "SEND" => "ارسال ایمیل فعال سازی برای حساب کاربری", + "TOKEN_NOT_FOUND" => "این حساب کاربری یا قبلا فعال شده است و یا کد فعال سازی موجود نیست.", + ] + ], + + "EMAIL" => [ + "INVALID" => "حساب کاربری با {{email}} ثبت نشده است.", + "IN_USE" => "ایمیل {{email}} قبلا استفاده شده است", + "VERIFICATION_REQUIRED" => "آدرس پست الکترونیکی را بصورت صحیح وارد کنید" + ], + + "EMAIL_OR_USERNAME" => "نام کاربری یا آدرس پست الکترونیکی", + + "FIRST_NAME" => "نام", + + "HEADER_MESSAGE_ROOT" => "شما بعنوان کاربر اصلی وارد شده اید", + + "LAST_NAME" => "نام خانوادگی", + + "LOCALE" => [ + "ACCOUNT" => "زبان انتخابی برای حساب شما", + "INVALID" => "{{locale}} زبان صحیحی نیست" + ], + + "LOGIN" => [ + "@TRANSLATION" => "ورود", + "ALREADY_COMPLETE" => "شما قبلا وارد شده اید.", + "SOCIAL" => "یا با روش های زیر وارد شوید", + "REQUIRED" => "برای دیدن این صفحه لازم است که وارد شوید" + ], + + "LOGOUT" => "خروج", + + "NAME" => "نام", + + "NAME_AND_EMAIL" => "نام و پست الکترونیکی", + + "PAGE" => [ + "LOGIN" => [ + "DESCRIPTION" => "به حساب کاربری خود در {{site_name}} وارد شوید و یا حساب کاربری جدیدی بسازید", + "SUBTITLE" => "ثبت نام کنید و یا با حساب کاربری خود وارد شوید", + "TITLE" => "بیایید شروع کنیم!", + ] + ], + + "PASSWORD" => [ + "@TRANSLATION" => "گذرواژه", + + "BETWEEN" => "بین {{min}}-{{max}} حرف", + + "CONFIRM" => "رمز عبور را وارد کنید", + "CONFIRM_CURRENT" => "لطفا رمز عبور فعلی را تایید کنید", + "CONFIRM_NEW" => "رمز عبور جدید را وارد کنید", + "CONFIRM_NEW_EXPLAIN" => "رمز عبور جدید را تکرار کنید", + "CONFIRM_NEW_HELP" => "فقط زمانی لازم است که می خواهید گذرواژه جدیدی انتخاب کنید", + "CURRENT" => "گذرواژه فعلی", + "CURRENT_EXPLAIN" => "شما باید گذرواژه فعلی خود را وارد کنید تا بتوانید اطلاعات را به روز رسانی کنید", + + "FORGOTTEN" => "فراموشی گذرواژه", + "FORGET" => [ + "@TRANSLATION" => "گذرواژه خود را فراموش کرده ام", + + "COULD_NOT_UPDATE" => "گذرواژه به روز رسانی نشد", + "EMAIL" => "لطفا آدرس پست الکترونیکی که در زمان ثبت نام استفاده کردید، وارد کنید. لینک بازیابی گذرواژه برای شما ایمیل خواهد شد.", + "EMAIL_SEND" => "لینک بازیابی گذرواژه ایمیل شود", + "INVALID" => "درخواست بازیابی کذرواژه پیدا نشد و یا منقضی شده است. لطفا درخواست را دوباره ارسال کنید", + "PAGE" => "دریافت لینک بازیابی گذرواژه", + "REQUEST_CANNED" => "درخواست فراموشی گذرواژه، حذف شد.", + "REQUEST_SENT" => "ایمیل بازیابی گذرواژه به {{email}} ارسال شد." + ], + + "RESET" => [ + "@TRANSLATION" => "تغییر گذرواژه", + "CHOOSE" => "لطفا گذرواژه جدید را انتخاب کنید", + "PAGE" => "برای حساب خود، گذرواژه جدیدی انتخاب کنید.", + "SEND" => "گذرواژه جدید را انتخاب کرده و وارد شوید" + ], + + "HASH_FAILED" => "هشینگ گذرواژه با مشکل روبرو شد. لطفا با مسولین وب سایت تماس برقرار کنید", + "INVALID" => "گذرواژه فعلی درست وارد نشده است", + "NEW" => "گذرواژه جدید", + "NOTHING_TO_UPDATE" => "شما نمیتوانید همان گذرواژه را دوباره وارد کنید", + "UPDATED" => "گذرواژه به روز رسانی شد" + ], + + "PROFILE" => [ + "SETTINGS" => "تنظیمات شخصی حساب", + "UPDATED" => "تنظیمات شخصی حساب به روز رسانی شد" + ], + + "REGISTER" => "ثبت نام", + "REGISTER_ME" => "ثبت نام کن", + + "REGISTRATION" => [ + "BROKEN" => "متاسفانه پروسه ثبت نام با مشکلی روبرو شد. برای دریافت کمک لطفا با ما تماس بگیرید.", + "COMPLETE_TYPE1" => "شما با موفقیت ثبت نام کردید. حالا میتوانید وارد شوید.", + "COMPLETE_TYPE2" => "شما با موفقیت ثبت نام کردید. لینک فعال سازی حساب به آدرس پست الکترونیکیتان {{email}} ارسال شد. بدون فعال سازی نمیتوانید وارد شوید.", + "DISABLED" => "با عرض تاسف، امکان ثبت در وب سایت، غیر فعال شده است.", + "LOGOUT" => "شما همزمان این که وارد شده اید نمیتوانید حساب کاربری جدیدی بسازید. لطفا ابتدا خارج شوید.", + "WELCOME" => "سریع و ساده ثبت نام کنید" + ], + + "RATE_LIMIT_EXCEEDED" => "شما محدودیت تعداد انجام این کار را پشت سر گذاشتید. لطفا {{delay}} ثانیه دیگر صبر کرده و دوباره تلاش کنید.", + "REMEMBER_ME" => "من را به خاطر بسپار!", + "REMEMBER_ME_ON_COMPUTER" => "من را در این دستگاه به خاطر بسپار (برای دستگاه های عمومی پیشنهاد نمی شود)", + + "SIGNIN" => "ورود", + "SIGNIN_OR_REGISTER" => "ثبت نام کنید و یا وارد شوید", + "SIGNUP" => "ثبت نام", + + "TOS" => "شرایط و مقررات", + "TOS_AGREEMENT" => "با ثبت نام در {{site_title}} موافقت خود با شرایط و مقررات را نشان میدهید.", + "TOS_FOR" => "شرایط و مقررات {{title}}", + + "USERNAME" => [ + "@TRANSLATION" => "نام کاربری", + + "CHOOSE" => "یک نام کاربری منحصر به فرد انتخاب کنید", + "INVALID" => "نام کاربری معتبر نیست", + "IN_USE" => "نام کاربری {{user_name}} قبلا استفاده شده است", + "NOT_AVAILABLE" => "نام کاربری {{user_name}} موجود نیست. لطفا نام کاربری دیگری انتخاب کنید" + ], + + "USER_ID_INVALID" => "آی دی کاربری مد نظر شما موجود نیست", + "USER_OR_EMAIL_INVALID" => "نام کاربری و یا آدرس پست الکترونیکی معتبر نیست", + "USER_OR_PASS_INVALID" => "کاربری یافت نشد و یا گذرواژه صحیح نیست", + + "WELCOME" => "خوش آمدید {{first_name}}" +]; diff --git a/main/app/sprinkles/account/locale/fa/validate.php b/main/app/sprinkles/account/locale/fa/validate.php new file mode 100755 index 0000000..a63cae1 --- /dev/null +++ b/main/app/sprinkles/account/locale/fa/validate.php @@ -0,0 +1,20 @@ + [ + "PASSWORD_MISMATCH" => "گذرواژه و تکرار آن باید با یکدیگر تطبیق پیدا کنند", + "USERNAME" => "نام کاربری فقط میتواند از حروف کوچک، اعداد، '.'، '-' و '_' متشکل شوند." + ] +]; diff --git a/main/app/sprinkles/account/locale/fr_FR/messages.php b/main/app/sprinkles/account/locale/fr_FR/messages.php new file mode 100755 index 0000000..6e5a032 --- /dev/null +++ b/main/app/sprinkles/account/locale/fr_FR/messages.php @@ -0,0 +1,179 @@ + [ + "@TRANSLATION" => "Compte d'utilisateur", + + "ACCESS_DENIED" => "Hmm, on dirait que vous n'avez pas la permission de faire ceci.", + + "DISABLED" => "Ce compte a été désactivé. Veuillez nous contacter pour plus d'informations.", + + "EMAIL_UPDATED" => "Adresse email mise à jour", + + "INVALID" => "Ce compte n'existe pas. Il a peut-être été supprimé. Veuillez nous contacter pour plus d'informations.", + + "MASTER_NOT_EXISTS" => "Vous ne pouvez pas enregistrer un compte tant que le compte principal n'a pas été créé!", + "MY" => "Mon compte", + + "SESSION_COMPROMISED" => [ + "@TRANSLATION" => "Votre session a été compromise. Vous devez vous déconnecter de tous les périphériques, puis vous reconnecter et vous assurer que vos données n'ont pas été altérées.", + "TITLE" => "Votre compte peut avoir été compromis" + ], + "SESSION_EXPIRED" => "Votre session a expiré. Veuillez vous connecter à nouveau.", + + "SETTINGS" => [ + "@TRANSLATION" => "Paramètres du compte", + "DESCRIPTION" => "Mettez à jour les paramètres de votre compte, y compris votre adresse e-mail, votre nom et votre mot de passe.", + "UPDATED" => "Paramètres du compte mis à jour" + ], + + "TOOLS" => "Outils du compte", + + "UNVERIFIED" => "Votre compte n'a pas encore été vérifié. Vérifiez vos emails / dossier spam pour les instructions d'activation du compte.", + + "VERIFICATION" => [ + "NEW_LINK_SENT" => "Nous avons envoyé un nouveau lien de vérification à {{email}}. Veuillez vérifier vos dossiers de boîte de réception et de spam pour ce courriel.", + "RESEND" => "Renvoyer le courriel de validation", + "COMPLETE" => "Votre compte a été validé. Vous pouvez maintenant vous connecter.", + "EMAIL" => "Veuillez saisir l'adresse email que vous avez utilisée pour vous inscrire et votre courriel de vérification sera renvoyé.", + "PAGE" => "Renvoyer l'email de validation de votre nouveau compte.", + "SEND" => "Envoyer le lien de validation de mon compte", + "TOKEN_NOT_FOUND" => "Le jeton de vérification n'existe pas / Le compte est déjà vérifié", + ] + ], + + "EMAIL" => [ + "INVALID" => "Il n'y a aucun compte pour {{email}}.", + "IN_USE" => "Le email {{email}} est déjà utilisé.", + "VERIFICATION_REQUIRED" => "Email (vérification requise - utiliser une adresse réelle!)" + ], + + "EMAIL_OR_USERNAME" => "Nom d'utilisateur ou adresse email", + + "FIRST_NAME" => "Prénom", + + "HEADER_MESSAGE_ROOT" => "VOUS ÊTES CONNECTÉ EN TANT QUE L'UTILISATEUR ROOT", + + "LAST_NAME" => "Nom de famille", + + "LOCALE" => [ + "ACCOUNT" => "La langue utilisé pour votre compte d'utilisateur", + "INVALID" => "{{locale}} n'est pas une langue valide." + ], + + "LOGIN" => [ + "@TRANSLATION" => "Connexion", + "ALREADY_COMPLETE" => "Vous êtes déjà connecté!", + "SOCIAL" => "Ou se connecter avec", + "REQUIRED" => "Désolé, vous devez être connecté pour accéder à cette ressource." + ], + + "LOGOUT" => "Déconnexion", + + "NAME" => "Nom", + + "NAME_AND_EMAIL" => "Nom et email", + + "PAGE" => [ + "LOGIN" => [ + "DESCRIPTION" => "Connectez-vous à votre compte {{site_name}} ou enregistrez-vous pour un nouveau compte.", + "SUBTITLE" => "Inscrivez-vous gratuitement ou connectez-vous avec un compte existant.", + "TITLE" => "Commençons!", + ] + ], + + "PASSWORD" => [ + "@TRANSLATION" => "Mot de passe", + + "BETWEEN" => "Entre {{min}} et {{max}} charactères", + + "CONFIRM" => "Confirmer le mot de passe", + "CONFIRM_CURRENT" => "Veuillez confirmer votre mot de passe actuel", + "CONFIRM_NEW" => "Confirmer le nouveau mot de passe", + "CONFIRM_NEW_EXPLAIN" => "Confirmer le mot de passe", + "CONFIRM_NEW_HELP" => "Obligatoire uniquement si vous sélectionnez un nouveau mot de passe", + "CURRENT" => "Mot de passe actuel", + "CURRENT_EXPLAIN" => "Vous devez confirmer votre mot de passe actuel pour apporter des modifications", + + "FORGOTTEN" => "Mot de passe oublié", + "FORGET" => [ + "@TRANSLATION" => "J'ai oublié mon mot de passe", + + "COULD_NOT_UPDATE" => "Impossible de mettre à jour le mot de passe.", + "EMAIL" => "Veuillez saisir l'adresse e-mail que vous avez utilisée pour vous inscrire. Un lien avec les instructions pour réinitialiser votre mot de passe vous sera envoyé par email.", + "EMAIL_SEND" => "Envoyer le lien de réinitialisation", + "INVALID" => "Cette requête de réinitialisation de mot de passe n'a pas pu être trouvée ou a expiré. Veuillez réessayer de soumettre votre demande .", + "PAGE" => "Obtenir un lien pour réinitialiser votre mot de passe.", + "REQUEST_CANNED" => "Demande de mot de passe perdu annulée.", + "REQUEST_SENT" => "Si l'adresse e-mail {{email}} correspond à un compte dans notre système, un lien de réinitialisation de mot de passe sera envoyé à {{email}}." + ], + + "RESET" => [ + "@TRANSLATION" => "Réinitialiser le mot de passe", + "CHOOSE" => "Veuillez choisir un nouveau mot de passe pour continuer.", + "PAGE" => "Choisissez un nouveau mot de passe pour votre compte.", + "SEND" => "Définir un nouveau mot de passe" + ], + + "HASH_FAILED" => "Le hachage du mot de passe a échoué. Veuillez contacter un administrateur de site.", + "INVALID" => "Le mot de passe actuel ne correspond pas à celui que nous avons au dossier", + "NEW" => "Nouveau mot de passe", + "NOTHING_TO_UPDATE" => "Vous ne pouvez pas mettre à jour avec le même mot de passe", + "UPDATED" => "Mot de passe du compte mis à jour" + ], + + "PROFILE" => [ + "SETTINGS" => "Paramètres du profil", + "UPDATED" => "Paramètres du profil mis à jour" + ], + + "REGISTER" => "S'inscrire", + "REGISTER_ME" => "S'inscrire", + + "REGISTRATION" => [ + "BROKEN" => "Nous sommes désolés, il ya un problème avec notre processus d'enregistrement de compte. Veuillez nous contacter directement pour obtenir de l'aide.", + "COMPLETE_TYPE1" => "Vous êtes inscrit avec succès. Vous pouvez maintenant vous connecter.", + "COMPLETE_TYPE2" => "Vous êtes inscrit avec succès. Vous recevrez bientôt un e-mail de validation contenant un lien pour activer votre compte. Vous ne pourrez pas vous connecter avant d'avoir terminé cette étape.", + "DISABLED" => "Désolé, l'enregistrement de compte a été désactivé.", + "LOGOUT" => "Désolé, vous ne pouvez pas vous inscrire tout en étant connecté. Veuillez vous déconnecter en premier.", + "WELCOME" => "L'inscription est rapide et simple." + ], + + "RATE_LIMIT_EXCEEDED" => "La limite de tentatives pour cette action a été dépassée. Vous devez attendre {{delay}} secondes avant de pouvoir effectuer une autre tentative.", + "REMEMBER_ME" => "Se souvenir de moi!", + "REMEMBER_ME_ON_COMPUTER" => "Se souvenir de moi sur cet ordinateur (non recommandé pour les ordinateurs publics)", + + "SIGNIN" => "Se connecter", + "SIGNIN_OR_REGISTER" => "Se connecter ou s'inscrire", + "SIGNUP" => "S'inscrire", + + "TOS" => "Termes et conditions", + "TOS_AGREEMENT" => "En créant un compte avec {{site_title}}, vous acceptez les termes et conditions.", + "TOS_FOR" => "Termes et conditions pour {{title}}", + + "USERNAME" => [ + "@TRANSLATION" => "Nom d'utilisateur", + + "CHOOSE" => "Choisissez un nom d'utilisateur unique", + "INVALID" => "Nom d'utilisateur invalide", + "IN_USE" => "Le nom d'utilisateur '{{username}}' est déjà utilisé.", + "NOT_AVAILABLE" => "Le nom d'utilisateur {{user_name}} n'est pas disponible. Choisissez un autre nom, ou cliquez sur « suggérer »." + ], + + "USER_ID_INVALID" => "L'identifiant d'utilisateur demandé n'existe pas.", + "USER_OR_EMAIL_INVALID" => "Nom d'utilisateur ou adresse e-mail non valide.", + "USER_OR_PASS_INVALID" => "Nom d'utilisateur ou mot de passe incorrect.", + + "WELCOME" => "Bienvenue {{first_name}}" +]; diff --git a/main/app/sprinkles/account/locale/fr_FR/validate.php b/main/app/sprinkles/account/locale/fr_FR/validate.php new file mode 100755 index 0000000..44b1bc1 --- /dev/null +++ b/main/app/sprinkles/account/locale/fr_FR/validate.php @@ -0,0 +1,18 @@ + [ + "PASSWORD_MISMATCH" => "Votre mot de passe et votre mot de passe de confirmation doivent correspondre." + ] +]; diff --git a/main/app/sprinkles/account/locale/it_IT/messages.php b/main/app/sprinkles/account/locale/it_IT/messages.php new file mode 100755 index 0000000..fee2e8c --- /dev/null +++ b/main/app/sprinkles/account/locale/it_IT/messages.php @@ -0,0 +1,186 @@ + [ + "@TRANSLATION" => "Account", + + "ACCESS_DENIED" => "Sembra tu non abbiamo il permesso di fare questo.", + + "DISABLED" => "Questo account è stato disattivato, contattaci per maggiori informazioni", + + "EMAIL_UPDATED" => "Email aggiornata", + + "INVALID" => "Questo account non esiste. Può essere stato cancellato. Vi preghiamo di contattarci per ulteriori informazioni.", + + "MASTER_NOT_EXISTS" => "Non puoi registrare un account finche l'account primario non sarà creato!", + "MY" => "Il mio account", + + "SESSION_COMPROMISED" => [ + "@TRANSLATION" => "La tua sessione è stata compromessa. Devi eseguire il logout su tutti i dispositivi, quindi riaccenderti e assicurati che i tuoi dati non siano stati manomessi.", + "TITLE" => "Il tuo account potrebbe essere stato compromesso", + "TEXT" => "Qualcuno potrebbe aver utilizzato le tue informazioni di accesso per accedere a questa pagina. Per la tua sicurezza tutte le sessioni sono state disconnesse. Accedi e controlla l'account per attività sospette. Potresti anche desiderare di cambiare la tua password." + ], + "SESSION_EXPIRED" => "La tua sessione è scaduta. Accedi nuovamente.", + + "SETTINGS" => [ + "@TRANSLATION" => "Impostazioni dell 'account", + "DESCRIPTION" => "Aggiorna le impostazioni del tuo account, tra cui email, nome e password.", + "UPDATED" => "Impostazioni account aggiornate" + ], + + "TOOLS" => "Account tools", + + "UNVERIFIED" => "Il tuo account non è stato attivato. Controlla nella tua mail ( anche nella cartella dello spam ) per riceve le instruzioni per attivare il tuo account", + + "VERIFICATION" => [ + "NEW_LINK_SENT" => "Ti è stato inviato un nuovo codice di attivazione, controlla la tua email ({{email}}).", + "RESEND" => "Invia nuovamente email di verifica.", + "COMPLETE" => "Hai verificato con successo il tuo account. È ora possibile accedere.", + "EMAIL" => "Inserisci l'indirizzo email che hai utilizzato per registrarti e la tua email di verifica sarà resentata.", + "PAGE" => "Ripeti l'email di verifica per il tuo nuovo account.", + "SEND" => "Inviilo il collegamento di verifica per il mio account", + "TOKEN_NOT_FOUND" => "Il token non esiste / l'account è già stato attivato" + ] + ], + + "EMAIL" => [ + "INVALID" => "Non esiste alcun account per {{email}}.", + "IN_USE" => "L'email '{{email}}' è già in uso", + "VERIFICATION_REQUIRED" => "Email (verifica richiesta - utilizzare un indirizzo reale!)" + ], + + "EMAIL_OR_USERNAME" => "Username o Indirizzo Email", + + "FIRST_NAME" => "Nome", + + "HEADER_MESSAGE_ROOT" => "LOGGATO COME ROOT", + + "LAST_NAME" => "Cognome", + "LOCALE" => [ + "ACCOUNT" => "La lingua e la località da utilizzare per il tuo account", + "INVALID" => "{{locale}} non è una località valida.", + + + ], + "LOGIN" => [ + "@TRANSLATION" => "Accesso", + "ALREADY_COMPLETE" => "Sei già loggato!", + "SOCIAL" => "O accedi con", + "REQUIRED" => "Devi essere loggato per accedere a questa risorsa" + ], + "LOGOUT" => "Logout", + + "NAME" => "Nome", + + "NAME_AND_EMAIL" => "Nome e email", + + "PAGE" => [ + "LOGIN" => [ + "DESCRIPTION" => "Accedi al tuo account {{site_name}} o registrati per un nuovo account.", + "SUBTITLE" => "Registrati gratuitamente o accedi con un account esistente.", + "TITLE" => "Iniziamo!", + ] + ], + + "PASSWORD" => [ + "@TRANSLATION" => "Password", + + "BETWEEN" => "La password deve essere tra {{min}} e i {{max}} caratteri", + + "CONFIRM" => "Conferma la password", + "CONFIRM_CURRENT" => "Conferma la password attuale", + "CONFIRM_NEW" => "Conferma la tua nuova password", + "CONFIRM_NEW_EXPLAIN" => "Inserisci nuovamente la nuova password", + "CONFIRM_NEW_HELP" => "Richiesto solo se si seleziona una nuova password", + "CREATE" => [ + "@TRANSLATION" => "Crea password", + "PAGE" => "Scegli una password per il tuo nuovo account.", + "SET" => "Imposta password e accedi" + ], + "CURRENT" => "Password attuale", + "CURRENT_EXPLAIN" => "Devi confermare la tua password corrente per apportare modifiche", + + "FORGOTTEN" => "Password dimenticata", + "FORGET" => [ + "@TRANSLATION" => "Ho dimenticato la mia password", + + "COULD_NOT_UPDATE" => "Password non aggiornata", + "EMAIL" => "Inserisci l'indirizzo email che hai utilizzato per iscriverti. Un collegamento con le istruzioni per reimpostare la tua password verrà inviata via email.", + "EMAIL_SEND" => "Email link di resetta password", + "INVALID" => "Questa richiesta di ripristino della password non è stata trovata o è scaduta. Prova a riprovare la tua richiesta.", + "PAGE" => "Ottieni un collegamento per reimpostare la tua password.", + "REQUEST_CANNED" => "Richiesta di recupero password cancellata.", + "REQUEST_SENT" => "Se l'email {{email}} corrisponde a un account nel nostro sistema, verrà inviato un collegamento per la reimpostazione della password a {{email}}." + ], + + "HASH_FAILED" => "Hash della password fallito. Contatta l'amministratore di sistema.", + "INVALID" => "La password corrente non corrisponde con quella in memoria", + "NEW" => "Nuova Password", + "NOTHING_TO_UPDATE" => "Non puoi aggiornare con la stessa password", + + "RESET" => [ + "@TRANSLATION" => "Resetta la Password", + "CHOOSE" => "Inserisci la tua nuova password", + "PAGE" => "Scegli una nuova password per il tuo account.", + "SEND" => "Impostare nuova password e accedere" + ], + + "UPDATED" => "Password aggiornata" + ], + + "PROFILE" => [ + "SETTINGS" => "Impostazioni del profilo", + "UPDATED" => "Le impostazioni del profilo sono aggiornate" + ], + + "RATE_LIMIT_EXCEEDED" => "Il limite di velocità per questa azione è stato superato. Devi aspettare un altro {{delay}} secondi prima che ti sia permesso di fare un altro tentativo.", + "REGISTER" => "Registrare", + "REGISTER_ME" => "Iscrivimi", + "REGISTRATION" => [ + "BROKEN" => "Ci dispiace, c'è un problema con il nostro processo di registrazione dell'account. Vi preghiamo di contattarci direttamente per assistenza.", + "COMPLETE_TYPE1" => "Sei stato registrato con successo ora puoi eseguire il login", + "COMPLETE_TYPE2" => "Sei stato registrato con successo. Riceverai presto una mail a {{email}} per l'attivazione. Devi attivare il tuo account prima di eseguire il login.", + "DISABLED" => "La registrazione di nuovi account è stata bloccata", + "LOGOUT" => "Non è possibile registrare un account mentre si è loggati", + "WELCOME" => "La registrazione è semplice e veloce" + ], + "REMEMBER_ME" => "Ricordami!", + "REMEMBER_ME_ON_COMPUTER" => "Ricordami su questo computer (non consigliato per i computer pubblici)", + + "SIGN_IN_HERE" => "Hai già un account? Accedi qui", + "SIGNIN" => "Accedi", + "SIGNIN_OR_REGISTER" => "Accedi o registri", + "SIGNUP" => "Registrazione", + + "TOS" => "Termini e condizioni", + "TOS_AGREEMENT" => "Registrando un account con {{site_title}}, accetti il termini e condizioni.", + "TOS_FOR" => "Termini e condizioni per {{title}}", + + "USERNAME" => [ + "@TRANSLATION" => "Username", + + "CHOOSE" => "Inserisci il tuo username", + "INVALID" => "Username non valido", + "IN_USE" => "Il nome utente '{{user_name}}' è già in uso", + "NOT_AVAILABLE" => "Il nome utente {{user_name}} non è disponibile. Scegli un nome diverso, oppure fai clic su \"suggerisci\"." + ], + + "USER_ID_INVALID" => "User ID richiesto non è valido", + "USER_OR_EMAIL_INVALID" => "L'indirizzo mail o il nome utente non sono validi", + "USER_OR_PASS_INVALID" => "Il nome utente o la password non sono validi", + + "WELCOME" => "Bentornato, {{display_name}}" +]; diff --git a/main/app/sprinkles/account/locale/it_IT/validate.php b/main/app/sprinkles/account/locale/it_IT/validate.php new file mode 100755 index 0000000..713ccba --- /dev/null +++ b/main/app/sprinkles/account/locale/it_IT/validate.php @@ -0,0 +1,21 @@ + [ + "PASSWORD_MISMATCH" => "I due campi devono combaciare", + "USERNAME" => "L'username può essere composto da caratteri alfanumerici, '.', '-', e '_'." + ] +]; diff --git a/main/app/sprinkles/account/locale/pt_PT/messages.php b/main/app/sprinkles/account/locale/pt_PT/messages.php new file mode 100755 index 0000000..3db4200 --- /dev/null +++ b/main/app/sprinkles/account/locale/pt_PT/messages.php @@ -0,0 +1,166 @@ + [ + "@TRANSLATION" => "Conta", + + "ACCESS_DENIED" => "Hmm, parece que não tem permissões para fazer isso.", + + "DISABLED" => "Esta conta foi desativada. Por favor contacte-nos para mais informações.", + + "EMAIL_UPDATED" => "Email da conta atualizado", + + "INVALID" => "Esta conta não existe. Pode ter sido removida. Por favor contacte-nos para mais informações.", + + "MASTER_NOT_EXISTS" => "Não pode registrar uma conta enquanto a conta principal não for criada!", + "MY" => "A minha conta", + + "SESSION_COMPROMISED" => [ + "@TRANSLATION" => "A sua sessão foi comprometida. Deverá fechar todas as sessões, voltar a iniciar sessão e verificar que os seus dados não foram alterados por alheios.", + "TITLE" => "A sua sessão pode ter sido comprometida" + ], + "SESSION_EXPIRED" => "A sua sessão expirou. Por favor inicie nova sessão.", + + "SETTINGS" => [ + "@TRANSLATION" => "Definições de conta", + "DESCRIPTION" => "Atualize as suas definições, incluindo email, nome e password.", + "UPDATED" => "Definições de conta atualizadas" + ], + + "TOOLS" => "Ferramentas de conta", + + "UNVERIFIED" => "A sua conta ainda não foi verificada. Consulte o seu email (incluindo a pasta de spam) para instruções de ativação.", + + "VERIFICATION" => [ + "NEW_LINK_SENT" => "Enviámos um link de verificação para o endereço {{email}}. Por favor consulte o seu email (incluindo a pasta de spam).", + "RESEND" => "Enviar novamente email de verificação", + "COMPLETE" => "Verificou com sucesso a sua conta. Pode iniciar sessão.", + "EMAIL" => "Por favor introduza o endereço de email que utilizou no registro e um email de verificação será enviado.", + "PAGE" => "Reenviar email de verificação para a sua nova conta.", + "SEND" => "Enviar email com link de verificação", + "TOKEN_NOT_FOUND" => "Token de verificação inexistente / Conta já verificada", + ] + ], + + "EMAIL" => [ + "INVALID" => "Não existe nenhuma conta para {{email}}.", + "IN_USE" => "O email {{email}} já se encontra em uso." + ], + + "FIRST_NAME" => "Primeiro nome", + + "HEADER_MESSAGE_ROOT" => "INICIOU SESSÃO COM A CONTA ROOT", + + "LAST_NAME" => "Último nome", + + "LOCALE.ACCOUNT" => "Linguagem e localização a utilizar na sua conta", + + "LOGIN" => [ + "@TRANSLATION" => "Entrar", + + "ALREADY_COMPLETE" => "Sessão já iniciada!", + "SOCIAL" => "Ou inicie sessão com", + "REQUIRED" => "Lamentamos, tem de iniciar sessão para aceder a este recurso." + ], + + "LOGOUT" => "Sair", + + "NAME" => "Nome", + + "PAGE" => [ + "LOGIN" => [ + "DESCRIPTION" => "Inicie sessão na sua conta {{site_name}}, ou registre-se para uma nova conta.", + "SUBTITLE" => "Registre-se gratuitamente, ou inicie sessão com uma conta existente.", + "TITLE" => "Vamos começar!", + ] + ], + + "PASSWORD" => [ + "@TRANSLATION" => "Password", + + "BETWEEN" => "Entre {{min}}-{{max}} carateres", + + "CONFIRM" => "Confirme a password", + "CONFIRM_CURRENT" => "Por favor confirme a sua password atual", + "CONFIRM_NEW" => "Confirmar Nova Password", + "CONFIRM_NEW_EXPLAIN" => "Re-introduza a sua nova password", + "CONFIRM_NEW_HELP" => "Apenas necessário se escolher uma nova password", + "CURRENT" => "Password Atual", + "CURRENT_EXPLAIN" => "Tem de confirmar a sua password atual para efetuar alterações", + + "FORGOTTEN" => "Password Esquecida", + "FORGET" => [ + "@TRANSLATION" => "Esqueci a minha password", + + "COULD_NOT_UPDATE" => "Não foi possível atualizar a password.", + "EMAIL" => "Por favor introduza o endereço de email que utilizou no registro. Enviaremos um email com instruções para efetuar o reset à sua password.", + "EMAIL_SEND" => "Enviar email com link de reset da password", + "INVALID" => "This password reset request could not be found, or has expired. Please try resubmitting your request.", + "PAGE" => "Obtenha um link para fazer reset à sua password.", + "REQUEST_CANNED" => "Pedido de password esquecida foi cancelado.", + "REQUEST_SENT" => "Se o email {{email}} corresponder a uma conta em nosso sistema, um link de redefinição de senha será enviado para {{email}}." + ], + + "RESET" => [ + "@TRANSLATION" => "Reset Password", + "CHOOSE" => "Por favor escolha uma nova password para continuar.", + "PAGE" => "Escolha uma nova password para a sua conta.", + "SEND" => "Definir nova password e registrar" + ], + + "HASH_FAILED" => "Falhou o hashing da password. Por favor contacte um administrador do site.", + "INVALID" => "A password atual não coincide com a que temos em sistema", + "NEW" => "Nova Password", + "NOTHING_TO_UPDATE" => "Não pode atualizar para a mesma password", + "UPDATED" => "Password da conta foi atualizada" + ], + + "REGISTER" => "Registrar", + "REGISTER_ME" => "Registrar-me", + + "REGISTRATION" => [ + "BROKEN" => "Lamentamos, existe um problema com o nosso processo de registro. Contacte-nos diretamente para assistência.", + "COMPLETE_TYPE1" => "Registrou-se com sucesso. Pode iniciar sessão.", + "COMPLETE_TYPE2" => "Registrou-se com sucesso. Receberá em breve um email de verificação contendo um link para verificar a sua conta. Não será possível iniciar sessão até completar este passo.", + "DISABLED" => "Lamentamos, o registro de novas contas foi desativado.", + "LOGOUT" => "Não pode registrar uma nova conta enquanto tiver sessão iniciada. Por favor feche a sua sessão primeiro.", + "WELCOME" => "O registro é rápido e simples." + ], + + "RATE_LIMIT_EXCEEDED" => "Excedeu o número de tentativas para esta ação. Tem de aguardar {{delay}} segundos até lhe ser permitida nova tentativa.", + "REMEMBER_ME" => "Lembrar de mim!", + "REMEMBER_ME_ON_COMPUTER" => "Lembrar de mim neste computador (não recomendado em computadores públicos)", + + "SIGNIN" => "Iniciar Sessão", + "SIGNIN_OR_REGISTER" => "Iniciar sessão ou registrar", + "SIGNUP" => "Registrar", + + "TOS" => "Termos e Condições", + "TOS_AGREEMENT" => "Ao registrar uma conta em {{site_title}}, está a aceitar os termos e condições.", + "TOS_FOR" => "Termos e Condições para {{title}}", + + "USERNAME" => [ + "@TRANSLATION" => "Nome de utilizador", + + "CHOOSE" => "Escolha um nome de utilizador único", + "INVALID" => "Nome de utilizador inválido", + "IN_USE" => "O nome de utilizador {{user_name}} já se encontra em uso." + ], + + "USER_ID_INVALID" => "O id de utilizador solicitado não existe.", + "USER_OR_EMAIL_INVALID" => "Nome de utilizador ou endereço de email inválidos.", + "USER_OR_PASS_INVALID" => "Nome de utilizador ou password inválidos.", + + "WELCOME" => "Bem-vindo, {{first_name}}" +]; diff --git a/main/app/sprinkles/account/locale/pt_PT/validate.php b/main/app/sprinkles/account/locale/pt_PT/validate.php new file mode 100755 index 0000000..c05f14c --- /dev/null +++ b/main/app/sprinkles/account/locale/pt_PT/validate.php @@ -0,0 +1,18 @@ + [ + "PASSWORD_MISMATCH" => "A password e respetiva confirmação têm de coincidir." + ] +]; diff --git a/main/app/sprinkles/account/locale/ru_RU/messages.php b/main/app/sprinkles/account/locale/ru_RU/messages.php new file mode 100755 index 0000000..328db13 --- /dev/null +++ b/main/app/sprinkles/account/locale/ru_RU/messages.php @@ -0,0 +1,183 @@ + [ + "@TRANSLATION" => "Аккаунт", + + "ACCESS_DENIED" => "Для получения доступа у вас недостаточно прав.", + + "DISABLED" => "Аккаунт отключен. Пожалуйста, свяжитесь с нами для получения дополнительной информации.", + + "EMAIL_UPDATED" => "Email аккаунта обновлён", + + "INVALID" => "Этот аккаунт не существует. Возможно, он удалён. Пожалуйста, свяжитесь с нами для получения дополнительной информации.", + + "MASTER_NOT_EXISTS" => "Вы не можете зарегистрировать аккаунт до тех пор, пока основная учётная запись не будет создана!", + "MY" => "Мой профиль", + + "SESSION_COMPROMISED" => [ + "@TRANSLATION" => "Ваша сессия была скомпрометирована. Вы должны выйти на всех устройствах, а затем снова войти и убедиться, что ваши данные не были изменены.", + "TITLE" => "Возможно, ваш аккаунт был скомпрометированн", + "TEXT" => "Возможно, кто-то использовал ваши данные для входа на эту страницу. В целях безопасности все сеансы были завершены. Пожалуйста, повторно войдите и проверьте свой аккаунт на подозрительную активность. Рекомендуем сменить пароль." + ], + "SESSION_EXPIRED" => "Срок вашей сессии истек. Пожалуйста войдите еще раз.", + + "SETTINGS" => [ + "@TRANSLATION" => "Настройки аккаунта", + "DESCRIPTION" => "Обновите настройки своего аккаунта, включая адрес электронной почты, имя и пароль.", + "UPDATED" => "Данные аккаунта обновлены" + ], + + "TOOLS" => "Инструменты аккаунта", + + "UNVERIFIED" => "Ваш аккаунт ещё не подтверждён. Проверьте вашу email почту, в том числе папку спам и следуйте инструкциям.", + + "VERIFICATION" => [ + "NEW_LINK_SENT" => "Мы отправили на ваш email новую ссылку для активации {{email}}. Пожалуйста, проверьте папку \"Входящие\" и \"Спам\".", + "RESEND" => "Повторно отправить письмо с подтверждением", + "COMPLETE" => "Вы успешно подтвердили свой аккаунт. Теперь вы можете войти.", + "EMAIL" => "Введите email, который вы использовали для регистрации, вам будет повторно отправлено письмо с подтверждением.", + "PAGE" => "Повторно оправить письмо подтверждения на email для нового аккаунта.", + "SEND" => "Проверка по электронной почте для аккаунта", + "TOKEN_NOT_FOUND" => "Код подтверждения не действителен либо аккаунт уже подтверждён", + ] + ], + + "EMAIL" => [ + "INVALID" => "Нет не одного аккаунта с {{email}} .", + "IN_USE" => "Email {{email}} уже используется.", + "VERIFICATION_REQUIRED" => "Email (указывайте верный - необходим для активации!)" + ], + + "EMAIL_OR_USERNAME" => "Имя пользователя или Email", + + "FIRST_NAME" => "Имя", + + "HEADER_MESSAGE_ROOT" => "ВЫ АВТОРИЗОВАНЫ С СУПЕР-ПРАВАМИ", + + "LAST_NAME" => "Фамилия", + "LOCALE" => [ + "ACCOUNT" => "Основной язык для вашего аккаунта", + "INVALID" => "{{locale}} язык недопустим." + ], + "LOGIN" => [ + "@TRANSLATION" => "Вход", + "ALREADY_COMPLETE" => "Вы уже выполнили вход!", + "SOCIAL" => "Или войти через", + "REQUIRED" => "Извините, Вы должны авторизоваться для доступа к этому ресурсу." + ], + "LOGOUT" => "Выход", + + "NAME" => "Имя", + + "NAME_AND_EMAIL" => "Имя и email", + + "PAGE" => [ + "LOGIN" => [ + "DESCRIPTION" => "Войдите в свой аккаунт {{site_name}}, или Зарегистрируйтесь.", + "SUBTITLE" => "Зарегистрироваться или войти в существующий аккаунт.", + "TITLE" => "Приступим!", + ] + ], + + "PASSWORD" => [ + "@TRANSLATION" => "Пароль", + + "BETWEEN" => "Кол-во {{min}}-{{max}} символов", + + "CONFIRM" => "Подтверждение пароля", + "CONFIRM_CURRENT" => "Пожалуйста, введите ваш текущий пароль", + "CONFIRM_NEW" => "Подтвердите новый пароль", + "CONFIRM_NEW_EXPLAIN" => "Повторно введите Ваш новый пароль", + "CONFIRM_NEW_HELP" => "Требуется только при выборе нового пароля", + "CREATE" => [ + "@TRANSLATION" => "Создать пароль", + "PAGE" => "Выберите пароль для вашего аккаунта.", + "SET" => "Установить пароль и войти" + ], + "CURRENT" => "Текущий пароль", + "CURRENT_EXPLAIN" => "Для продолжения вы должны ввести текущий пароль", + + "FORGOTTEN" => "Забытый пароль?", + "FORGET" => [ + "@TRANSLATION" => "Я забыл свой пароль", + + "COULD_NOT_UPDATE" => "Не удалось обновить пароль.", + "EMAIL" => "Пожалуйста, введите адрес электронной почты, который Вы использовали при регистрации. Ссылка с инструкцией по сбросу пароля будет отправлена вам по электронной почте.", + "EMAIL_SEND" => "Ссылка сброса пароля по Email", + "INVALID" => "Этот запрос сброса пароля не может быть найден, или истек. Пожалуйста, попробуйте повторно сбросить пароль.", + "PAGE" => "Получите ссылку для сброса пароля.", + "REQUEST_CANNED" => "Запрос на сброс пароля отменен.", + "REQUEST_SENT" => "Если email {{email}} существует в нашей системе у какого-либо аккаунта, ссылка на сброс пароля будет направлена на {{email}}." + ], + + "HASH_FAILED" => "Хэширование пароля не удалось. Пожалуйста, попробуйте другой пароль, либо свяжитесь с администратором сайта.", + "INVALID" => "Текущий пароль не соответствует тому, который задан в системе.", + "NEW" => "Новый пароль", + "NOTHING_TO_UPDATE" => "Невозможно обновить с тем же паролем", + + "RESET" => [ + "@TRANSLATION" => "Сбросить пароль", + "CHOOSE" => "Пожалуйста, выберите новый пароль, чтобы продолжить.", + "PAGE" => "Выберите новый пароль для вашего аккаунта.", + "SEND" => "Задать новый пароль и войти" + ], + + "UPDATED" => "Пароль аккаунта обновлён" + ], + + "PROFILE" => [ + "SETTINGS" => "Настройки профиля", + "UPDATED" => "Настройки профиля обновлены" + ], + + "RATE_LIMIT_EXCEEDED" => "Превышен лимит попыток для этого действия. Вы должны подождать еще {{delay}} секунд, прежде чем вам вам будет разрешено сделать ещё попытку.", + + "REGISTER" => "Регистрация", + "REGISTER_ME" => "Зарегистрируйте меня", + "REGISTRATION" => [ + "BROKEN" => "К сожалению, есть проблема с регистрации аккаунта. Свяжитесь с нами напрямую для получения помощи.", + "COMPLETE_TYPE1" => "Вы успешно зарегистрировались. Теперь вы можете войти.", + "COMPLETE_TYPE2" => "Вы успешно зарегистрировались. Ссылка для активации вашего аккаунта была отправлена на {{email}}. Вы сможете войти в систему только после активации аккаунта.", + "DISABLED" => "Извините, регистрация аккаунта была отключена.", + "LOGOUT" => "Извините, вы не можете зарегистрироваться когда уже авторизовались в системе. Сначала выйдите из системы.", + "WELCOME" => "Быстрая и простая регистрация." + ], + "REMEMBER_ME" => "Запомнить", + "REMEMBER_ME_ON_COMPUTER" => "Запомнить меня на этом компьютере (не рекомендуется для общедоступных компьютеров)", + + "SIGN_IN_HERE" => "Уже есть аккаунт? войти.", + "SIGNIN" => "Вход", + "SIGNIN_OR_REGISTER" => "Регистрация или вход", + "SIGNUP" => "Вход", + + "TOS" => "Пользовательское соглашение", + "TOS_AGREEMENT" => "Регистрируя аккаунт на {{site_title}}, вы принимаете условия и положения.", + "TOS_FOR" => "Правила и условия для {{title}}", + + "USERNAME" => [ + "@TRANSLATION" => "Пользователь", + + "CHOOSE" => "Выберите имя пользователя", + "INVALID" => "Недопустимое имя пользователя", + "IN_USE" => "{{user_name}} имя пользователя уже используется.", + "NOT_AVAILABLE" => "Имя пользователя {{user_name}} не доступно. Выберите другое имя или нажмите кнопку «предложить»." + ], + + "USER_ID_INVALID" => "ID запрашиваемого пользователя не существует.", + "USER_OR_EMAIL_INVALID" => "Имя пользователя или email не верный.", + "USER_OR_PASS_INVALID" => "Пользователь не найден или пароль является недействительным.", + + "WELCOME" => "Добро пожаловать, {{first_name}}" +]; diff --git a/main/app/sprinkles/account/locale/ru_RU/validate.php b/main/app/sprinkles/account/locale/ru_RU/validate.php new file mode 100755 index 0000000..8ede5d8 --- /dev/null +++ b/main/app/sprinkles/account/locale/ru_RU/validate.php @@ -0,0 +1,19 @@ + [ + "PASSWORD_MISMATCH" => "Пароли не совпадают.", + "USERNAME" => "Имя может состоять только из строчных букв, цифр, '.', '-' и «_»." + ] +]; diff --git a/main/app/sprinkles/account/locale/th_TH/messages.php b/main/app/sprinkles/account/locale/th_TH/messages.php new file mode 100755 index 0000000..642a7c5 --- /dev/null +++ b/main/app/sprinkles/account/locale/th_TH/messages.php @@ -0,0 +1,164 @@ + [ + "@TRANSLATION" => "บัญชี", + + "ACCESS_DENIED" => "หืมม ดูเหมือนว่าคุณไม่ได้รับอนุญาตให้ทำเช่นนั้น", + + "DISABLED" => "บัญชีนี้ถูกปิดการใช้งานไปแล้ว กรุณาติดต่อเราสำหรับข้อมูลเพิ่มเติม", + + "EMAIL_UPDATED" => "ปรับปรุงบัญชีอีเมลแล้ว", + + "INVALID" => "ไม่พบบัญชีนี้ มันอาจถูกลบไปแล้ว กรุณาติดต่อเราสำหรับข้อมูลเพิ่มเติม", + + "MASTER_NOT_EXISTS" => "คุณไม่สามารถสมัครสมาชิกได้จนกว่าจะสร้างบัญชีหลัก!", + "MY" => "บัญชีของฉัน", + + "SESSION_COMPROMISED" => "เซสชันของคุณถูกลักลอบใช้ คุณควรจะออกจากระบบบนอุปกรณ์ทั้งหมดแล้วกลับเข้าสู่ระบบและตรวจสอบให้แน่ใจว่าไม่มีการแก้ไขข้อมูลของคุณ", + "SESSION_COMPROMISED_TITLE" => "บัญชีของคุณอาจถูกบุกรุก", + "SESSION_EXPIRED" => "เซสชันของคุณหมดอายุ กรุณาเข้าสู่ระบบอีกครั้ง", + + "SETTINGS" => [ + "@TRANSLATION" => "การตั้งค่าบัญชี", + "DESCRIPTION" => "ปรับปรุงการตั้งค่าบัญชีของคุณ รวมไปถึงอีเมล ชื่อ และรหัสผ่าน", + "UPDATED" => "ปรับปรุงการตั้งค่าบัญชีของคุณแล้ว" + ], + + "TOOLS" => "เครื่องมือบัญชี", + + "UNVERIFIED" => "บัญชีของคุณยังไม่ได้รับการยืนยัน กรุณาตรวจสอบกล่องอีเมลและจดหมายขยะของคุณสำหรับขั้นตอนการเปิดใช้งานบัญชี", + + "VERIFICATION" => [ + "NEW_LINK_SENT" => "เราได้ส่งลิงก์สำหรับการยืนยันใหม่ไปยังอีเมล {{email}} กรุณาตรวจสอบอีเมลนี้ในกล่องอีเมลและจดหมายขยะของคุณ", + "RESEND" => "ส่งอีเมลยืนยันอีกครั้ง", + "COMPLETE" => "คุณได้ยืนยันอีเมลของคุณเรียบร้อยแล้ว คุณสามารถเข้าสู่ระบบได้ทันที", + "EMAIL" => "กรุณากรอกอีเมลที่คุณได้ใช้สมัครไว้แล้วอีเมลยืนยันจะถูกส่งไปให้ใหม่", + "PAGE" => "ส่งอีเมลยืนยันสำหรับบัญชีของฉันใหม่", + "SEND" => "ส่งอีเมลยืนยันให้บัญชีของฉัน", + "TOKEN_NOT_FOUND" => "ไม่พบโทเคนยืนยันอีเมล / บัญชีนี้ได้ยืนยันแล้ว", + ] + ], + + "EMAIL" => [ + "INVALID" => "อีเมล {{email}} ไม่มีอยู่จริง", + "IN_USE" => "อีเมล {{email}} ได้ถูกใช้งานแล้ว" + ], + + "FIRST_NAME" => "ชื่อจริง", + + "HEADER_MESSAGE_ROOT" => "คุณได้เข้าสู่ระบบเป็นผู้ดูแลสูงสุด", + + "LAST_NAME" => "นามสกุล", + + "LOCALE.ACCOUNT" => "ภาษาและสถานที่ที่จะใช้สำหรับบัญชีของคุณ", + + "LOGIN" => [ + "@TRANSLATION" => "เข้าสู่ะระบบ", + + "ALREADY_COMPLETE" => "คุณได้เข้าสู่ระบบอยู่แล้ว!", + "SOCIAL" => "หรือเข้าสู่ระบบด้วย", + "REQUIRED" => "ขออภัย คุณจะต้องเข้าสู่ระบบเพื่อเข้าถึงส่วนนี้" + ], + + "LOGOUT" => "ออกจากระบบ", + + "NAME" => "ชื่อ", + + "PAGE" => [ + "LOGIN" => [ + "DESCRIPTION" => "เข้าสู่ระบบไปยังบัญชี {{site_name}} หรือสมัครสมาชิกสำหรับบัญชีใหม่", + "SUBTITLE" => "สมัครสมาชิกฟรี หรือเข้าสู่ระบบด้วยบัญชีที่มีอยู่", + "TITLE" => "มาเริ่มกันเลย!", + ] + ], + + "PASSWORD" => [ + "@TRANSLATION" => "รหัสผ่าน", + + "BETWEEN" => "ระหว่าง {{min}}-{{max}} ตัวอักษร", + + "CONFIRM" => "ยืนยันรหัสผ่าน", + "CONFIRM_CURRENT" => "กรุณายืนยันรหัสผ่านปัจจุบันของคุณ", + "CONFIRM_NEW" => "ยืนยันรหัสผ่านใหม่", + "CONFIRM_NEW_EXPLAIN" => "กรอกรหัสผ่านใหม่ของคุณอีกครั้ง", + "CONFIRM_NEW_HELP" => "กรอกเฉพาะเมื่อคุณต้องการตั้งรหัสผ่านใหม่", + "CURRENT" => "รหัสผ่านปัจจุบัน", + "CURRENT_EXPLAIN" => "คุณจะต้องยืนยันรหัสผ่านปัจจุบันเพื่อแก้ไขข้อมูล", + + "FORGOTTEN" => "ลืมรหัสผ่าน", + "FORGET" => [ + "@TRANSLATION" => "ฉันลืมรหัสผ่านของฉัน", + + "COULD_NOT_UPDATE" => "ไม่สามารถปรับปรุงรหัสผ่าน", + "EMAIL" => "กรุณากรอกที่อยู่อีเมลที่คุณเคยใช้เข้าสู่ระบบ ลิงก์ขั้นตอนการรีเซ็ตรหัสผ่านของคุณจะถูกส่งไปให้คุณ", + "EMAIL_SEND" => "ลิงก์รีเซ็ตรหัสผ่านจากอีเมล", + "INVALID" => "ขอรีเซ็ตรหัสผ่านนี้ไม่มีอยู่ หรือหมดอายุไปแล้ว กรุณาลอง ส่งคำขอของคุณอีกครั้ง", + "PAGE" => "รับลิงก์สำหรับการรีเซ็ตรหัสผ่านของคุณ", + "REQUEST_CANNED" => "คำขอลืมรหัสผ่านได้ถูกยกเลิก", + "REQUEST_SENT" => "หากอีเมล {{email}} ตรงกับบัญชีในระบบของเราลิงก์การรีเซ็ตรหัสผ่านจะถูกส่งไปที่ {{email}}" + ], + + "RESET" => [ + "@TRANSLATION" => "รีเซ็ตรหัสผ่าน", + "CHOOSE" => "กรุณาเลือกรหัสผ่านใหม่เพื่อดำเนินการต่อ", + "PAGE" => "เลือกรหัสผ่านใหม่สำหรับบัญชีของคุณ", + "SEND" => "ตั้งรหัสผ่านใหม่และเข้าสู่ระบบ" + ], + + "HASH_FAILED" => "เข้ารหัสรหัสผ่านล้มเหลว กรุณาติดต่อผู้ดูแลระบบของเว็บไซต์", + "INVALID" => "รหัสผ่านปัจจุบันไม่ตรงกับรหัสผ่านที่เราบันทึกไว้", + "NEW" => "รหัสผ่านใหม่", + "NOTHING_TO_UPDATE" => "คุณไม่สามารถปรังปรุงด้วยรหัสผ่านเดียวกัน", + "UPDATED" => "ปรังปรุงรหัสผ่านของบัญชีแล้ว" + ], + + "REGISTER" => "สมัครสมาชิก", + "REGISTER_ME" => "ให้ฉันสมัครสมาชิกด้วย", + + "REGISTRATION" => [ + "BROKEN" => "เราขออภัย มันมีปัญหาในการดำเนินการสมัครสมาชิกของเรา กรุณาติดต่อเราโดยตรงเพื่อขอความช่วยเหลือ", + "COMPLETE_TYPE1" => "คุณได้สมัครสมาชิกเรียบร้อยแล้ว คุณสามารถเข้าสู่ระบบได้ทันที", + "COMPLETE_TYPE2" => "คุณได้สมัครสมาชิกเรียบร้อยแล้ว คุณจะได้รับอีเมลยืนยันที่มีลิงก์สำหรับเปิดใช้งานบัญชีของคุณอยู่ คุณจะไม่สามารถเข้าสู่ระบบจนกว่าคุณจะยืนยันอีเมลแล้ว", + "DISABLED" => "เราขออภัย ระบบสมัครสมาชิกได้ถูกปิดไว้", + "LOGOUT" => "เราขออภัย คุณไม่สามารถสมัครสมาชิกขณะที่เข้าสู่ระบบอยู่ กรุณาออกจากระบบก่อน", + "WELCOME" => "การสมัครสมาชิกนั้นรวดเร็ว และง่ายดาย" + ], + + "RATE_LIMIT_EXCEEDED" => "ถึงขีดจำกัดสำหรับการกระทำนี้แล้ว คุณจะต้องรออีก {{delay}} วินาที ก่อนที่คุณจะได้รับอนุญาตให้ลองใหม่อีกครั้ง", + "REMEMBER_ME" => "จำฉันไว้ในระบบ!", + "REMEMBER_ME_ON_COMPUTER" => "จำฉันไว้ในระบบบนคอมพิวเตอร์นี้ (ไม่แนะนำสำหรับคอมพิวเตอร์สาธารณะ)", + + "SIGNIN" => "เข้าสู่ะระบบ", + "SIGNIN_OR_REGISTER" => "เข้าสู่ระบบหรือสมัครสมาชิก", + "SIGNUP" => "สมัครสมาชิก", + + "TOS" => "ข้อตกลงและเงื่อนไข", + "TOS_AGREEMENT" => "ในการสมัครสมาชิกกับ {{site_title}} หมายถึงคุณยอมรับ ข้อตกลงและเงื่อนไข แล้ว", + "TOS_FOR" => "ข้อตกลงและเงื่อนไขสำหรับ {{title}}", + + "USERNAME" => [ + "@TRANSLATION" => "ชื่อผู้ใช้", + + "CHOOSE" => "เลือกชื่อผู้ใช้ที่เป็นเป็นเอกลักษณ์", + "INVALID" => "ชื่อผู้ใช้ไม่ถูกต้อง", + "IN_USE" => "ชื่อผู้ใช้ {{user_name}} ถูกใช้งานแล้ว" + ], + + "USER_ID_INVALID" => "ไม่พบหมายเลขผู้ใช้ที่ร้องขอมา", + "USER_OR_EMAIL_INVALID" => "ชื่อผู้ใช้หรือที่อยู่อีเมลไม่ถูกต้อง", + "USER_OR_PASS_INVALID" => "ชื่อผู้ใช้หรือรหัสผ่านไม่ถูกต้อง", + + "WELCOME" => "ยินดีต้อนรับ {{first_name}}" +]; diff --git a/main/app/sprinkles/account/locale/th_TH/validate.php b/main/app/sprinkles/account/locale/th_TH/validate.php new file mode 100755 index 0000000..1a2e90a --- /dev/null +++ b/main/app/sprinkles/account/locale/th_TH/validate.php @@ -0,0 +1,18 @@ + [ + "PASSWORD_MISMATCH" => "รหัสผ่านและรหัสผ่านยืนยันของคุณจะต้องตรงกัน" + ] +]; diff --git a/main/app/sprinkles/account/locale/tr/messages.php b/main/app/sprinkles/account/locale/tr/messages.php new file mode 100755 index 0000000..5213490 --- /dev/null +++ b/main/app/sprinkles/account/locale/tr/messages.php @@ -0,0 +1,183 @@ + [ + "@TRANSLATION" => "Hesap", + + "ACCESS_DENIED" => "Hmm. görünüşe göre böyle bir şey için izne sahip değilsiniz.", + + "DISABLED" => "Bu hesap durduruldu. Daha çok bilgi için bizimle iletişime geçin.", + + "EMAIL_UPDATED" => "Hesap maili güncellendi", + + "INVALID" => "Bu hesap bulunamadı. Silinmiş olabilir. Daha çok bilgi için bizimle iletişime geçin.", + + "MASTER_NOT_EXISTS" => "Ana hesap oluşturuluncaya kadar bir hesap oluşturamazsın!", + "MY" => "Hesabım", + + "SESSION_COMPROMISED" => [ + "@TRANSLATION" => "Oturumunuz tehlikeye atıldı. Tüm cihazlardan çıkmanız, daha sonra giriş yapmanız ve bilgilerinizin değiştirilmediğini kontrol etmeniz gerekir.", + "TITLE" => "Hesabınız tehlikeye atılmış olabilir", + "TEXT" => "Birisi bu sayfayı ele geçirmek için giriş verilerinizi kullanmış olabilir. Güvenliğiniz için tüm oturumlar günlüğe kaydedildi. Lütfen giriş yapınve şüpheli hareketler için hesabınızı kontrol edin. Ayrıca şifrenizi değiştirmek isteyebilirsiniz." + ], + "SESSION_EXPIRED" => "Oturumunuz sona erdi. Lütfen tekrar oturum açın.", + + "SETTINGS" => [ + "@TRANSLATION" => "Hesap ayarları", + "DESCRIPTION" => "E-posta, isim ve parolanız da dahil olmak üzere hesap ayarlarınızı güncelleyin.", + "UPDATED" => "Hesap ayarları güncellendi" + ], + + "TOOLS" => "Hesap araçları", + + "UNVERIFIED" => "Hesap henüz onaylanmadı. Hesap etkinleştirme talimatları için e-postalarınızı ve spam klasörünüzü kontrol edin.", + + "VERIFICATION" => [ + "NEW_LINK_SENT" => "{{email}} için yeni bir doğrulama bağlantısı e-posta ile gönderildi. Lütfen bu e-postanın gelen kutusunu ve spam klasörlerini kontrol edin.", + "RESEND" => "Doğrulama e-postasını tekrar gönder", + "COMPLETE" => "Hesabınızı başarıyla doğruladınız. Şimdi giriş yapabilirsiniz.", + "EMAIL" => "Kaydolmak için kullandığınız e-posta adresinizi giriniz, ve doğrulama e-postanızı tekrar gönderin.", + "PAGE" => "Yeni hesabınız için doğrulama e-postasını tekrar gönder.", + "SEND" => "Hesabım için doğrulama bağlantısını e-posta ile gönder", + "TOKEN_NOT_FOUND" => "Doğrulama belirteci bulunumadı / Hesap zaten doğrulandı", + ] + ], + + "EMAIL" => [ + "INVALID" => "{{email}} için hesap yoktur.", + "IN_USE" => "E-posta {{email}} zaten kullanılıyor.", + "VERIFICATION_REQUIRED" => "E-posta (doğrulama gerekli - gerçek bir adres kullanın!)" + ], + + "EMAIL_OR_USERNAME" => "Kullanıcı adı veya e-posta adresi", + + "FIRST_NAME" => "Adınız", + + "HEADER_MESSAGE_ROOT" => "Kök kullanıcı olarak giriş yaptın", + + "LAST_NAME" => "Soyadı", + "LOCALE" => [ + "ACCOUNT" => "Hesabınız için kullanılacak dil ve yerel ayar", + "INVALID" => "{{locale}} geçersiz bir yerel." + ], + "LOGIN" => [ + "@TRANSLATION" => "Oturum Aç", + "ALREADY_COMPLETE" => "Zaten oturum açtınız!", + "SOCIAL" => "Veya şununla oturum aç", + "REQUIRED" => "Üzgünüm, bu sayfaya ulaşmak için oturum açmalısın." + ], + "LOGOUT" => "Oturumu kapat", + + "NAME" => "Ad", + + "NAME_AND_EMAIL" => "Ad ve e-posta", + + "PAGE" => [ + "LOGIN" => [ + "DESCRIPTION" => "{{site_name}} hesabınız ile giriş yapın ya da yeni bir hesap oluşturun.", + "SUBTITLE" => "Ücretsiz üye ol veya mevcut bir hesap ile giriş yapın.", + "TITLE" => "Hadi başlayalım!", + ] + ], + + "PASSWORD" => [ + "@TRANSLATION" => "Parola", + + "BETWEEN" => "{{min}}-{{max}} karakterler arasında", + + "CONFIRM" => "Şifreyi onayla", + "CONFIRM_CURRENT" => "Lütfen şuanki parolanızı giriniz", + "CONFIRM_NEW" => "Yeni parolayı onayla", + "CONFIRM_NEW_EXPLAIN" => "Yeni parolayı tekrar gir", + "CONFIRM_NEW_HELP" => "Sadece yeni bir şifre seçerseniz gerekli", + "CREATE" => [ + "@TRANSLATION" => "Parola Oluştur", + "PAGE" => "Yeni hesabınız için bir şifre belirleyin.", + "SET" => "Parolayı Ayarla ve Giriş Yap" + ], + "CURRENT" => "Şimdiki Parola", + "CURRENT_EXPLAIN" => "Değişiklikler için şimdiki parolanız ile onaylamalısınız", + + "FORGOTTEN" => "Unutulan Şifre", + "FORGET" => [ + "@TRANSLATION" => "Şifremi unuttum", + + "COULD_NOT_UPDATE" => "Şifre güncellenemedi.", + "EMAIL" => "Lütfen kaydolmak için kullandığınız e-posta adresini giriniz. Şifrenizi sıfırlama talimatlarıyla bir bir bağlantı e-postanıza gönderilecektir.", + "EMAIL_SEND" => "E-posta şifre sıfırlama bağlantısı", + "INVALID" => "Bu şifre sıfırlama isteği bulunamadı ya da süresi bitmiş. Lütfen isteğinizi yeniden göndermeyideneyin.", + "PAGE" => "Şifrenizi sıfırlamak için bir bağlantı oluşturun.", + "REQUEST_CANNED" => "Kayıp parola isteği iptal edildi.", + "REQUEST_SENT" => "Eğer e-posta{{email}} sistemdeki bir hesap ile eşleşirse, bir şifre yenileme bağlantısı{{email}} gönderilir." + ], + + "HASH_FAILED" => "Parola karma başarısız oldu. Lütfen bir site yöneticisiyle iletişime geçin.", + "INVALID" => "Şimdiki şifre kayıt edilen şifre ile eşleşmiyor", + "NEW" => "Yeni Şifre", + "NOTHING_TO_UPDATE" => "Aynı şifre ile güncelleyemezsiniz", + + "RESET" => [ + "@TRANSLATION" => "Şifre sıfırlama", + "CHOOSE" => "Lütfen devam etmek için yeni bir şifre belirleyiniz.", + "PAGE" => "Hesabınız için yeni bir şifre belirleyiniz.", + "SEND" => "Yeni şifre ayarla ve giriş yap" + ], + + "UPDATED" => "Hesap şifresi güncellendi" + ], + + "PROFILE" => [ + "SETTINGS" => "Profil ayarları", + "UPDATED" => "Profil ayarları güncellendi" + ], + + "RATE_LIMIT_EXCEEDED" => "Bu işlem için belirlenen son oran aşıldı. Başka bir deneme yapmanıza izin verilene kadar {{delay}} bir süre beklemelisiniz.", + + "REGISTER" => "Kaydol", + "REGISTER_ME" => "Beni kaydet", + "REGISTRATION" => [ + "BROKEN" => "Üzgünüz, hesap kayıt işlemimizde bir sorun var. Lütfen destek almak için doğrudan bizimle iletişime geçin.", + "COMPLETE_TYPE1" => "Kaydınız başarıyla tamamlandı. Şimdi giriş yapabilirsiniz.", + "COMPLETE_TYPE2" => "Kaydınız başarıyla tamamlandı. Hesabınızı aktifleştirmek için bir bağlantı gönderildi{{email}}. Bu adımı tamamlayana kadar oturum açamazsınız.", + "DISABLED" => "Üzgünüz, hesap kaydı devre dışı bırakıldı.", + "LOGOUT" => "Üzgünüm, oturumunuz açıkken yeni bir hesap oluşturamazsınız. Lütfen önce oturumunuzdan çıkış yapınız.", + "WELCOME" => "Kaydolmak hızlı ve basittir." + ], + "REMEMBER_ME" => "Beni hatırla!", + "REMEMBER_ME_ON_COMPUTER" => "Bu bilgisayarda beni hatırla ( genel bilgisayarlar için önerilmez)", + + "SIGN_IN_HERE" => "Zaten bir hesaba sahip misiniz?burada giriş yap", + "SIGNIN" => "Giriş yap", + "SIGNIN_OR_REGISTER" => "Giriş yap veya kayıt ol", + "SIGNUP" => "Üye ol", + + "TOS" => "Şartlar ve Koşullar", + "TOS_AGREEMENT" => "Bir hesap ile kaydolarak {{site_title}} sen kabul edersin şartlar ve koşulları.", + "TOS_FOR" => "{{title}} için şartlar ve koşullar", + + "USERNAME" => [ + "@TRANSLATION" => "Kullanıcı Adı", + + "CHOOSE" => "Benzersiz bir kullanıcı adı seç", + "INVALID" => "Geçersiz kullanıcı adı", + "IN_USE" => "{{user_name}} kullanıcı adı zaten mevcut.", + "NOT_AVAILABLE" => "{{user_name}} kullanıcı adı kullanılamaz. Farklı bir isim veya 'öneriye' tıklayın." + ], + + "USER_ID_INVALID" => "İstenen kullanıcı adı mevcut değil.", + "USER_OR_EMAIL_INVALID" => "Kullanıcı adı veya e-posta adresi hatalı.", + "USER_OR_PASS_INVALID" => "Kullanıcı bulunamadı ya da şifre hatalı.", + + "WELCOME" => "Tekrar Hoşgeldiniz.{{first_name}}" +]; diff --git a/main/app/sprinkles/account/locale/tr/validate.php b/main/app/sprinkles/account/locale/tr/validate.php new file mode 100755 index 0000000..298bdbc --- /dev/null +++ b/main/app/sprinkles/account/locale/tr/validate.php @@ -0,0 +1,19 @@ + [ + "PASSWORD_MISMATCH" => "Şifreniz ve onaylama şifreniz eşleşmiyor.", + "USERNAME" => "Kullanıcı adınız sadece küçük harfler, sayılar, '.', '-', ve '_' içerebilir." + ] +]; diff --git a/main/app/sprinkles/account/locale/zh_CN/messages.php b/main/app/sprinkles/account/locale/zh_CN/messages.php new file mode 100755 index 0000000..60adcf0 --- /dev/null +++ b/main/app/sprinkles/account/locale/zh_CN/messages.php @@ -0,0 +1,177 @@ + [ + "@TRANSLATION" => "账户", + + "ACCESS_DENIED" => "噢, 你好像没有权限这么做.", + + "DISABLED" => "这个账户已被禁用. 请联系我们获取更多信息.", + + "EMAIL_UPDATED" => "账户邮箱更新成功", + + "INVALID" => "此账户不存在. 可能已被删除. 请联系我们获取更多信息.", + + "MASTER_NOT_EXISTS" => "在创建超级账户之前你不能注册", + "MY" => "我的账户", + + "SESSION_COMPROMISED" => "你的会话已泄露. 你应该在所有的设备上注销, 然后再登陆确保你的数据没被修改.", + "SESSION_COMPROMISED_TITLE" => "你的账户可能被盗用", + "SESSION_EXPIRED" => "会话已过期. 请重新登陆.", + + "SETTINGS" => [ + "@TRANSLATION" => "账户设置", + "DESCRIPTION" => "更新你的账户, 包括邮箱、姓名和密码.", + "UPDATED" => "账户更新成功" + ], + + "TOOLS" => "账户工具", + + "UNVERIFIED" => "你的账户还没有验证. 检查你的(垃圾)邮箱文件夹进行验证.", + + "VERIFICATION" => [ + "NEW_LINK_SENT" => "我们发送了新的验证链接 {{email}}. 请检查你的收件箱或垃圾邮件进行验证.", + "RESEND" => "重新发送验证邮件", + "COMPLETE" => "你已成功验证. 现在可以登陆了.", + "EMAIL" => "请输入你登陆时的邮箱, 然后将会发送验证邮件.", + "PAGE" => "重新发送验证邮件给你的新账户.", + "SEND" => "为我的账户发送验证邮件", + "TOKEN_NOT_FOUND" => "验证令牌不存在 / 账户已经验证", + ] + ], + + "EMAIL" => [ + "INVALID" => "{{email}} 没有账户注册.", + "IN_USE" => "邮箱 {{email}} 已被使用.", + "VERIFICATION_REQUIRED" => "邮箱 (需要进行验证 - 请使用一个有效的!)" + ], + + "EMAIL_OR_USERNAME" => "用户名或邮箱地址", + + "FIRST_NAME" => "名字", + + "HEADER_MESSAGE_ROOT" => "你现在以超级用户登陆", + + "LAST_NAME" => "姓氏", + + "LOCALE" => [ + "ACCOUNT" => "设置你账户的地区和语言", + "INVALID" => "{{locale}} 不是一个有效的地区." + ], + + "LOGIN" => [ + "@TRANSLATION" => "登陆", + "ALREADY_COMPLETE" => "你已经登陆!", + "SOCIAL" => "用其他方式登陆", + "REQUIRED" => "对不起, 你需要登陆才能获取资源." + ], + + "LOGOUT" => "注销", + + "NAME" => "名字", + + "NAME_AND_EMAIL" => "名字和邮箱", + + "PAGE" => [ + "LOGIN" => [ + "DESCRIPTION" => "用 {{site_name}} 账户登陆, 或者创建新账户.", + "SUBTITLE" => "免费注册, 或用已有账户登陆.", + "TITLE" => "让我们开始吧!", + ] + ], + + "PASSWORD" => [ + "@TRANSLATION" => "密码", + + "BETWEEN" => "字符长度 {{min}}-{{max}} ", + + "CONFIRM" => "确认密码", + "CONFIRM_CURRENT" => "请确认当前密码", + "CONFIRM_NEW" => "确认新密码", + "CONFIRM_NEW_EXPLAIN" => "重新输入新密码", + "CONFIRM_NEW_HELP" => "选择了新密码时才需要", + "CURRENT" => "密码正确", + "CURRENT_EXPLAIN" => "你必须要确认密码再进行修改", + + "FORGOTTEN" => "忘记密码", + "FORGET" => [ + "@TRANSLATION" => "我忘记了密码", + + "COULD_NOT_UPDATE" => "无法更新密码.", + "EMAIL" => "请输入你登陆时的邮箱. 重置密码的链接将会发送给你.", + "EMAIL_SEND" => "发送重置密码链接", + "INVALID" => "这个重置密码请求无法使用, 或已过期. 请 重新发送请求.", + "PAGE" => "获取重置密码的链接.", + "REQUEST_CANNED" => "取消重置请求.", + "REQUEST_SENT" => "重置密码的链接已经发送 {{email}}." + ], + + "RESET" => [ + "@TRANSLATION" => "重置密码", + "CHOOSE" => "请输入新密码.", + "PAGE" => "为账户设置新密码.", + "SEND" => "设置密码并登陆" + ], + + "HASH_FAILED" => "密码验证失败. 请联系网站管理.", + "INVALID" => "当前密码无法与记录匹配", + "NEW" => "新密码", + "NOTHING_TO_UPDATE" => "新密码不能与旧密码相同", + "UPDATED" => "账户密码更新成功" + ], + + "PROFILE" => [ + "SETTINGS" => "简介设置", + "UPDATED" => "简介设置成功" + ], + + "REGISTER" => "注册", + "REGISTER_ME" => "注册", + + "REGISTRATION" => [ + "BROKEN" => "抱歉, 账户注册过程发送错误. 请联系我们寻求帮助.", + "COMPLETE_TYPE1" => "你已注册成功. 现在可以登陆了.", + "COMPLETE_TYPE2" => "成功注册. 激活链接已经发送给 {{email}}. 激活之前无法登陆.", + "DISABLED" => "抱歉, 账户注册以禁用.", + "LOGOUT" => "抱歉, 登陆时不能注册. 请先注销.", + "WELCOME" => "注册简单快速." + ], + + "RATE_LIMIT_EXCEEDED" => "行动速度过快. 请等 {{delay}} 秒后再尝试新的操作.", + "REMEMBER_ME" => "记住我!", + "REMEMBER_ME_ON_COMPUTER" => "在此电脑上记住我 (不推荐在公共电脑上)", + + "SIGNIN" => "登陆", + "SIGNIN_OR_REGISTER" => "登陆或注册", + "SIGNUP" => "注销", + + "TOS" => "条款和说明", + "TOS_AGREEMENT" => "在 {{site_title}} 注册, 你需要接收 条款和说明.", + "TOS_FOR" => "{{title}}的条款和说明", + + "USERNAME" => [ + "@TRANSLATION" => "用户名", + + "CHOOSE" => "取一个唯一的用户名", + "INVALID" => "无效的用户名", + "IN_USE" => "用户名 {{user_name}} 已存在.", + "NOT_AVAILABLE" => "用户名 {{user_name}} 不可用. 重新选择用户名, 或者点击 '建议'." + ], + + "USER_ID_INVALID" => "请求的用户不存在.", + "USER_OR_EMAIL_INVALID" => "用户名或邮箱无效.", + "USER_OR_PASS_INVALID" => "没有发现用户或密码错误.", + + "WELCOME" => "欢迎回来, {{first_name}}" +]; diff --git a/main/app/sprinkles/account/locale/zh_CN/validate.php b/main/app/sprinkles/account/locale/zh_CN/validate.php new file mode 100755 index 0000000..3ca368a --- /dev/null +++ b/main/app/sprinkles/account/locale/zh_CN/validate.php @@ -0,0 +1,19 @@ + [ + "PASSWORD_MISMATCH" => "密码不一致.", + "USERNAME" => "用户名必须以小写字母, 数字, '.', '-', 和 '_'组成." + ] +]; diff --git a/main/app/sprinkles/account/routes/routes.php b/main/app/sprinkles/account/routes/routes.php new file mode 100755 index 0000000..8198255 --- /dev/null +++ b/main/app/sprinkles/account/routes/routes.php @@ -0,0 +1,59 @@ +group('/account', function () { + $this->get('/captcha', 'UserFrosting\Sprinkle\Account\Controller\AccountController:imageCaptcha'); + + $this->get('/check-username', 'UserFrosting\Sprinkle\Account\Controller\AccountController:checkUsername'); + + $this->get('/forgot-password', 'UserFrosting\Sprinkle\Account\Controller\AccountController:pageForgotPassword') + ->setName('forgot-password'); + + $this->get('/logout', 'UserFrosting\Sprinkle\Account\Controller\AccountController:logout') + ->add('authGuard'); + + $this->get('/resend-verification', 'UserFrosting\Sprinkle\Account\Controller\AccountController:pageResendVerification'); + + $this->get('/set-password/confirm', 'UserFrosting\Sprinkle\Account\Controller\AccountController:pageResetPassword'); + + $this->get('/set-password/deny', 'UserFrosting\Sprinkle\Account\Controller\AccountController:denyResetPassword'); + + $this->get('/register', 'UserFrosting\Sprinkle\Account\Controller\AccountController:pageRegister') + ->add('checkEnvironment') + ->setName('register'); + + $this->get('/settings', 'UserFrosting\Sprinkle\Account\Controller\AccountController:pageSettings') + ->add('authGuard'); + + $this->get('/sign-in', 'UserFrosting\Sprinkle\Account\Controller\AccountController:pageSignIn') + ->add('checkEnvironment') + ->setName('login'); + + $this->get('/suggest-username', 'UserFrosting\Sprinkle\Account\Controller\AccountController:suggestUsername'); + + $this->get('/verify', 'UserFrosting\Sprinkle\Account\Controller\AccountController:verify'); + + $this->post('/forgot-password', 'UserFrosting\Sprinkle\Account\Controller\AccountController:forgotPassword'); + + $this->post('/login', 'UserFrosting\Sprinkle\Account\Controller\AccountController:login'); + + $this->post('/register', 'UserFrosting\Sprinkle\Account\Controller\AccountController:register'); + + $this->post('/resend-verification', 'UserFrosting\Sprinkle\Account\Controller\AccountController:resendVerification'); + + $this->post('/set-password', 'UserFrosting\Sprinkle\Account\Controller\AccountController:setPassword'); + + $this->post('/settings', 'UserFrosting\Sprinkle\Account\Controller\AccountController:settings') + ->add('authGuard') + ->setName('settings'); + + $this->post('/settings/profile', 'UserFrosting\Sprinkle\Account\Controller\AccountController:profile') + ->add('authGuard'); +}); + +$app->get('/modals/account/tos', 'UserFrosting\Sprinkle\Account\Controller\AccountController:getModalAccountTos'); diff --git a/main/app/sprinkles/account/schema/requests/account-settings.yaml b/main/app/sprinkles/account/schema/requests/account-settings.yaml new file mode 100755 index 0000000..4a2d368 --- /dev/null +++ b/main/app/sprinkles/account/schema/requests/account-settings.yaml @@ -0,0 +1,35 @@ +--- +passwordcheck: + validators: + required: + message: PASSWORD.CONFIRM_CURRENT +email: + validators: + required: + label: "&EMAIL" + message: VALIDATE.REQUIRED + length: + label: "&EMAIL" + min: 1 + max: 150 + message: VALIDATE.LENGTH_RANGE + email: + message: VALIDATE.INVALID_EMAIL +password: + validators: + length: + label: "&PASSWORD" + min: 12 + max: 100 + message: VALIDATE.LENGTH_RANGE +passwordc: + validators: + matches: + field: password + label: "&PASSWORD.CONFIRM" + message: VALIDATE.PASSWORD_MISMATCH + length: + label: "&PASSWORD.CONFIRM" + min: 12 + max: 100 + message: VALIDATE.LENGTH_RANGE diff --git a/main/app/sprinkles/account/schema/requests/account-verify.yaml b/main/app/sprinkles/account/schema/requests/account-verify.yaml new file mode 100755 index 0000000..01f3155 --- /dev/null +++ b/main/app/sprinkles/account/schema/requests/account-verify.yaml @@ -0,0 +1,6 @@ +--- +token: + validators: + required: + label: validation token + message: VALIDATION.REQUIRED diff --git a/main/app/sprinkles/account/schema/requests/check-username.yaml b/main/app/sprinkles/account/schema/requests/check-username.yaml new file mode 100755 index 0000000..778b5e5 --- /dev/null +++ b/main/app/sprinkles/account/schema/requests/check-username.yaml @@ -0,0 +1,17 @@ +--- +user_name: + validators: + length: + label: "&USERNAME" + min: 1 + max: 50 + message: VALIDATE.LENGTH_RANGE + no_leading_whitespace: + label: "&USERNAME" + message: VALIDATE.NO_LEAD_WS + no_trailing_whitespace: + label: "&USERNAME" + message: VALIDATE.NO_TRAIL_WS + username: + label: "&USERNAME" + message: VALIDATE.USERNAME diff --git a/main/app/sprinkles/account/schema/requests/deny-password.yaml b/main/app/sprinkles/account/schema/requests/deny-password.yaml new file mode 100755 index 0000000..3b3e919 --- /dev/null +++ b/main/app/sprinkles/account/schema/requests/deny-password.yaml @@ -0,0 +1,5 @@ +--- +token: + validators: + required: + message: PASSWORD.FORGET.INVALID diff --git a/main/app/sprinkles/account/schema/requests/forgot-password.yaml b/main/app/sprinkles/account/schema/requests/forgot-password.yaml new file mode 100755 index 0000000..70072b5 --- /dev/null +++ b/main/app/sprinkles/account/schema/requests/forgot-password.yaml @@ -0,0 +1,6 @@ +--- +email: + validators: + required: + label: "&EMAIL" + message: VALIDATE.REQUIRED diff --git a/main/app/sprinkles/account/schema/requests/login.yaml b/main/app/sprinkles/account/schema/requests/login.yaml new file mode 100755 index 0000000..b78596a --- /dev/null +++ b/main/app/sprinkles/account/schema/requests/login.yaml @@ -0,0 +1,19 @@ +--- +user_name: + validators: + required: + label: "&USERNAME" + message: VALIDATE.REQUIRED + no_leading_whitespace: + label: "&USERNAME" + message: VALIDATE.NO_LEAD_WS + no_trailing_whitespace: + label: "&USERNAME" + message: VALIDATE.NO_TRAIL_WS +password: + validators: + required: + label: "&PASSWORD" + message: VALIDATE.REQUIRED +rememberme: + default: false diff --git a/main/app/sprinkles/account/schema/requests/profile-settings.yaml b/main/app/sprinkles/account/schema/requests/profile-settings.yaml new file mode 100755 index 0000000..c2b5ee8 --- /dev/null +++ b/main/app/sprinkles/account/schema/requests/profile-settings.yaml @@ -0,0 +1,24 @@ +--- +first_name: + validators: + length: + label: "&FIRST_NAME" + min: 1 + max: 20 + message: VALIDATE.LENGTH_RANGE + required: + label: "&FIRST_NAME" + message: VALIDATE.REQUIRED +last_name: + validators: + length: + label: "&LAST_NAME" + min: 1 + max: 30 + message: VALIDATE.LENGTH_RANGE +locale: + validators: + required: + label: "&LOCALE" + domain: server + message: VALIDATE.REQUIRED diff --git a/main/app/sprinkles/account/schema/requests/register.yaml b/main/app/sprinkles/account/schema/requests/register.yaml new file mode 100755 index 0000000..75dae59 --- /dev/null +++ b/main/app/sprinkles/account/schema/requests/register.yaml @@ -0,0 +1,75 @@ +--- +user_name: + validators: + length: + label: "&USERNAME" + min: 1 + max: 50 + message: VALIDATE.LENGTH_RANGE + no_leading_whitespace: + label: "&USERNAME" + message: VALIDATE.NO_LEAD_WS + no_trailing_whitespace: + label: "&USERNAME" + message: VALIDATE.NO_TRAIL_WS + required: + label: "&USERNAME" + message: VALIDATE.REQUIRED + username: + label: "&USERNAME" + message: VALIDATE.USERNAME +first_name: + validators: + length: + label: "&FIRST_NAME" + min: 1 + max: 20 + message: VALIDATE.LENGTH_RANGE + required: + label: "&FIRST_NAME" + message: VALIDATE.REQUIRED +last_name: + validators: + length: + label: "&LAST_NAME" + min: 1 + max: 30 + message: VALIDATE.LENGTH_RANGE +email: + validators: + required: + label: "&EMAIL" + message: VALIDATE.REQUIRED + length: + label: "&EMAIL" + min: 1 + max: 150 + message: VALIDATE.LENGTH_RANGE + email: + message: VALIDATE.INVALID_EMAIL +password: + validators: + required: + label: "&PASSWORD" + message: VALIDATE.REQUIRED + length: + label: "&PASSWORD" + min: 12 + max: 100 + message: VALIDATE.LENGTH_RANGE +passwordc: + validators: + required: + label: "&PASSWORD.CONFIRM" + message: VALIDATE.REQUIRED + matches: + field: password + label: "&PASSWORD.CONFIRM" + message: VALIDATE.PASSWORD_MISMATCH + length: + label: "&PASSWORD.CONFIRM" + min: 12 + max: 100 + message: VALIDATE.LENGTH_RANGE +captcha: + validators: diff --git a/main/app/sprinkles/account/schema/requests/resend-verification.yaml b/main/app/sprinkles/account/schema/requests/resend-verification.yaml new file mode 100755 index 0000000..70072b5 --- /dev/null +++ b/main/app/sprinkles/account/schema/requests/resend-verification.yaml @@ -0,0 +1,6 @@ +--- +email: + validators: + required: + label: "&EMAIL" + message: VALIDATE.REQUIRED diff --git a/main/app/sprinkles/account/schema/requests/set-password.yaml b/main/app/sprinkles/account/schema/requests/set-password.yaml new file mode 100755 index 0000000..ae59d1c --- /dev/null +++ b/main/app/sprinkles/account/schema/requests/set-password.yaml @@ -0,0 +1,29 @@ +--- +password: + validators: + required: + label: "&PASSWORD" + message: VALIDATE.REQUIRED + length: + label: "&PASSWORD" + min: 12 + max: 100 + message: VALIDATE.LENGTH_RANGE +passwordc: + validators: + required: + label: "&PASSWORD.CONFIRM" + message: VALIDATE.REQUIRED + matches: + field: password + label: "&PASSWORD.CONFIRM" + message: VALIDATE.PASSWORD_MISMATCH + length: + label: "&PASSWORD.CONFIRM" + min: 12 + max: 100 + message: VALIDATE.LENGTH_RANGE +token: + validators: + required: + message: PASSWORD.FORGET.INVALID diff --git a/main/app/sprinkles/account/src/Account.php b/main/app/sprinkles/account/src/Account.php new file mode 100755 index 0000000..49c2de9 --- /dev/null +++ b/main/app/sprinkles/account/src/Account.php @@ -0,0 +1,20 @@ +authenticator = $authenticator; + } + + /** + * Invoke the AuthGuard middleware, throwing an exception if there is no authenticated user in the session. + * + * @param \Psr\Http\Message\ServerRequestInterface $request PSR7 request + * @param \Psr\Http\Message\ResponseInterface $response PSR7 response + * @param callable $next Next middleware + * + * @return \Psr\Http\Message\ResponseInterface + */ + public function __invoke($request, $response, $next) + { + if (!$this->authenticator->check()) { + throw new AuthExpiredException(); + } else { + return $next($request, $response); + } + + return $response; + } +} diff --git a/main/app/sprinkles/account/src/Authenticate/Authenticator.php b/main/app/sprinkles/account/src/Authenticate/Authenticator.php new file mode 100755 index 0000000..5fb8920 --- /dev/null +++ b/main/app/sprinkles/account/src/Authenticate/Authenticator.php @@ -0,0 +1,419 @@ +classMapper = $classMapper; + $this->session = $session; + $this->config = $config; + $this->cache = $cache; + + // Initialize RememberMe storage + $this->rememberMeStorage = new RememberMePDO($this->config['remember_me.table']); + + // Get the actual PDO instance from Eloquent + $pdo = Capsule::connection()->getPdo(); + + $this->rememberMeStorage->setConnection($pdo); + + // Set up RememberMe + $this->rememberMe = new RememberMe($this->rememberMeStorage); + // Set cookie name + $cookieName = $this->config['session.name'] . '-' . $this->config['remember_me.cookie.name']; + $this->rememberMe->getCookie()->setName($cookieName); + + // Change cookie path + $this->rememberMe->getCookie()->setPath($this->config['remember_me.session.path']); + + // Set expire time, if specified + if ($this->config->has('remember_me.expire_time') && ($this->config->has('remember_me.expire_time') != null)) { + $this->rememberMe->getCookie()->setExpireTime($this->config['remember_me.expire_time']); + } + + $this->user = null; + + $this->viaRemember = false; + } + + /** + * Attempts to authenticate a user based on a supplied identity and password. + * + * If successful, the user's id is stored in session. + */ + public function attempt($identityColumn, $identityValue, $password, $rememberMe = false) + { + // Try to load the user, using the specified conditions + $user = $this->classMapper->staticMethod('user', 'where', $identityColumn, $identityValue)->first(); + + if (!$user) { + throw new InvalidCredentialsException(); + } + + // Check that the user has a password set (so, rule out newly created accounts without a password) + if (!$user->password) { + throw new InvalidCredentialsException(); + } + + // Check that the user's account is enabled + if ($user->flag_enabled == 0) { + throw new AccountDisabledException(); + } + + // Check that the user's account is verified (if verification is required) + if ($this->config['site.registration.require_email_verification'] && $user->flag_verified == 0) { + throw new AccountNotVerifiedException(); + } + + // Here is my password. May I please assume the identify of this user now? + if (Password::verify($password, $user->password)) { + $this->login($user, $rememberMe); + return $user; + } else { + // We know the password is at fault here (as opposed to the identity), but lets not give away the combination in case of someone bruteforcing + throw new InvalidCredentialsException(); + } + } + + /** + * Determine if the current user is authenticated. + * + * @return bool + */ + public function check() + { + return !is_null($this->user()); + } + + /** + * Determine if the current user is a guest (unauthenticated). + * + * @return bool + */ + public function guest() + { + return !$this->check(); + } + + /** + * Process an account login request. + * + * This method logs in the specified user, allowing the client to assume the user's identity for the duration of the session. + * @param User $user The user to log in. + * @param bool $rememberMe Set to true to make this a "persistent session", i.e. one that will re-login even after the session expires. + * @todo Figure out a way to update the currentUser service to reflect the logged-in user *immediately* in the service provider. + * As it stands, the currentUser service will still reflect a "guest user" for the remainder of the request. + */ + public function login($user, $rememberMe = false) + { + $oldId = session_id(); + $this->session->regenerateId(true); + + // Since regenerateId deletes the old session, we'll do the same in cache + $this->flushSessionCache($oldId); + + // If the user wants to be remembered, create Rememberme cookie + if ($rememberMe) { + $this->rememberMe->createCookie($user->id); + } else { + $this->rememberMe->clearCookie(); + } + + // Assume identity + $key = $this->config['session.keys.current_user_id']; + $this->session[$key] = $user->id; + + // Set auth mode + $this->viaRemember = false; + + // User login actions + $user->onLogin(); + } + + /** + * Processes an account logout request. + * + * Logs the currently authenticated user out, destroying the PHP session and clearing the persistent session. + * This can optionally remove persistent sessions across all browsers/devices, since there can be a "RememberMe" cookie + * and corresponding database entries in multiple browsers/devices. See http://jaspan.com/improved_persistent_login_cookie_best_practice. + * + * @param bool $complete If set to true, will ensure that the user is logged out from *all* browsers on all devices. + */ + public function logout($complete = false) + { + $currentUserId = $this->session->get($this->config['session.keys.current_user_id']); + + // This removes all of the user's persistent logins from the database + if ($complete) { + $this->storage->cleanAllTriplets($currentUserId); + } + + // Clear the rememberMe cookie + $this->rememberMe->clearCookie(); + + // User logout actions + if ($currentUserId) { + $currentUser = $this->classMapper->staticMethod('user', 'find', $currentUserId); + if ($currentUser) { + $currentUser->onLogout(); + } + } + + $this->user = null; + $this->loggedOut = true; + + $oldId = session_id(); + + // Completely destroy the session + $this->session->destroy(); + + // Since regenerateId deletes the old session, we'll do the same in cache + $this->flushSessionCache($oldId); + + // Restart the session service + $this->session->start(); + } + + /** + * Try to get the currently authenticated user, returning a guest user if none was found. + * + * Tries to re-establish a session for "remember-me" users who have been logged out due to an expired session. + * @return User|null + * @throws AuthExpiredException + * @throws AuthCompromisedException + * @throws AccountInvalidException + * @throws AccountDisabledException + */ + public function user() + { + $user = null; + + if (!$this->loggedOut) { + + // Return any cached user + if (!is_null($this->user)) { + return $this->user; + } + + // If this throws a PDOException we catch it and return null than allowing the exception to propagate. + // This is because the error handler relies on Twig, which relies on a Twig Extension, which relies on the global current_user variable. + // So, we really don't want this method to throw any database exceptions. + try { + // Now, check to see if we have a user in session + $user = $this->loginSessionUser(); + + // If no user was found in the session, try to login via RememberMe cookie + if (!$user) { + $user = $this->loginRememberedUser(); + } + } catch (\PDOException $e) { + $user = null; + } + } + + return $this->user = $user; + } + + /** + * Determine whether the current user was authenticated using a remember me cookie. + * + * This function is useful when users are performing sensitive operations, and you may want to force them to re-authenticate. + * @return bool + */ + public function viaRemember() + { + return $this->viaRemember; + } + + /** + * Attempt to log in the client from their rememberMe token (in their cookie). + * + * @return User|bool If successful, the User object of the remembered user. Otherwise, return false. + * @throws AuthCompromisedException The client attempted to log in with an invalid rememberMe token. + */ + protected function loginRememberedUser() + { + /** @var \Birke\Rememberme\LoginResult $loginResult */ + $loginResult = $this->rememberMe->login(); + + if ($loginResult->isSuccess()) { + // Update in session + $this->session[$this->config['session.keys.current_user_id']] = $loginResult->getCredential(); + // There is a chance that an attacker has stolen the login token, + // so we store the fact that the user was logged in via RememberMe (instead of login form) + $this->viaRemember = true; + } else { + // If $rememberMe->login() was not successfull, check if the token was invalid as well. This means the cookie was stolen. + if ($loginResult->hasPossibleManipulation()) { + throw new AuthCompromisedException(); + } + } + + return $this->validateUserAccount($loginResult->getCredential()); + } + + /** + * Attempt to log in the client from the session. + * + * @return User|null If successful, the User object of the user in session. Otherwise, return null. + * @throws AuthExpiredException The client attempted to use an expired rememberMe token. + */ + protected function loginSessionUser() + { + $userId = $this->session->get($this->config['session.keys.current_user_id']); + + // If a user_id was found in the session, check any rememberMe cookie that was submitted. + // If they submitted an expired rememberMe cookie, then we need to log them out. + if ($userId) { + if (!$this->validateRememberMeCookie()) { + $this->logout(); + throw new AuthExpiredException(); + } + } + + return $this->validateUserAccount($userId); + } + + /** + * Determine if the cookie contains a valid rememberMe token. + * + * @return bool + */ + protected function validateRememberMeCookie() + { + $cookieValue = $this->rememberMe->getCookie()->getValue(); + if (!$cookieValue) { + return true; + } + $triplet = RememberMeTriplet::fromString($cookieValue); + if (!$triplet->isValid()) { + return false; + } + + return true; + } + + /** + * Tries to load the specified user by id from the database. + * + * Checks that the account is valid and enabled, throwing an exception if not. + * @param int $userId + * @return User|null + * @throws AccountInvalidException + * @throws AccountDisabledException + */ + protected function validateUserAccount($userId) + { + if ($userId) { + $user = $this->classMapper->staticMethod('user', 'find', $userId); + + // If the user doesn't exist any more, throw an exception. + if (!$user) { + throw new AccountInvalidException(); + } + + // If the user has been disabled since their last request, throw an exception. + if (!$user->flag_enabled) { + throw new AccountDisabledException(); + } + + return $user; + } else { + return null; + } + } + + /** + * Flush the cache associated with a session id + * + * @param string $id The session id + * @return bool + */ + public function flushSessionCache($id) + { + return $this->cache->tags('_s' . $id)->flush(); + } +} diff --git a/main/app/sprinkles/account/src/Authenticate/Exception/AccountDisabledException.php b/main/app/sprinkles/account/src/Authenticate/Exception/AccountDisabledException.php new file mode 100755 index 0000000..e79ceb5 --- /dev/null +++ b/main/app/sprinkles/account/src/Authenticate/Exception/AccountDisabledException.php @@ -0,0 +1,21 @@ + $this->cost($options), + ]); + + if (!$hash) { + throw new HashFailedException(); + } + + return $hash; + } + + /** + * Verify a plaintext password against the user's hashed password. + * + * @param string $password The plaintext password to verify. + * @param string $hash The hash to compare against. + * @param array $options + * @return boolean True if the password matches, false otherwise. + */ + public function verify($password, $hash, array $options = []) + { + $hashType = $this->getHashType($hash); + + if ($hashType == 'sha1') { + // Legacy UserCake passwords + $salt = substr($hash, 0, 25); // Extract the salt from the hash + $inputHash = $salt . sha1($salt . $password); + + return (hash_equals($inputHash, $hash) === true); + + } elseif ($hashType == 'legacy') { + // Homegrown implementation (assuming that current install has been using a cost parameter of 12) + // Used for manual implementation of bcrypt. + // Note that this legacy hashing put the salt at the _end_ for some reason. + $salt = substr($hash, 60); + $inputHash = crypt($password, '$2y$12$' . $salt); + $correctHash = substr($hash, 0, 60); + + return (hash_equals($inputHash, $correctHash) === true); + } + + // Modern implementation + return password_verify($password, $hash); + } + + /** + * Extract the cost value from the options array. + * + * @param array $options + * @return int + */ + protected function cost(array $options = []) + { + return isset($options['rounds']) ? $options['rounds'] : $this->defaultRounds; + } +} diff --git a/main/app/sprinkles/account/src/Authorize/AccessConditionExpression.php b/main/app/sprinkles/account/src/Authorize/AccessConditionExpression.php new file mode 100755 index 0000000..dd5647e --- /dev/null +++ b/main/app/sprinkles/account/src/Authorize/AccessConditionExpression.php @@ -0,0 +1,139 @@ +nodeVisitor = $nodeVisitor; + $this->user = $user; + $this->parser = new Parser(new EmulativeLexer); + $this->traverser = new NodeTraverser; + $this->traverser->addVisitor($nodeVisitor); + $this->prettyPrinter = new StandardPrettyPrinter; + $this->logger = $logger; + $this->debug = $debug; + } + + /** + * Evaluates a condition expression, based on the given parameters. + * + * The special parameter `self` is an array of the current user's data. + * This get included automatically, and so does not need to be passed in. + * @param string $condition a boolean expression composed of calls to AccessCondition functions. + * @param array[mixed] $params the parameters to be used when evaluating the expression. + * @return bool true if the condition is passed for the given parameters, otherwise returns false. + */ + public function evaluateCondition($condition, $params) + { + // Set the reserved `self` parameters. + // This replaces any values of `self` specified in the arguments, thus preventing them from being overridden in malicious user input. + // (For example, from an unfiltered request body). + $params['self'] = $this->user->export(); + + $this->nodeVisitor->setParams($params); + + $code = "debug) { + $this->logger->debug("Evaluating access condition '$condition' with parameters:", $params); + } + + // Traverse the parse tree, and execute any callbacks found using the supplied parameters. + // Replace the function node with the return value of the callback. + try { + // parse + $stmts = $this->parser->parse($code); + + // traverse + $stmts = $this->traverser->traverse($stmts); + + // Evaluate boolean statement. It is safe to use eval() here, because our expression has been reduced entirely to a boolean expression. + $expr = $this->prettyPrinter->prettyPrintExpr($stmts[0]); + $expr_eval = "return " . $expr . ";\n"; + $result = eval($expr_eval); + + if ($this->debug) { + $this->logger->debug("Expression '$expr' evaluates to " . ($result == true ? "true" : "false")); + } + + return $result; + } catch (PhpParserException $e) { + if ($this->debug) { + $this->logger->debug("Error parsing access condition '$condition':" . $e->getMessage()); + } + return false; // Access fails if the access condition can't be parsed. + } catch (AuthorizationException $e) { + if ($this->debug) { + $this->logger->debug("Error parsing access condition '$condition':" . $e->getMessage()); + } + return false; + } + } +} diff --git a/main/app/sprinkles/account/src/Authorize/AuthorizationException.php b/main/app/sprinkles/account/src/Authorize/AuthorizationException.php new file mode 100755 index 0000000..251b67f --- /dev/null +++ b/main/app/sprinkles/account/src/Authorize/AuthorizationException.php @@ -0,0 +1,23 @@ +ci = $ci; + $this->callbacks = $callbacks; + } + + /** + * Register an authorization callback, which can then be used in permission conditions. + * + * To add additional callbacks, simply extend the `authorizer` service in your Sprinkle's service provider. + * @param string $name + * @param callable $callback + */ + public function addCallback($name, $callback) + { + $this->callbacks[$name] = $callback; + return $this; + } + + /** + * Get all authorization callbacks. + * + * @return callable[] + */ + public function getCallbacks() + { + return $this->callbacks; + } + + /** + * Checks whether or not a user has access on a particular permission slug. + * + * Determine if this user has access to the given $slug under the given $params. + * + * @param UserFrosting\Sprinkle\Account\Database\Models\User $user + * @param string $slug The permission slug to check for access. + * @param array $params[optional] An array of field names => values, specifying any additional data to provide the authorization module + * when determining whether or not this user has access. + * @return boolean True if the user has access, false otherwise. + */ + public function checkAccess(User $user, $slug, array $params = []) + { + $debug = $this->ci->config['debug.auth']; + + if ($debug) { + $trace = array_slice(debug_backtrace(DEBUG_BACKTRACE_IGNORE_ARGS, 3), 1); + $this->ci->authLogger->debug("Authorization check requested at: ", $trace); + $this->ci->authLogger->debug("Checking authorization for user {$user->id} ('{$user->user_name}') on permission '$slug'..."); + } + + if ($this->ci->authenticator->guest()) { + if ($debug) { + $this->ci->authLogger->debug("User is not logged in. Access denied."); + } + return false; + } + + // The master (root) account has access to everything. + // Need to use loose comparison for now, because some DBs return `id` as a string. + + if ($user->id == $this->ci->config['reserved_user_ids.master']) { + if ($debug) { + $this->ci->authLogger->debug("User is the master (root) user. Access granted."); + } + return true; + } + + // Find all permissions that apply to this user (via roles), and check if any evaluate to true. + $permissions = $user->getCachedPermissions(); + + if (empty($permissions) || !isset($permissions[$slug])) { + if ($debug) { + $this->ci->authLogger->debug("No matching permissions found. Access denied."); + } + return false; + } + + $permissions = $permissions[$slug]; + + if ($debug) { + $this->ci->authLogger->debug("Found matching permissions: \n" . print_r($this->getPermissionsArrayDebugInfo($permissions), true)); + } + + $nodeVisitor = new ParserNodeFunctionEvaluator($this->callbacks, $this->ci->authLogger, $debug); + $ace = new AccessConditionExpression($nodeVisitor, $user, $this->ci->authLogger, $debug); + + foreach ($permissions as $permission) { + $pass = $ace->evaluateCondition($permission->conditions, $params); + if ($pass) { + if ($debug) { + $this->ci->authLogger->debug("User passed conditions '{$permission->conditions}' . Access granted."); + } + return true; + } + } + + if ($debug) { + $this->ci->authLogger->debug("User failed to pass any of the matched permissions. Access denied."); + } + + return false; + } + + /** + * Remove extraneous information from the permission to reduce verbosity. + * + * @param array + * @return array + */ + protected function getPermissionsArrayDebugInfo($permissions) + { + $permissionsInfo = []; + foreach ($permissions as $permission) { + $permissionData = array_only($permission->toArray(), ['id', 'slug', 'name', 'conditions', 'description']); + // Remove this until we can find an efficient way to only load these once during debugging + //$permissionData['roles_via'] = $permission->roles_via->pluck('id')->all(); + $permissionsInfo[] = $permissionData; + } + + return $permissionsInfo; + } +} diff --git a/main/app/sprinkles/account/src/Authorize/ParserNodeFunctionEvaluator.php b/main/app/sprinkles/account/src/Authorize/ParserNodeFunctionEvaluator.php new file mode 100755 index 0000000..e8e5cde --- /dev/null +++ b/main/app/sprinkles/account/src/Authorize/ParserNodeFunctionEvaluator.php @@ -0,0 +1,193 @@ +callbacks = $callbacks; + $this->prettyPrinter = new StandardPrettyPrinter; + $this->logger = $logger; + $this->debug = $debug; + $this->params = []; + } + + public function leaveNode(Node $node) + { + // Look for function calls + if ($node instanceof \PhpParser\Node\Expr\FuncCall) { + $eval = new \PhpParser\Node\Scalar\LNumber; + + // Get the method name + $callbackName = $node->name->toString(); + // Get the method arguments + $argNodes = $node->args; + + $args = []; + $argsInfo = []; + foreach ($argNodes as $arg) { + $argString = $this->prettyPrinter->prettyPrintExpr($arg->value); + + // Debugger info + $currentArgInfo = [ + 'expression' => $argString + ]; + // Resolve parameter placeholders ('variable' names (either single-word or array-dot identifiers)) + if (($arg->value instanceof \PhpParser\Node\Expr\BinaryOp\Concat) || ($arg->value instanceof \PhpParser\Node\Expr\ConstFetch)) { + $value = $this->resolveParamPath($argString); + $currentArgInfo['type'] = "parameter"; + $currentArgInfo['resolved_value'] = $value; + // Resolve arrays + } elseif ($arg->value instanceof \PhpParser\Node\Expr\Array_) { + $value = $this->resolveArray($arg); + $currentArgInfo['type'] = "array"; + $currentArgInfo['resolved_value'] = print_r($value, true); + // Resolve strings + } elseif ($arg->value instanceof \PhpParser\Node\Scalar\String_) { + $value = $arg->value->value; + $currentArgInfo['type'] = "string"; + $currentArgInfo['resolved_value'] = $value; + // Resolve numbers + } elseif ($arg->value instanceof \PhpParser\Node\Scalar\DNumber) { + $value = $arg->value->value; + $currentArgInfo['type'] = "float"; + $currentArgInfo['resolved_value'] = $value; + } elseif ($arg->value instanceof \PhpParser\Node\Scalar\LNumber) { + $value = $arg->value->value; + $currentArgInfo['type'] = "integer"; + $currentArgInfo['resolved_value'] = $value; + // Anything else is simply interpreted as its literal string value + } else { + $value = $argString; + $currentArgInfo['type'] = "unknown"; + $currentArgInfo['resolved_value'] = $value; + } + + $args[] = $value; + $argsInfo[] = $currentArgInfo; + } + + if ($this->debug) { + if (count($args)) { + $this->logger->debug("Evaluating callback '$callbackName' on: ", $argsInfo); + } else { + $this->logger->debug("Evaluating callback '$callbackName'..."); + } + } + + // Call the specified access condition callback with the specified arguments. + if (isset($this->callbacks[$callbackName]) && is_callable($this->callbacks[$callbackName])) { + $result = call_user_func_array($this->callbacks[$callbackName], $args); + } else { + throw new AuthorizationException("Authorization failed: Access condition method '$callbackName' does not exist."); + } + + if ($this->debug) { + $this->logger->debug("Result: " . ($result ? "1" : "0")); + } + + return new \PhpParser\Node\Scalar\LNumber($result ? "1" : "0"); + } + } + + public function setParams($params) + { + $this->params = $params; + } + + /** + * Resolve an array expression in a condition expression into an actual array. + * + * @param string $arg the array, represented as a string. + * @return array[mixed] the array, as a plain ol' PHP array. + */ + private function resolveArray($arg) + { + $arr = []; + $items = (array) $arg->value->items; + foreach ($items as $item) { + if ($item->key) { + $arr[$item->key] = $item->value->value; + } else { + $arr[] = $item->value->value; + } + } + return $arr; + } + + /** + * Resolve a parameter path (e.g. "user.id", "post", etc) into its value. + * + * @param string $path the name of the parameter to resolve, based on the parameters set in this object. + * @throws Exception the path could not be resolved. Path is malformed or key does not exist. + * @return mixed the value of the specified parameter. + */ + private function resolveParamPath($path) + { + $pathTokens = explode(".", $path); + $value = $this->params; + foreach ($pathTokens as $token) { + $token = trim($token); + if (is_array($value) && isset($value[$token])) { + $value = $value[$token]; + continue; + } elseif (is_object($value) && isset($value->$token)) { + $value = $value->$token; + continue; + } else { + throw new AuthorizationException("Cannot resolve the path \"$path\". Error at token \"$token\"."); + } + } + return $value; + } +} diff --git a/main/app/sprinkles/account/src/Bakery/CreateAdminUser.php b/main/app/sprinkles/account/src/Bakery/CreateAdminUser.php new file mode 100755 index 0000000..cfaacef --- /dev/null +++ b/main/app/sprinkles/account/src/Bakery/CreateAdminUser.php @@ -0,0 +1,334 @@ +setName("create-admin") + ->setDescription("Create the initial admin (root) user account"); + } + + /** + * {@inheritDoc} + */ + protected function execute(InputInterface $input, OutputInterface $output) + { + $this->io->title("Root account setup"); + + // Need the database + try { + $this->io->writeln("Testing database connection", OutputInterface::VERBOSITY_VERBOSE); + $this->testDB(); + $this->io->writeln("Ok", OutputInterface::VERBOSITY_VERBOSE); + } catch (\Exception $e) { + $this->io->error($e->getMessage()); + exit(1); + } + + // Need migration table + if (!Capsule::schema()->hasColumn('migrations', 'id')) { + $this->io->error("Migrations doesn't appear to have been run! Make sure the database is properly migrated by using the `php bakery migrate` command."); + exit(1); + } + + // Make sure the required mirgations have been run + foreach ($this->dependencies as $migration) { + if (!Migrations::where('migration', $migration)->exists()) { + $this->io->error("Migration `$migration` doesn't appear to have been run! Make sure all migrations are up to date by using the `php bakery migrate` command."); + exit(1); + } + } + + // Make sure that there are no users currently in the user table + // We setup the root account here so it can be done independent of the version check + if (User::count() > 0) { + + $this->io->note("Table 'users' is not empty. Skipping root account setup. To set up the root account again, please truncate or drop the table and try again."); + + } else { + + $this->io->writeln("Please answer the following questions to create the root account:\n"); + + // Get the account details + $userName = $this->askUsername(); + $email = $this->askEmail(); + $firstName = $this->askFirstName(); + $lastName = $this->askLastName(); + $password = $this->askPassword(); + + // Ok, now we've got the info and we can create the new user. + $this->io->write("\nSaving the root user details..."); + $rootUser = new User([ + "user_name" => $userName, + "email" => $email, + "first_name" => $firstName, + "last_name" => $lastName, + "password" => Password::hash($password) + ]); + + $rootUser->save(); + + $defaultRoles = [ + 'user' => Role::where('slug', 'user')->first(), + 'group-admin' => Role::where('slug', 'group-admin')->first(), + 'site-admin' => Role::where('slug', 'site-admin')->first() + ]; + + foreach ($defaultRoles as $slug => $role) { + if ($role) { + $rootUser->roles()->attach($role->id); + } + } + + $this->io->success("Root user creation successful!"); + } + } + + /** + * Ask for the username + * + * @access protected + * @return void + */ + protected function askUsername() + { + while (!isset($userName) || !$this->validateUsername($userName)) { + $userName = $this->io->ask("Choose a root username (1-50 characters, no leading or trailing whitespace)"); + } + return $userName; + } + + /** + * Validate the username. + * + * @access protected + * @param mixed $userName + * @return void + */ + protected function validateUsername($userName) + { + // Validate length + if (strlen($userName) < 1 || strlen($userName) > 50) { + $this->io->error("Username must be between 1-50 characters"); + return false; + } + + // Validate format + $options = [ + 'options' => [ + 'regexp' => "/^\S((.*\S)|)$/" + ] + ]; + $validate = filter_var($userName, FILTER_VALIDATE_REGEXP, $options); + if (!$validate) { + $this->io->error("Username can't have any leading or trailing whitespace"); + return false; + } + + return true; + } + + /** + * Ask for the email + * + * @access protected + * @return void + */ + protected function askEmail() + { + while (!isset($email) || !$this->validateEmail($email)) { + $email = $this->io->ask("Enter a valid email address (1-254 characters, must be compatible with FILTER_VALIDATE_EMAIL)"); + } + return $email; + } + + /** + * Validate the email. + * + * @access protected + * @param mixed $email + * @return void + */ + protected function validateEmail($email) + { + // Validate length + if (strlen($email) < 1 || strlen($email) > 254) { + $this->io->error("Email must be between 1-254 characters"); + return false; + } + + // Validate format + if (!filter_var($email, FILTER_VALIDATE_EMAIL)) { + $this->io->error("Email must be compatible with FILTER_VALIDATE_EMAIL"); + return false; + } + + return true; + } + + /** + * Ask for the first name + * + * @access protected + * @return void + */ + protected function askFirstName() + { + while (!isset($firstName) || !$this->validateFirstName($firstName)) { + $firstName = $this->io->ask("Enter the user first name (1-20 characters)"); + } + return $firstName; + } + + /** + * validateFirstName function. + * + * @access protected + * @param mixed $name + * @return void + */ + protected function validateFirstName($firstName) + { + // Validate length + if (strlen($firstName) < 1 || strlen($firstName) > 20) { + $this->io->error("First name must be between 1-20 characters"); + return false; + } + + return true; + } + + /** + * Ask for the last name + * + * @access protected + * @return void + */ + protected function askLastName() + { + while (!isset($lastName) || !$this->validateLastName($lastName)) { + $lastName = $this->io->ask("Enter the user last name (1-30 characters)"); + } + return $lastName; + } + + /** + * validateLastName function. + * + * @access protected + * @param mixed $lastName + * @return void + */ + protected function validateLastName($lastName) + { + // Validate length + if (strlen($lastName) < 1 || strlen($lastName) > 30) { + $this->io->error("Last name must be between 1-30 characters"); + return false; + } + + return true; + } + + /** + * Ask for the password + * + * @access protected + * @return void + */ + protected function askPassword() + { + while (!isset($password) || !$this->validatePassword($password) || !$this->confirmPassword($password)) { + $password = $this->io->askHidden("Enter password (12-255 characters)"); + } + return $password; + } + + /** + * validatePassword function. + * + * @access protected + * @param mixed $password + * @return void + */ + protected function validatePassword($password) + { + if (strlen($password) < 12 || strlen($password) > 255) { + $this->io->error("Password must be between 12-255 characters"); + return false; + } + + return true; + } + + /** + * confirmPassword function. + * + * @access protected + * @param mixed $passwordToConfirm + * @return void + */ + protected function confirmPassword($passwordToConfirm) + { + while (!isset($password)) { + $password = $this->io->askHidden("Please re-enter the chosen password"); + } + return $this->validatePasswordConfirmation($password, $passwordToConfirm); + } + + /** + * validatePasswordConfirmation function. + * + * @access protected + * @param mixed $password + * @param mixed $passwordToConfirm + * @return void + */ + protected function validatePasswordConfirmation($password, $passwordToConfirm) + { + if ($password != $passwordToConfirm) { + $this->io->error("Passwords do not match, please try again."); + return false; + } + + return true; + } +} \ No newline at end of file diff --git a/main/app/sprinkles/account/src/Controller/AccountController.php b/main/app/sprinkles/account/src/Controller/AccountController.php new file mode 100755 index 0000000..ce99370 --- /dev/null +++ b/main/app/sprinkles/account/src/Controller/AccountController.php @@ -0,0 +1,1293 @@ +ci->alerts; + + // GET parameters + $params = $request->getQueryParams(); + + // Load request schema + $schema = new RequestSchema('schema://requests/check-username.yaml'); + + // Whitelist and set parameter defaults + $transformer = new RequestDataTransformer($schema); + $data = $transformer->transform($params); + + // Validate, and halt on validation errors. + $validator = new ServerSideValidator($schema, $this->ci->translator); + if (!$validator->validate($data)) { + // TODO: encapsulate the communication of error messages from ServerSideValidator to the BadRequestException + $e = new BadRequestException('Missing or malformed request data!'); + foreach ($validator->errors() as $idx => $field) { + foreach($field as $eidx => $error) { + $e->addUserMessage($error); + } + } + throw $e; + } + + /** @var \UserFrosting\Sprinkle\Core\Throttle\Throttler $throttler */ + $throttler = $this->ci->throttler; + $delay = $throttler->getDelay('check_username_request'); + + // Throttle requests + if ($delay > 0) { + return $response->withStatus(429); + } + + /** @var \UserFrosting\Sprinkle\Core\Util\ClassMapper $classMapper */ + $classMapper = $this->ci->classMapper; + + /** @var \UserFrosting\I18n\MessageTranslator $translator */ + $translator = $this->ci->translator; + + // Log throttleable event + $throttler->logEvent('check_username_request'); + + if ($classMapper->staticMethod('user', 'findUnique', $data['user_name'], 'user_name')) { + $message = $translator->translate('USERNAME.NOT_AVAILABLE', $data); + return $response->write($message)->withStatus(200); + } else { + return $response->write('true')->withStatus(200); + } + } + + /** + * Processes a request to cancel a password reset request. + * + * This is provided so that users can cancel a password reset request, if they made it in error or if it was not initiated by themselves. + * Processes the request from the password reset link, checking that: + * 1. The provided token is associated with an existing user account, who has a pending password reset request. + * Request type: GET + * + * @param Request $request + * @param Response $response + * @param array $args + * @return void + */ + public function denyResetPassword(Request $request, Response $response, $args) + { + // GET parameters + $params = $request->getQueryParams(); + + /** @var \UserFrosting\Sprinkle\Core\Alert\AlertStream $ms */ + $ms = $this->ci->alerts; + + /** @var \UserFrosting\Sprinkle\Core\Util\ClassMapper $classMapper */ + $classMapper = $this->ci->classMapper; + + $loginPage = $this->ci->router->pathFor('login'); + + // Load validation rules + $schema = new RequestSchema('schema://requests/deny-password.yaml'); + + // Whitelist and set parameter defaults + $transformer = new RequestDataTransformer($schema); + $data = $transformer->transform($params); + + // Validate, and halt on validation errors. Since this is a GET request, we need to redirect on failure + $validator = new ServerSideValidator($schema, $this->ci->translator); + if (!$validator->validate($data)) { + $ms->addValidationErrors($validator); + // 400 code + redirect is perfectly fine, according to user Dilaz in #laravel + return $response->withRedirect($loginPage, 400); + } + + $passwordReset = $this->ci->repoPasswordReset->cancel($data['token']); + + if (!$passwordReset) { + $ms->addMessageTranslated('danger', 'PASSWORD.FORGET.INVALID'); + return $response->withRedirect($loginPage, 400); + } + + $ms->addMessageTranslated('success', 'PASSWORD.FORGET.REQUEST_CANNED'); + return $response->withRedirect($loginPage); + } + + /** + * Processes a request to email a forgotten password reset link to the user. + * + * Processes the request from the form on the "forgot password" page, checking that: + * 1. The rate limit for this type of request is being observed. + * 2. The provided email address belongs to a registered account; + * 3. The submitted data is valid. + * Note that we have removed the requirement that a password reset request not already be in progress. + * This is because we need to allow users to re-request a reset, even if they lose the first reset email. + * This route is "public access". + * Request type: POST + * @todo require additional user information + * @todo prevent password reset requests for root account? + * + * @param Request $request + * @param Response $response + * @param array $args + * @return void + */ + public function forgotPassword(Request $request, Response $response, $args) + { + /** @var \UserFrosting\Sprinkle\Core\Alert\AlertStream $ms */ + $ms = $this->ci->alerts; + + /** @var \UserFrosting\Sprinkle\Core\Util\ClassMapper $classMapper */ + $classMapper = $this->ci->classMapper; + + /** @var \UserFrosting\Support\Repository\Repository $config */ + $config = $this->ci->config; + + // Get POST parameters + $params = $request->getParsedBody(); + + // Load the request schema + $schema = new RequestSchema('schema://requests/forgot-password.yaml'); + + // Whitelist and set parameter defaults + $transformer = new RequestDataTransformer($schema); + $data = $transformer->transform($params); + + // Validate, and halt on validation errors. Failed validation attempts do not count towards throttling limit. + $validator = new ServerSideValidator($schema, $this->ci->translator); + if (!$validator->validate($data)) { + $ms->addValidationErrors($validator); + return $response->withStatus(400); + } + + // Throttle requests + + /** @var \UserFrosting\Sprinkle\Core\Throttle\Throttler $throttler */ + $throttler = $this->ci->throttler; + + $throttleData = [ + 'email' => $data['email'] + ]; + $delay = $throttler->getDelay('password_reset_request', $throttleData); + + if ($delay > 0) { + $ms->addMessageTranslated('danger', 'RATE_LIMIT_EXCEEDED', ['delay' => $delay]); + return $response->withStatus(429); + } + + // All checks passed! log events/activities, update user, and send email + // Begin transaction - DB will be rolled back if an exception occurs + Capsule::transaction( function() use ($classMapper, $data, $throttler, $throttleData, $config) { + + // Log throttleable event + $throttler->logEvent('password_reset_request', $throttleData); + + // Load the user, by email address + $user = $classMapper->staticMethod('user', 'where', 'email', $data['email'])->first(); + + // Check that the email exists. + // If there is no user with that email address, we should still pretend like we succeeded, to prevent account enumeration + if ($user) { + // Try to generate a new password reset request. + // Use timeout for "reset password" + $passwordReset = $this->ci->repoPasswordReset->create($user, $config['password_reset.timeouts.reset']); + + // Create and send email + $message = new TwigMailMessage($this->ci->view, 'mail/password-reset.html.twig'); + $message->from($config['address_book.admin']) + ->addEmailRecipient(new EmailRecipient($user->email, $user->full_name)) + ->addParams([ + 'user' => $user, + 'token' => $passwordReset->getToken(), + 'request_date' => Carbon::now()->format('Y-m-d H:i:s') + ]); + + $this->ci->mailer->send($message); + } + }); + + // TODO: create delay to prevent timing-based attacks + + $ms->addMessageTranslated('success', 'PASSWORD.FORGET.REQUEST_SENT', ['email' => $data['email']]); + return $response->withStatus(200); + } + + /** + * Returns a modal containing account terms of service. + * + * This does NOT render a complete page. Instead, it renders the HTML for the form, which can be embedded in other pages. + * Request type: GET + * + * @param Request $request + * @param Response $response + * @param array $args + * @return void + */ + public function getModalAccountTos(Request $request, Response $response, $args) + { + return $this->ci->view->render($response, 'modals/tos.html.twig'); + } + + /** + * Generate a random captcha, store it to the session, and return the captcha image. + * + * Request type: GET + * + * @param Request $request + * @param Response $response + * @param array $args + * @return void + */ + public function imageCaptcha(Request $request, Response $response, $args) + { + $captcha = new Captcha($this->ci->session, $this->ci->config['session.keys.captcha']); + $captcha->generateRandomCode(); + + return $response->withStatus(200) + ->withHeader('Content-Type', 'image/png;base64') + ->write($captcha->getImage()); + } + + /** + * Processes an account login request. + * + * Processes the request from the form on the login page, checking that: + * 1. The user is not already logged in. + * 2. The rate limit for this type of request is being observed. + * 3. Email login is enabled, if an email address was used. + * 4. The user account exists. + * 5. The user account is enabled and verified. + * 6. The user entered a valid username/email and password. + * This route, by definition, is "public access". + * Request type: POST + * + * @param Request $request + * @param Response $response + * @param array $args + * @return void + */ + public function login(Request $request, Response $response, $args) + { + /** @var \UserFrosting\Sprinkle\Core\Alert\AlertStream $ms */ + $ms = $this->ci->alerts; + + /** @var \UserFrosting\Sprinkle\Account\Database\Models\User $currentUser */ + $currentUser = $this->ci->currentUser; + + /** @var \UserFrosting\Sprinkle\Account\Authenticate\Authenticator $authenticator */ + $authenticator = $this->ci->authenticator; + + // Return 200 success if user is already logged in + if ($authenticator->check()) { + $ms->addMessageTranslated('warning', 'LOGIN.ALREADY_COMPLETE'); + return $response->withStatus(200); + } + + /** @var \UserFrosting\Support\Repository\Repository $config */ + $config = $this->ci->config; + + // Get POST parameters + $params = $request->getParsedBody(); + + // Load the request schema + $schema = new RequestSchema('schema://requests/login.yaml'); + + // Whitelist and set parameter defaults + $transformer = new RequestDataTransformer($schema); + $data = $transformer->transform($params); + + // Validate, and halt on validation errors. Failed validation attempts do not count towards throttling limit. + $validator = new ServerSideValidator($schema, $this->ci->translator); + if (!$validator->validate($data)) { + $ms->addValidationErrors($validator); + return $response->withStatus(400); + } + + // Determine whether we are trying to log in with an email address or a username + $isEmail = filter_var($data['user_name'], FILTER_VALIDATE_EMAIL); + + // Throttle requests + + /** @var \UserFrosting\Sprinkle\Core\Throttle\Throttler $throttler */ + $throttler = $this->ci->throttler; + + $userIdentifier = $data['user_name']; + + $throttleData = [ + 'user_identifier' => $userIdentifier + ]; + + $delay = $throttler->getDelay('sign_in_attempt', $throttleData); + if ($delay > 0) { + $ms->addMessageTranslated('danger', 'RATE_LIMIT_EXCEEDED', [ + 'delay' => $delay + ]); + return $response->withStatus(429); + } + + // Log throttleable event + $throttler->logEvent('sign_in_attempt', $throttleData); + + // If credential is an email address, but email login is not enabled, raise an error. + // Note that we do this after logging throttle event, so this error counts towards throttling limit. + if ($isEmail && !$config['site.login.enable_email']) { + $ms->addMessageTranslated('danger', 'USER_OR_PASS_INVALID'); + return $response->withStatus(403); + } + + // Try to authenticate the user. Authenticator will throw an exception on failure. + /** @var \UserFrosting\Sprinkle\Account\Authenticate\Authenticator $authenticator */ + $authenticator = $this->ci->authenticator; + + $currentUser = $authenticator->attempt(($isEmail ? 'email' : 'user_name'), $userIdentifier, $data['password'], $data['rememberme']); + + $ms->addMessageTranslated('success', 'WELCOME', $currentUser->export()); + + // Set redirect, if relevant + $redirectOnLogin = $this->ci->get('redirect.onLogin'); + + return $redirectOnLogin($request, $response, $args); + } + + /** + * Log the user out completely, including destroying any "remember me" token. + * + * Request type: GET + * + * @param Request $request + * @param Response $response + * @param array $args + * @return void + */ + public function logout(Request $request, Response $response, $args) + { + // Destroy the session + $this->ci->authenticator->logout(); + + // Return to home page + $config = $this->ci->config; + return $response->withStatus(302)->withHeader('Location', $config['site.uri.public']); + } + + /** + * Render the "forgot password" page. + * + * This creates a simple form to allow users who forgot their password to have a time-limited password reset link emailed to them. + * By default, this is a "public page" (does not require authentication). + * Request type: GET + * + * @param Request $request + * @param Response $response + * @param array $args + * @return void + */ + public function pageForgotPassword(Request $request, Response $response, $args) + { + // Load validation rules + $schema = new RequestSchema('schema://requests/forgot-password.yaml'); + $validator = new JqueryValidationAdapter($schema, $this->ci->translator); + + return $this->ci->view->render($response, 'pages/forgot-password.html.twig', [ + 'page' => [ + 'validators' => [ + 'forgot_password' => $validator->rules('json', false) + ] + ] + ]); + } + + + /** + * Render the account registration page for UserFrosting. + * + * This allows new (non-authenticated) users to create a new account for themselves on your website (if enabled). + * By definition, this is a "public page" (does not require authentication). + * Request type: GET + * + * @param Request $request + * @param Response $response + * @param array $args + * @return void + */ + public function pageRegister(Request $request, Response $response, $args) + { + /** @var \UserFrosting\Support\Repository\Repository $config */ + $config = $this->ci->config; + + if (!$config['site.registration.enabled']) { + throw new NotFoundException($request, $response); + } + + /** @var \UserFrosting\Sprinkle\Account\Authenticate\Authenticator $authenticator */ + $authenticator = $this->ci->authenticator; + + // Redirect if user is already logged in + if ($authenticator->check()) { + $redirect = $this->ci->get('redirect.onAlreadyLoggedIn'); + + return $redirect($request, $response, $args); + } + + // Load validation rules + $schema = new RequestSchema('schema://requests/register.yaml'); + $validatorRegister = new JqueryValidationAdapter($schema, $this->ci->translator); + + return $this->ci->view->render($response, 'pages/register.html.twig', [ + 'page' => [ + 'validators' => [ + 'register' => $validatorRegister->rules('json', false) + ] + ] + ]); + } + + /** + * Render the "resend verification email" page. + * + * This is a form that allows users who lost their account verification link to have the link resent to their email address. + * By default, this is a "public page" (does not require authentication). + * Request type: GET + * + * @param Request $request + * @param Response $response + * @param array $args + * @return void + */ + public function pageResendVerification(Request $request, Response $response, $args) + { + // Load validation rules + $schema = new RequestSchema('schema://requests/resend-verification.yaml'); + $validator = new JqueryValidationAdapter($schema, $this->ci->translator); + + return $this->ci->view->render($response, 'pages/resend-verification.html.twig', [ + 'page' => [ + 'validators' => [ + 'resend_verification' => $validator->rules('json', false) + ] + ] + ]); + } + + /** + * Reset password page. + * + * Renders the new password page for password reset requests. + * Request type: GET + * + * @param Request $request + * @param Response $response + * @param array $args + * @return void + */ + public function pageResetPassword(Request $request, Response $response, $args) + { + // Insert the user's secret token from the link into the password reset form + $params = $request->getQueryParams(); + + // Load validation rules - note this uses the same schema as "set password" + $schema = new RequestSchema('schema://requests/set-password.yaml'); + $validator = new JqueryValidationAdapter($schema, $this->ci->translator); + + return $this->ci->view->render($response, 'pages/reset-password.html.twig', [ + 'page' => [ + 'validators' => [ + 'set_password' => $validator->rules('json', false) + ] + ], + 'token' => isset($params['token']) ? $params['token'] : '', + ]); + } + + /** + * Render the "set password" page. + * + * Renders the page where new users who have had accounts created for them by another user, can set their password. + * By default, this is a "public page" (does not require authentication). + * Request type: GET + * + * @param Request $request + * @param Response $response + * @param array $args + * @return void + */ + public function pageSetPassword(Request $request, Response $response, $args) + { + // Insert the user's secret token from the link into the password set form + $params = $request->getQueryParams(); + + // Load validation rules + $schema = new RequestSchema('schema://requests/set-password.yaml'); + $validator = new JqueryValidationAdapter($schema, $this->ci->translator); + + return $this->ci->view->render($response, 'pages/set-password.html.twig', [ + 'page' => [ + 'validators' => [ + 'set_password' => $validator->rules('json', false) + ] + ], + 'token' => isset($params['token']) ? $params['token'] : '', + ]); + } + + /** + * Account settings page. + * + * Provides a form for users to modify various properties of their account, such as name, email, locale, etc. + * Any fields that the user does not have permission to modify will be automatically disabled. + * This page requires authentication. + * Request type: GET + * + * @param Request $request + * @param Response $response + * @param array $args + * @return void + */ + public function pageSettings(Request $request, Response $response, $args) + { + /** @var \UserFrosting\Sprinkle\Account\Authorize\AuthorizationManager */ + $authorizer = $this->ci->authorizer; + + /** @var \UserFrosting\Sprinkle\Account\Database\Models\User $currentUser */ + $currentUser = $this->ci->currentUser; + + // Access-controlled page + if (!$authorizer->checkAccess($currentUser, 'uri_account_settings')) { + throw new ForbiddenException(); + } + + // Load validation rules + $schema = new RequestSchema('schema://requests/account-settings.yaml'); + $validatorAccountSettings = new JqueryValidationAdapter($schema, $this->ci->translator); + + $schema = new RequestSchema('schema://requests/profile-settings.yaml'); + $validatorProfileSettings = new JqueryValidationAdapter($schema, $this->ci->translator); + + /** @var \UserFrosting\Support\Repository\Repository $config */ + $config = $this->ci->config; + + // Get a list of all locales + $locales = $config->getDefined('site.locales.available'); + + return $this->ci->view->render($response, 'pages/account-settings.html.twig', [ + 'locales' => $locales, + 'page' => [ + 'validators' => [ + 'account_settings' => $validatorAccountSettings->rules('json', false), + 'profile_settings' => $validatorProfileSettings->rules('json', false) + ], + 'visibility' => ($authorizer->checkAccess($currentUser, 'update_account_settings') ? '' : 'disabled') + ] + ]); + } + + /** + * Render the account sign-in page for UserFrosting. + * + * This allows existing users to sign in. + * By definition, this is a "public page" (does not require authentication). + * Request type: GET + * + * @param Request $request + * @param Response $response + * @param array $args + * @return void + */ + public function pageSignIn(Request $request, Response $response, $args) + { + /** @var \UserFrosting\Support\Repository\Repository $config */ + $config = $this->ci->config; + + /** @var \UserFrosting\Sprinkle\Account\Authenticate\Authenticator $authenticator */ + $authenticator = $this->ci->authenticator; + + // Redirect if user is already logged in + if ($authenticator->check()) { + $redirect = $this->ci->get('redirect.onAlreadyLoggedIn'); + + return $redirect($request, $response, $args); + } + + // Load validation rules + $schema = new RequestSchema('schema://requests/login.yaml'); + $validatorLogin = new JqueryValidationAdapter($schema, $this->ci->translator); + + return $this->ci->view->render($response, 'pages/sign-in.html.twig', [ + 'page' => [ + 'validators' => [ + 'login' => $validatorLogin->rules('json', false) + ] + ] + ]); + } + + /** + * Processes a request to update a user's profile information. + * + * Processes the request from the user profile settings form, checking that: + * 1. They have the necessary permissions to update the posted field(s); + * 2. The submitted data is valid. + * This route requires authentication. + * Request type: POST + * + * @param Request $request + * @param Response $response + * @param array $args + * @return void + */ + public function profile(Request $request, Response $response, $args) + { + /** @var \UserFrosting\Sprinkle\Core\Alert\AlertStream $ms */ + $ms = $this->ci->alerts; + + /** @var \UserFrosting\Sprinkle\Account\Authorize\AuthorizationManager */ + $authorizer = $this->ci->authorizer; + + /** @var \UserFrosting\Sprinkle\Account\Database\Models\User $currentUser */ + $currentUser = $this->ci->currentUser; + + // Access control for entire resource - check that the current user has permission to modify themselves + // See recipe "per-field access control" for dynamic fine-grained control over which properties a user can modify. + if (!$authorizer->checkAccess($currentUser, 'update_account_settings')) { + $ms->addMessageTranslated('danger', 'ACCOUNT.ACCESS_DENIED'); + return $response->withStatus(403); + } + + /** @var \UserFrosting\Sprinkle\Core\Util\ClassMapper $classMapper */ + $classMapper = $this->ci->classMapper; + + /** @var \UserFrosting\Support\Repository\Repository $config */ + $config = $this->ci->config; + + // POST parameters + $params = $request->getParsedBody(); + + // Load the request schema + $schema = new RequestSchema('schema://requests/profile-settings.yaml'); + + // Whitelist and set parameter defaults + $transformer = new RequestDataTransformer($schema); + $data = $transformer->transform($params); + + $error = false; + + // Validate, and halt on validation errors. + $validator = new ServerSideValidator($schema, $this->ci->translator); + if (!$validator->validate($data)) { + $ms->addValidationErrors($validator); + $error = true; + } + + // Check that locale is valid + $locales = $config->getDefined('site.locales.available'); + if (!array_key_exists($data['locale'], $locales)) { + $ms->addMessageTranslated('danger', 'LOCALE.INVALID', $data); + $error = true; + } + + if ($error) { + return $response->withStatus(400); + } + + // Looks good, let's update with new values! + // Note that only fields listed in `profile-settings.yaml` will be permitted in $data, so this prevents the user from updating all columns in the DB + $currentUser->fill($data); + + $currentUser->save(); + + // Create activity record + $this->ci->userActivityLogger->info("User {$currentUser->user_name} updated their profile settings.", [ + 'type' => 'update_profile_settings' + ]); + + $ms->addMessageTranslated('success', 'PROFILE.UPDATED'); + return $response->withStatus(200); + } + + /** + * Processes an new account registration request. + * + * This is throttled to prevent account enumeration, since it needs to divulge when a username/email has been used. + * Processes the request from the form on the registration page, checking that: + * 1. The honeypot was not modified; + * 2. The master account has already been created (during installation); + * 3. Account registration is enabled; + * 4. The user is not already logged in; + * 5. Valid information was entered; + * 6. The captcha, if enabled, is correct; + * 7. The username and email are not already taken. + * Automatically sends an activation link upon success, if account activation is enabled. + * This route is "public access". + * Request type: POST + * Returns the User Object for the user record that was created. + * + * @param Request $request + * @param Response $response + * @param array $args + * @return void + */ + public function register(Request $request, Response $response, $args) + { + /** @var \UserFrosting\Sprinkle\Core\Alert\AlertStream $ms */ + $ms = $this->ci->alerts; + + /** @var \UserFrosting\Sprinkle\Core\Util\ClassMapper $classMapper */ + $classMapper = $this->ci->classMapper; + + /** @var \UserFrosting\Support\Repository\Repository $config */ + $config = $this->ci->config; + + // Get POST parameters: user_name, first_name, last_name, email, password, passwordc, captcha, spiderbro, csrf_token + $params = $request->getParsedBody(); + + // Check the honeypot. 'spiderbro' is not a real field, it is hidden on the main page and must be submitted with its default value for this to be processed. + if (!isset($params['spiderbro']) || $params['spiderbro'] != 'http://') { + throw new SpammyRequestException('Possible spam received:' . print_r($params, true)); + } + + // Security measure: do not allow registering new users until the master account has been created. + if (!$classMapper->staticMethod('user', 'find', $config['reserved_user_ids.master'])) { + $ms->addMessageTranslated('danger', 'ACCOUNT.MASTER_NOT_EXISTS'); + return $response->withStatus(403); + } + + // Check if registration is currently enabled + if (!$config['site.registration.enabled']) { + $ms->addMessageTranslated('danger', 'REGISTRATION.DISABLED'); + return $response->withStatus(403); + } + + /** @var \UserFrosting\Sprinkle\Account\Authenticate\Authenticator $authenticator */ + $authenticator = $this->ci->authenticator; + + // Prevent the user from registering if he/she is already logged in + if ($authenticator->check()) { + $ms->addMessageTranslated('danger', 'REGISTRATION.LOGOUT'); + return $response->withStatus(403); + } + + // Load the request schema + $schema = new RequestSchema('schema://requests/register.yaml'); + + // Whitelist and set parameter defaults + $transformer = new RequestDataTransformer($schema); + $data = $transformer->transform($params); + + $error = false; + + // Validate request data + $validator = new ServerSideValidator($schema, $this->ci->translator); + if (!$validator->validate($data)) { + $ms->addValidationErrors($validator); + $error = true; + } + + /** @var \UserFrosting\Sprinkle\Core\Throttle\Throttler $throttler */ + $throttler = $this->ci->throttler; + $delay = $throttler->getDelay('registration_attempt'); + + // Throttle requests + if ($delay > 0) { + return $response->withStatus(429); + } + + // Check if username or email already exists + if ($classMapper->staticMethod('user', 'findUnique', $data['user_name'], 'user_name')) { + $ms->addMessageTranslated('danger', 'USERNAME.IN_USE', $data); + $error = true; + } + + if ($classMapper->staticMethod('user', 'findUnique', $data['email'], 'email')) { + $ms->addMessageTranslated('danger', 'EMAIL.IN_USE', $data); + $error = true; + } + + // Check captcha, if required + if ($config['site.registration.captcha']) { + $captcha = new Captcha($this->ci->session, $this->ci->config['session.keys.captcha']); + if (!$data['captcha'] || !$captcha->verifyCode($data['captcha'])) { + $ms->addMessageTranslated('danger', 'CAPTCHA.FAIL'); + $error = true; + } + } + + if ($error) { + return $response->withStatus(400); + } + + // Remove captcha, password confirmation from object data after validation + unset($data['captcha']); + unset($data['passwordc']); + + if ($config['site.registration.require_email_verification']) { + $data['flag_verified'] = false; + } else { + $data['flag_verified'] = true; + } + + // Load default group + $groupSlug = $config['site.registration.user_defaults.group']; + $defaultGroup = $classMapper->staticMethod('group', 'where', 'slug', $groupSlug)->first(); + + if (!$defaultGroup) { + $e = new HttpException("Account registration is not working because the default group '$groupSlug' does not exist."); + $e->addUserMessage('REGISTRATION.BROKEN'); + throw $e; + } + + // Set default group + $data['group_id'] = $defaultGroup->id; + + // Set default locale + $data['locale'] = $config['site.registration.user_defaults.locale']; + + // Hash password + $data['password'] = Password::hash($data['password']); + + // All checks passed! log events/activities, create user, and send verification email (if required) + // Begin transaction - DB will be rolled back if an exception occurs + Capsule::transaction( function() use ($classMapper, $data, $ms, $config, $throttler) { + // Log throttleable event + $throttler->logEvent('registration_attempt'); + + // Create the user + $user = $classMapper->createInstance('user', $data); + + // Store new user to database + $user->save(); + + // Create activity record + $this->ci->userActivityLogger->info("User {$user->user_name} registered for a new account.", [ + 'type' => 'sign_up', + 'user_id' => $user->id + ]); + + // Load default roles + $defaultRoleSlugs = $classMapper->staticMethod('role', 'getDefaultSlugs'); + $defaultRoles = $classMapper->staticMethod('role', 'whereIn', 'slug', $defaultRoleSlugs)->get(); + $defaultRoleIds = $defaultRoles->pluck('id')->all(); + + // Attach default roles + $user->roles()->attach($defaultRoleIds); + + // Verification email + if ($config['site.registration.require_email_verification']) { + // Try to generate a new verification request + $verification = $this->ci->repoVerification->create($user, $config['verification.timeout']); + + // Create and send verification email + $message = new TwigMailMessage($this->ci->view, 'mail/verify-account.html.twig'); + + $message->from($config['address_book.admin']) + ->addEmailRecipient(new EmailRecipient($user->email, $user->full_name)) + ->addParams([ + 'user' => $user, + 'token' => $verification->getToken() + ]); + + $this->ci->mailer->send($message); + + $ms->addMessageTranslated('success', 'REGISTRATION.COMPLETE_TYPE2', $user->toArray()); + } else { + // No verification required + $ms->addMessageTranslated('success', 'REGISTRATION.COMPLETE_TYPE1'); + } + }); + + return $response->withStatus(200); + } + + /** + * Processes a request to resend the verification email for a new user account. + * + * Processes the request from the resend verification email form, checking that: + * 1. The rate limit on this type of request is observed; + * 2. The provided email is associated with an existing user account; + * 3. The user account is not already verified; + * 4. The submitted data is valid. + * This route is "public access". + * Request type: POST + * + * @param Request $request + * @param Response $response + * @param array $args + * @return void + */ + public function resendVerification(Request $request, Response $response, $args) + { + /** @var \UserFrosting\Sprinkle\Core\Alert\AlertStream $ms */ + $ms = $this->ci->alerts; + + /** @var \UserFrosting\Sprinkle\Core\Util\ClassMapper $classMapper */ + $classMapper = $this->ci->classMapper; + + /** @var \UserFrosting\Support\Repository\Repository $config */ + $config = $this->ci->config; + + // Get POST parameters + $params = $request->getParsedBody(); + + // Load the request schema + $schema = new RequestSchema('schema://requests/resend-verification.yaml'); + + // Whitelist and set parameter defaults + $transformer = new RequestDataTransformer($schema); + $data = $transformer->transform($params); + + // Validate, and halt on validation errors. Failed validation attempts do not count towards throttling limit. + $validator = new ServerSideValidator($schema, $this->ci->translator); + if (!$validator->validate($data)) { + $ms->addValidationErrors($validator); + return $response->withStatus(400); + } + + // Throttle requests + + /** @var \UserFrosting\Sprinkle\Core\Throttle\Throttler $throttler */ + $throttler = $this->ci->throttler; + + $throttleData = [ + 'email' => $data['email'] + ]; + $delay = $throttler->getDelay('verification_request', $throttleData); + + if ($delay > 0) { + $ms->addMessageTranslated('danger', 'RATE_LIMIT_EXCEEDED', ['delay' => $delay]); + return $response->withStatus(429); + } + + // All checks passed! log events/activities, create user, and send verification email (if required) + // Begin transaction - DB will be rolled back if an exception occurs + Capsule::transaction( function() use ($classMapper, $data, $throttler, $throttleData, $config) { + // Log throttleable event + $throttler->logEvent('verification_request', $throttleData); + + // Load the user, by email address + $user = $classMapper->staticMethod('user', 'where', 'email', $data['email'])->first(); + + // Check that the user exists and is not already verified. + // If there is no user with that email address, or the user exists and is already verified, + // we pretend like we succeeded to prevent account enumeration + if ($user && $user->flag_verified != '1') { + // We're good to go - record user activity and send the email + $verification = $this->ci->repoVerification->create($user, $config['verification.timeout']); + + // Create and send verification email + $message = new TwigMailMessage($this->ci->view, 'mail/resend-verification.html.twig'); + + $message->from($config['address_book.admin']) + ->addEmailRecipient(new EmailRecipient($user->email, $user->full_name)) + ->addParams([ + 'user' => $user, + 'token' => $verification->getToken() + ]); + + $this->ci->mailer->send($message); + } + }); + + $ms->addMessageTranslated('success', 'ACCOUNT.VERIFICATION.NEW_LINK_SENT', ['email' => $data['email']]); + return $response->withStatus(200); + } + + /** + * Processes a request to set the password for a new or current user. + * + * Processes the request from the password create/reset form, which should have the secret token embedded in it, checking that: + * 1. The provided secret token is associated with an existing user account; + * 2. The user has a password set/reset request in progress; + * 3. The token has not expired; + * 4. The submitted data (new password) is valid. + * This route is "public access". + * Request type: POST + * + * @param Request $request + * @param Response $response + * @param array $args + * @return void + */ + public function setPassword(Request $request, Response $response, $args) + { + /** @var \UserFrosting\Sprinkle\Core\Alert\AlertStream $ms */ + $ms = $this->ci->alerts; + + /** @var \UserFrosting\Sprinkle\Core\Util\ClassMapper $classMapper */ + $classMapper = $this->ci->classMapper; + + /** @var \UserFrosting\Support\Repository\Repository $config */ + $config = $this->ci->config; + + // Get POST parameters + $params = $request->getParsedBody(); + + // Load the request schema + $schema = new RequestSchema('schema://requests/set-password.yaml'); + + // Whitelist and set parameter defaults + $transformer = new RequestDataTransformer($schema); + $data = $transformer->transform($params); + + // Validate, and halt on validation errors. Failed validation attempts do not count towards throttling limit. + $validator = new ServerSideValidator($schema, $this->ci->translator); + if (!$validator->validate($data)) { + $ms->addValidationErrors($validator); + return $response->withStatus(400); + } + + $forgotPasswordPage = $this->ci->router->pathFor('forgot-password'); + + // Ok, try to complete the request with the specified token and new password + $passwordReset = $this->ci->repoPasswordReset->complete($data['token'], [ + 'password' => $data['password'] + ]); + + if (!$passwordReset) { + $ms->addMessageTranslated('danger', 'PASSWORD.FORGET.INVALID', ['url' => $forgotPasswordPage]); + return $response->withStatus(400); + } + + $ms->addMessageTranslated('success', 'PASSWORD.UPDATED'); + + /** @var \UserFrosting\Sprinkle\Account\Authenticate\Authenticator $authenticator */ + $authenticator = $this->ci->authenticator; + + // Log out any existing user, and create a new session + if ($authenticator->check()) { + $authenticator->logout(); + } + + // Auto-login the user (without "remember me") + $user = $passwordReset->user; + $authenticator->login($user); + + $ms->addMessageTranslated('success', 'WELCOME', $user->export()); + return $response->withStatus(200); + } + + /** + * Processes a request to update a user's account information. + * + * Processes the request from the user account settings form, checking that: + * 1. The user correctly input their current password; + * 2. They have the necessary permissions to update the posted field(s); + * 3. The submitted data is valid. + * This route requires authentication. + * Request type: POST + * + * @param Request $request + * @param Response $response + * @param array $args + * @return void + */ + public function settings(Request $request, Response $response, $args) + { + /** @var \UserFrosting\Sprinkle\Core\Alert\AlertStream $ms */ + $ms = $this->ci->alerts; + + /** @var \UserFrosting\Sprinkle\Account\Authorize\AuthorizationManager */ + $authorizer = $this->ci->authorizer; + + /** @var \UserFrosting\Sprinkle\Account\Database\Models\User $currentUser */ + $currentUser = $this->ci->currentUser; + + // Access control for entire resource - check that the current user has permission to modify themselves + // See recipe "per-field access control" for dynamic fine-grained control over which properties a user can modify. + if (!$authorizer->checkAccess($currentUser, 'update_account_settings')) { + $ms->addMessageTranslated('danger', 'ACCOUNT.ACCESS_DENIED'); + return $response->withStatus(403); + } + + /** @var \UserFrosting\Sprinkle\Core\Util\ClassMapper $classMapper */ + $classMapper = $this->ci->classMapper; + + /** @var \UserFrosting\Support\Repository\Repository $config */ + $config = $this->ci->config; + + // POST parameters + $params = $request->getParsedBody(); + + // Load the request schema + $schema = new RequestSchema('schema://requests/account-settings.yaml'); + + // Whitelist and set parameter defaults + $transformer = new RequestDataTransformer($schema); + $data = $transformer->transform($params); + + $error = false; + + // Validate, and halt on validation errors. + $validator = new ServerSideValidator($schema, $this->ci->translator); + if (!$validator->validate($data)) { + $ms->addValidationErrors($validator); + $error = true; + } + + // Confirm current password + if (!isset($data['passwordcheck']) || !Password::verify($data['passwordcheck'], $currentUser->password)) { + $ms->addMessageTranslated('danger', 'PASSWORD.INVALID'); + $error = true; + } + + // Remove password check, password confirmation from object data after validation + unset($data['passwordcheck']); + unset($data['passwordc']); + + // If new email was submitted, check that the email address is not in use + if (isset($data['email']) && $data['email'] != $currentUser->email && $classMapper->staticMethod('user', 'findUnique', $data['email'], 'email')) { + $ms->addMessageTranslated('danger', 'EMAIL.IN_USE', $data); + $error = true; + } + + if ($error) { + return $response->withStatus(400); + } + + // Hash new password, if specified + if (isset($data['password']) && !empty($data['password'])) { + $data['password'] = Password::hash($data['password']); + } else { + // Do not pass to model if no password is specified + unset($data['password']); + } + + // Looks good, let's update with new values! + // Note that only fields listed in `account-settings.yaml` will be permitted in $data, so this prevents the user from updating all columns in the DB + $currentUser->fill($data); + + $currentUser->save(); + + // Create activity record + $this->ci->userActivityLogger->info("User {$currentUser->user_name} updated their account settings.", [ + 'type' => 'update_account_settings' + ]); + + $ms->addMessageTranslated('success', 'ACCOUNT.SETTINGS.UPDATED'); + return $response->withStatus(200); + } + + /** + * Suggest an available username for a specified first/last name. + * + * This route is "public access". + * Request type: GET + * @todo Can this route be abused for account enumeration? If so we should throttle it as well. + * + * @param Request $request + * @param Response $response + * @param array $args + * @return void + */ + public function suggestUsername(Request $request, Response $response, $args) + { + /** @var \UserFrosting\Sprinkle\Core\Alert\AlertStream $ms */ + $ms = $this->ci->alerts; + + /** @var \UserFrosting\Sprinkle\Core\Util\ClassMapper $classMapper */ + $classMapper = $this->ci->classMapper; + + $suggestion = AccountUtil::randomUniqueUsername($classMapper, 50, 10); + + // Be careful how you consume this data - it has not been escaped and contains untrusted user-supplied content. + // For example, if you plan to insert it into an HTML DOM, you must escape it on the client side (or use client-side templating). + return $response->withJson([ + 'user_name' => $suggestion + ], 200, JSON_PRETTY_PRINT); + } + + /** + * Processes an new email verification request. + * + * Processes the request from the email verification link that was emailed to the user, checking that: + * 1. The token provided matches a user in the database; + * 2. The user account is not already verified; + * This route is "public access". + * Request type: GET + * + * @param Request $request + * @param Response $response + * @param array $args + * @return void + */ + public function verify(Request $request, Response $response, $args) + { + /** @var \UserFrosting\Sprinkle\Core\Alert\AlertStream $ms */ + $ms = $this->ci->alerts; + + /** @var \UserFrosting\Sprinkle\Core\Util\ClassMapper $classMapper */ + $classMapper = $this->ci->classMapper; + + /** @var \UserFrosting\Support\Repository\Repository $config */ + $config = $this->ci->config; + + $loginPage = $this->ci->router->pathFor('login'); + + // GET parameters + $params = $request->getQueryParams(); + + // Load request schema + $schema = new RequestSchema('schema://requests/account-verify.yaml'); + + // Whitelist and set parameter defaults + $transformer = new RequestDataTransformer($schema); + $data = $transformer->transform($params); + + // Validate, and halt on validation errors. This is a GET request, so we redirect on validation error. + $validator = new ServerSideValidator($schema, $this->ci->translator); + if (!$validator->validate($data)) { + $ms->addValidationErrors($validator); + // 400 code + redirect is perfectly fine, according to user Dilaz in #laravel + return $response->withRedirect($loginPage, 400); + } + + $verification = $this->ci->repoVerification->complete($data['token']); + + if (!$verification) { + $ms->addMessageTranslated('danger', 'ACCOUNT.VERIFICATION.TOKEN_NOT_FOUND'); + return $response->withRedirect($loginPage, 400); + } + + $ms->addMessageTranslated('success', 'ACCOUNT.VERIFICATION.COMPLETE'); + + // Forward to login page + return $response->withRedirect($loginPage); + } +} diff --git a/main/app/sprinkles/account/src/Controller/Exception/SpammyRequestException.php b/main/app/sprinkles/account/src/Controller/Exception/SpammyRequestException.php new file mode 100755 index 0000000..9713360 --- /dev/null +++ b/main/app/sprinkles/account/src/Controller/Exception/SpammyRequestException.php @@ -0,0 +1,20 @@ +schema->hasTable('activities')) { + $this->schema->create('activities', function (Blueprint $table) { + $table->increments('id'); + $table->string('ip_address', 45)->nullable(); + $table->integer('user_id')->unsigned(); + $table->string('type', 255)->comment('An identifier used to track the type of activity.'); + $table->timestamp('occurred_at')->nullable(); + $table->text('description')->nullable(); + + $table->engine = 'InnoDB'; + $table->collation = 'utf8_unicode_ci'; + $table->charset = 'utf8'; + //$table->foreign('user_id')->references('id')->on('users'); + $table->index('user_id'); + }); + } + } + + /** + * {@inheritDoc} + */ + public function down() + { + $this->schema->drop('activities'); + } +} \ No newline at end of file diff --git a/main/app/sprinkles/account/src/Database/Migrations/v400/GroupsTable.php b/main/app/sprinkles/account/src/Database/Migrations/v400/GroupsTable.php new file mode 100755 index 0000000..c74615f --- /dev/null +++ b/main/app/sprinkles/account/src/Database/Migrations/v400/GroupsTable.php @@ -0,0 +1,82 @@ +schema->hasTable('groups')) { + $this->schema->create('groups', function(Blueprint $table) { + $table->increments('id'); + $table->string('slug'); + $table->string('name'); + $table->text('description')->nullable(); + $table->string('icon', 100)->nullable(false)->default('fa fa-user')->comment('The icon representing users in this group.'); + $table->timestamps(); + + $table->engine = 'InnoDB'; + $table->collation = 'utf8_unicode_ci'; + $table->charset = 'utf8'; + $table->unique('slug'); + $table->index('slug'); + }); + + // Add default groups + $groups = [ + 'terran' => new Group([ + 'slug' => 'terran', + 'name' => 'Terran', + 'description' => 'The terrans are a young species with psionic potential. The terrans of the Koprulu sector descend from the survivors of a disastrous 23rd century colonization mission from Earth.', + 'icon' => 'sc sc-terran' + ]), + 'zerg' => new Group([ + 'slug' => 'zerg', + 'name' => 'Zerg', + 'description' => 'Dedicated to the pursuit of genetic perfection, the zerg relentlessly hunt down and assimilate advanced species across the galaxy, incorporating useful genetic code into their own.', + 'icon' => 'sc sc-zerg' + ]), + 'protoss' => new Group([ + 'slug' => 'protoss', + 'name' => 'Protoss', + 'description' => 'The protoss, a.k.a. the Firstborn, are a sapient humanoid race native to Aiur. Their advanced technology complements and enhances their psionic mastery.', + 'icon' => 'sc sc-protoss' + ]) + ]; + + foreach ($groups as $slug => $group) { + $group->save(); + } + } + } + + /** + * {@inheritDoc} + */ + public function down() + { + $this->schema->drop('groups'); + } +} diff --git a/main/app/sprinkles/account/src/Database/Migrations/v400/PasswordResetsTable.php b/main/app/sprinkles/account/src/Database/Migrations/v400/PasswordResetsTable.php new file mode 100755 index 0000000..e785ccc --- /dev/null +++ b/main/app/sprinkles/account/src/Database/Migrations/v400/PasswordResetsTable.php @@ -0,0 +1,57 @@ +schema->hasTable('password_resets')) { + $this->schema->create('password_resets', function (Blueprint $table) { + $table->increments('id'); + $table->integer('user_id')->unsigned(); + $table->string('hash'); + $table->boolean('completed')->default(0); + $table->timestamp('expires_at')->nullable(); + $table->timestamp('completed_at')->nullable(); + $table->timestamps(); + + $table->engine = 'InnoDB'; + $table->collation = 'utf8_unicode_ci'; + $table->charset = 'utf8'; + //$table->foreign('user_id')->references('id')->on('users'); + $table->index('user_id'); + $table->index('hash'); + }); + } + } + + /** + * {@inheritDoc} + */ + public function down() + { + $this->schema->drop('password_resets'); + } +} diff --git a/main/app/sprinkles/account/src/Database/Migrations/v400/PermissionRolesTable.php b/main/app/sprinkles/account/src/Database/Migrations/v400/PermissionRolesTable.php new file mode 100755 index 0000000..2c2990c --- /dev/null +++ b/main/app/sprinkles/account/src/Database/Migrations/v400/PermissionRolesTable.php @@ -0,0 +1,55 @@ +schema->hasTable('permission_roles')) { + $this->schema->create('permission_roles', function (Blueprint $table) { + $table->integer('permission_id')->unsigned(); + $table->integer('role_id')->unsigned(); + $table->timestamps(); + + $table->engine = 'InnoDB'; + $table->collation = 'utf8_unicode_ci'; + $table->charset = 'utf8'; + $table->primary(['permission_id', 'role_id']); + //$table->foreign('permission_id')->references('id')->on('permissions'); + //$table->foreign('role_id')->references('id')->on('roles'); + $table->index('permission_id'); + $table->index('role_id'); + }); + } + } + + /** + * {@inheritDoc} + */ + public function down() + { + $this->schema->drop('permission_roles'); + } +} diff --git a/main/app/sprinkles/account/src/Database/Migrations/v400/PermissionsTable.php b/main/app/sprinkles/account/src/Database/Migrations/v400/PermissionsTable.php new file mode 100755 index 0000000..684b01a --- /dev/null +++ b/main/app/sprinkles/account/src/Database/Migrations/v400/PermissionsTable.php @@ -0,0 +1,262 @@ +schema->hasTable('permissions')) { + $this->schema->create('permissions', function(Blueprint $table) { + $table->increments('id'); + $table->string('slug')->comment('A code that references a specific action or URI that an assignee of this permission has access to.'); + $table->string('name'); + $table->text('conditions')->comment('The conditions under which members of this group have access to this hook.'); + $table->text('description')->nullable(); + $table->timestamps(); + + $table->engine = 'InnoDB'; + $table->collation = 'utf8_unicode_ci'; + $table->charset = 'utf8'; + }); + } + } + + /** + * {@inheritDoc} + */ + public function down() + { + $this->schema->drop('permissions'); + } + + /** + * {@inheritDoc} + */ + public function seed() + { + // Skip this if table is not empty + if (Permission::count() == 0) { + + $defaultRoleIds = [ + 'user' => Role::where('slug', 'user')->first()->id, + 'group-admin' => Role::where('slug', 'group-admin')->first()->id, + 'site-admin' => Role::where('slug', 'site-admin')->first()->id + ]; + + // Add default permissions + $permissions = [ + 'create_group' => new Permission([ + 'slug' => 'create_group', + 'name' => 'Create group', + 'conditions' => 'always()', + 'description' => 'Create a new group.' + ]), + 'create_user' => new Permission([ + 'slug' => 'create_user', + 'name' => 'Create user', + 'conditions' => 'always()', + 'description' => 'Create a new user in your own group and assign default roles.' + ]), + 'create_user_field' => new Permission([ + 'slug' => 'create_user_field', + 'name' => 'Set new user group', + 'conditions' => "subset(fields,['group'])", + 'description' => 'Set the group when creating a new user.' + ]), + 'delete_group' => new Permission([ + 'slug' => 'delete_group', + 'name' => 'Delete group', + 'conditions' => "always()", + 'description' => 'Delete a group.' + ]), + 'delete_user' => new Permission([ + 'slug' => 'delete_user', + 'name' => 'Delete user', + 'conditions' => "!has_role(user.id,{$defaultRoleIds['site-admin']}) && !is_master(user.id)", + 'description' => 'Delete users who are not Site Administrators.' + ]), + 'update_account_settings' => new Permission([ + 'slug' => 'update_account_settings', + 'name' => 'Edit user', + 'conditions' => 'always()', + 'description' => 'Edit your own account settings.' + ]), + 'update_group_field' => new Permission([ + 'slug' => 'update_group_field', + 'name' => 'Edit group', + 'conditions' => 'always()', + 'description' => 'Edit basic properties of any group.' + ]), + 'update_user_field' => new Permission([ + 'slug' => 'update_user_field', + 'name' => 'Edit user', + 'conditions' => "!has_role(user.id,{$defaultRoleIds['site-admin']}) && subset(fields,['name','email','locale','group','flag_enabled','flag_verified','password'])", + 'description' => 'Edit users who are not Site Administrators.' + ]), + 'update_user_field_group' => new Permission([ + 'slug' => 'update_user_field', + 'name' => 'Edit group user', + 'conditions' => "equals_num(self.group_id,user.group_id) && !is_master(user.id) && !has_role(user.id,{$defaultRoleIds['site-admin']}) && (!has_role(user.id,{$defaultRoleIds['group-admin']}) || equals_num(self.id,user.id)) && subset(fields,['name','email','locale','flag_enabled','flag_verified','password'])", + 'description' => 'Edit users in your own group who are not Site or Group Administrators, except yourself.' + ]), + 'uri_account_settings' => new Permission([ + 'slug' => 'uri_account_settings', + 'name' => 'Account settings page', + 'conditions' => 'always()', + 'description' => 'View the account settings page.' + ]), + 'uri_activities' => new Permission([ + 'slug' => 'uri_activities', + 'name' => 'Activity monitor', + 'conditions' => 'always()', + 'description' => 'View a list of all activities for all users.' + ]), + 'uri_dashboard' => new Permission([ + 'slug' => 'uri_dashboard', + 'name' => 'Admin dashboard', + 'conditions' => 'always()', + 'description' => 'View the administrative dashboard.' + ]), + 'uri_group' => new Permission([ + 'slug' => 'uri_group', + 'name' => 'View group', + 'conditions' => 'always()', + 'description' => 'View the group page of any group.' + ]), + 'uri_group_own' => new Permission([ + 'slug' => 'uri_group', + 'name' => 'View own group', + 'conditions' => 'equals_num(self.group_id,group.id)', + 'description' => 'View the group page of your own group.' + ]), + 'uri_groups' => new Permission([ + 'slug' => 'uri_groups', + 'name' => 'Group management page', + 'conditions' => 'always()', + 'description' => 'View a page containing a list of groups.' + ]), + 'uri_user' => new Permission([ + 'slug' => 'uri_user', + 'name' => 'View user', + 'conditions' => 'always()', + 'description' => 'View the user page of any user.' + ]), + 'uri_user_in_group' => new Permission([ + 'slug' => 'uri_user', + 'name' => 'View user', + 'conditions' => "equals_num(self.group_id,user.group_id) && !is_master(user.id) && !has_role(user.id,{$defaultRoleIds['site-admin']}) && (!has_role(user.id,{$defaultRoleIds['group-admin']}) || equals_num(self.id,user.id))", + 'description' => 'View the user page of any user in your group, except the master user and Site and Group Administrators (except yourself).' + ]), + 'uri_users' => new Permission([ + 'slug' => 'uri_users', + 'name' => 'User management page', + 'conditions' => 'always()', + 'description' => 'View a page containing a table of users.' + ]), + 'view_group_field' => new Permission([ + 'slug' => 'view_group_field', + 'name' => 'View group', + 'conditions' => "in(property,['name','icon','slug','description','users'])", + 'description' => 'View certain properties of any group.' + ]), + 'view_group_field_own' => new Permission([ + 'slug' => 'view_group_field', + 'name' => 'View group', + 'conditions' => "equals_num(self.group_id,group.id) && in(property,['name','icon','slug','description','users'])", + 'description' => 'View certain properties of your own group.' + ]), + 'view_user_field' => new Permission([ + 'slug' => 'view_user_field', + 'name' => 'View user', + 'conditions' => "in(property,['user_name','name','email','locale','theme','roles','group','activities'])", + 'description' => 'View certain properties of any user.' + ]), + 'view_user_field_group' => new Permission([ + 'slug' => 'view_user_field', + 'name' => 'View user', + 'conditions' => "equals_num(self.group_id,user.group_id) && !is_master(user.id) && !has_role(user.id,{$defaultRoleIds['site-admin']}) && (!has_role(user.id,{$defaultRoleIds['group-admin']}) || equals_num(self.id,user.id)) && in(property,['user_name','name','email','locale','roles','group','activities'])", + 'description' => 'View certain properties of any user in your own group, except the master user and Site and Group Administrators (except yourself).' + ]) + ]; + + foreach ($permissions as $slug => $permission) { + $permission->save(); + } + + // Add default mappings to permissions + $roleUser = Role::where('slug', 'user')->first(); + if ($roleUser) { + $roleUser->permissions()->sync([ + $permissions['update_account_settings']->id, + $permissions['uri_account_settings']->id, + $permissions['uri_dashboard']->id + ]); + } + + $roleSiteAdmin = Role::where('slug', 'site-admin')->first(); + if ($roleSiteAdmin) { + $roleSiteAdmin->permissions()->sync([ + $permissions['create_group']->id, + $permissions['create_user']->id, + $permissions['create_user_field']->id, + $permissions['delete_group']->id, + $permissions['delete_user']->id, + $permissions['update_user_field']->id, + $permissions['update_group_field']->id, + $permissions['uri_activities']->id, + $permissions['uri_group']->id, + $permissions['uri_groups']->id, + $permissions['uri_user']->id, + $permissions['uri_users']->id, + $permissions['view_group_field']->id, + $permissions['view_user_field']->id + ]); + } + + $roleGroupAdmin = Role::where('slug', 'group-admin')->first(); + if ($roleGroupAdmin) { + $roleGroupAdmin->permissions()->sync([ + $permissions['create_user']->id, + $permissions['update_user_field_group']->id, + $permissions['uri_group_own']->id, + $permissions['uri_user_in_group']->id, + $permissions['view_group_field_own']->id, + $permissions['view_user_field_group']->id + ]); + } + } + } +} diff --git a/main/app/sprinkles/account/src/Database/Migrations/v400/PersistencesTable.php b/main/app/sprinkles/account/src/Database/Migrations/v400/PersistencesTable.php new file mode 100755 index 0000000..b96e327 --- /dev/null +++ b/main/app/sprinkles/account/src/Database/Migrations/v400/PersistencesTable.php @@ -0,0 +1,57 @@ +schema->hasTable('persistences')) { + $this->schema->create('persistences', function (Blueprint $table) { + $table->increments('id'); + $table->integer('user_id')->unsigned(); + $table->string('token', 40); + $table->string('persistent_token', 40); + $table->timestamp('expires_at')->nullable(); + $table->timestamps(); + + $table->engine = 'InnoDB'; + $table->collation = 'utf8_unicode_ci'; + $table->charset = 'utf8'; + //$table->foreign('user_id')->references('id')->on('users'); + $table->index('user_id'); + $table->index('token'); + $table->index('persistent_token'); + }); + } + } + + /** + * {@inheritDoc} + */ + public function down() + { + $this->schema->drop('persistences'); + } +} diff --git a/main/app/sprinkles/account/src/Database/Migrations/v400/RoleUsersTable.php b/main/app/sprinkles/account/src/Database/Migrations/v400/RoleUsersTable.php new file mode 100755 index 0000000..7f3648b --- /dev/null +++ b/main/app/sprinkles/account/src/Database/Migrations/v400/RoleUsersTable.php @@ -0,0 +1,55 @@ +schema->hasTable('role_users')) { + $this->schema->create('role_users', function (Blueprint $table) { + $table->integer('user_id')->unsigned(); + $table->integer('role_id')->unsigned(); + $table->timestamps(); + + $table->engine = 'InnoDB'; + $table->collation = 'utf8_unicode_ci'; + $table->charset = 'utf8'; + $table->primary(['user_id', 'role_id']); + //$table->foreign('user_id')->references('id')->on('users'); + //$table->foreign('role_id')->references('id')->on('roles'); + $table->index('user_id'); + $table->index('role_id'); + }); + } + } + + /** + * {@inheritDoc} + */ + public function down() + { + $this->schema->drop('role_users'); + } +} diff --git a/main/app/sprinkles/account/src/Database/Migrations/v400/RolesTable.php b/main/app/sprinkles/account/src/Database/Migrations/v400/RolesTable.php new file mode 100755 index 0000000..9cef494 --- /dev/null +++ b/main/app/sprinkles/account/src/Database/Migrations/v400/RolesTable.php @@ -0,0 +1,78 @@ +schema->hasTable('roles')) { + $this->schema->create('roles', function (Blueprint $table) { + $table->increments('id'); + $table->string('slug'); + $table->string('name'); + $table->text('description')->nullable(); + $table->timestamps(); + + $table->engine = 'InnoDB'; + $table->collation = 'utf8_unicode_ci'; + $table->charset = 'utf8'; + $table->unique('slug'); + $table->index('slug'); + }); + + // Add default roles + $roles = [ + 'user' => new Role([ + 'slug' => 'user', + 'name' => 'User', + 'description' => 'This role provides basic user functionality.' + ]), + 'site-admin' => new Role([ + 'slug' => 'site-admin', + 'name' => 'Site Administrator', + 'description' => 'This role is meant for "site administrators", who can basically do anything except create, edit, or delete other administrators.' + ]), + 'group-admin' => new Role([ + 'slug' => 'group-admin', + 'name' => 'Group Administrator', + 'description' => 'This role is meant for "group administrators", who can basically do anything with users in their own group, except other administrators of that group.' + ]) + ]; + + foreach ($roles as $slug => $role) { + $role->save(); + } + } + } + + /** + * {@inheritDoc} + */ + public function down() + { + $this->schema->drop('roles'); + } +} diff --git a/main/app/sprinkles/account/src/Database/Migrations/v400/UsersTable.php b/main/app/sprinkles/account/src/Database/Migrations/v400/UsersTable.php new file mode 100755 index 0000000..a65eeed --- /dev/null +++ b/main/app/sprinkles/account/src/Database/Migrations/v400/UsersTable.php @@ -0,0 +1,69 @@ +schema->hasTable('users')) { + $this->schema->create('users', function (Blueprint $table) { + $table->increments('id'); + $table->string('user_name', 50); + $table->string('email', 254); + $table->string('first_name', 20); + $table->string('last_name', 30); + $table->string('locale', 10)->default('en_US')->comment('The language and locale to use for this user.'); + $table->string('theme', 100)->nullable()->comment("The user theme."); + $table->integer('group_id')->unsigned()->default(1)->comment("The id of the user group."); + $table->boolean('flag_verified')->default(1)->comment("Set to 1 if the user has verified their account via email, 0 otherwise."); + $table->boolean('flag_enabled')->default(1)->comment("Set to 1 if the user account is currently enabled, 0 otherwise. Disabled accounts cannot be logged in to, but they retain all of their data and settings."); + $table->integer('last_activity_id')->unsigned()->nullable()->comment("The id of the last activity performed by this user."); + $table->string('password', 255); + $table->softDeletes(); + $table->timestamps(); + + $table->engine = 'InnoDB'; + $table->collation = 'utf8_unicode_ci'; + $table->charset = 'utf8'; + //$table->foreign('group_id')->references('id')->on('groups'); + //$table->foreign('last_activity_id')->references('id')->on('activities'); + $table->unique('user_name'); + $table->index('user_name'); + $table->unique('email'); + $table->index('email'); + $table->index('group_id'); + $table->index('last_activity_id'); + }); + } + } + + /** + * {@inheritDoc} + */ + public function down() + { + $this->schema->drop('users'); + } +} diff --git a/main/app/sprinkles/account/src/Database/Migrations/v400/VerificationsTable.php b/main/app/sprinkles/account/src/Database/Migrations/v400/VerificationsTable.php new file mode 100755 index 0000000..fa54da6 --- /dev/null +++ b/main/app/sprinkles/account/src/Database/Migrations/v400/VerificationsTable.php @@ -0,0 +1,57 @@ +schema->hasTable('verifications')) { + $this->schema->create('verifications', function (Blueprint $table) { + $table->increments('id'); + $table->integer('user_id')->unsigned(); + $table->string('hash'); + $table->boolean('completed')->default(0); + $table->timestamp('expires_at')->nullable(); + $table->timestamp('completed_at')->nullable(); + $table->timestamps(); + + $table->engine = 'InnoDB'; + $table->collation = 'utf8_unicode_ci'; + $table->charset = 'utf8'; + //$table->foreign('user_id')->references('id')->on('users'); + $table->index('user_id'); + $table->index('hash'); + }); + } + } + + /** + * {@inheritDoc} + */ + public function down() + { + $this->schema->drop('verifications'); + } +} diff --git a/main/app/sprinkles/account/src/Database/Models/Activity.php b/main/app/sprinkles/account/src/Database/Models/Activity.php new file mode 100755 index 0000000..d5be589 --- /dev/null +++ b/main/app/sprinkles/account/src/Database/Models/Activity.php @@ -0,0 +1,86 @@ +select('activities.*'); + + $query = $query->leftJoin('users', 'activities.user_id', '=', 'users.id'); + + return $query; + } + + /** + * Add clauses to select the most recent event of each type for each user, to the query. + * + * @return \Illuminate\Database\Query\Builder + */ + public function scopeMostRecentEvents($query) + { + return $query->select('user_id', 'event_type', Capsule::raw('MAX(occurred_at) as occurred_at')) + ->groupBy('user_id') + ->groupBy('type'); + } + + /** + * Add clauses to select the most recent event of a given type for each user, to the query. + * + * @param string $type The type of event, matching the `event_type` field in the user_event table. + * @return \Illuminate\Database\Query\Builder + */ + public function scopeMostRecentEventsByType($query, $type) + { + return $query->select('user_id', Capsule::raw('MAX(occurred_at) as occurred_at')) + ->where('type', $type) + ->groupBy('user_id'); + } + + /** + * Get the user associated with this activity. + */ + public function user() + { + /** @var UserFrosting\Sprinkle\Core\Util\ClassMapper $classMapper */ + $classMapper = static::$ci->classMapper; + + return $this->belongsTo($classMapper->getClassMapping('user'), 'user_id'); + } +} diff --git a/main/app/sprinkles/account/src/Database/Models/Group.php b/main/app/sprinkles/account/src/Database/Models/Group.php new file mode 100755 index 0000000..f10e066 --- /dev/null +++ b/main/app/sprinkles/account/src/Database/Models/Group.php @@ -0,0 +1,69 @@ +classMapper; + + return $this->hasMany($classMapper->getClassMapping('user'), 'group_id'); + } +} diff --git a/main/app/sprinkles/account/src/Database/Models/PasswordReset.php b/main/app/sprinkles/account/src/Database/Models/PasswordReset.php new file mode 100755 index 0000000..ac8a930 --- /dev/null +++ b/main/app/sprinkles/account/src/Database/Models/PasswordReset.php @@ -0,0 +1,76 @@ +token; + } + + /** + * @param string $value + */ + public function setToken($value) + { + $this->token = $value; + return $this; + } + + /** + * Get the user associated with this reset request. + */ + public function user() + { + /** @var UserFrosting\Sprinkle\Core\Util\ClassMapper $classMapper */ + $classMapper = static::$ci->classMapper; + + return $this->belongsTo($classMapper->getClassMapping('user'), 'user_id'); + } +} diff --git a/main/app/sprinkles/account/src/Database/Models/Permission.php b/main/app/sprinkles/account/src/Database/Models/Permission.php new file mode 100755 index 0000000..463af8d --- /dev/null +++ b/main/app/sprinkles/account/src/Database/Models/Permission.php @@ -0,0 +1,121 @@ +roles()->detach(); + + // Delete the permission + $result = parent::delete(); + + return $result; + } + + /** + * Get a list of roles to which this permission is assigned. + * + * @return \Illuminate\Database\Eloquent\Relations\BelongsToMany + */ + public function roles() + { + /** @var UserFrosting\Sprinkle\Core\Util\ClassMapper $classMapper */ + $classMapper = static::$ci->classMapper; + + return $this->belongsToMany($classMapper->getClassMapping('role'), 'permission_roles', 'permission_id', 'role_id')->withTimestamps(); + } + + /** + * Query scope to get all permissions assigned to a specific role. + * + * @param \Illuminate\Database\Eloquent\Builder $query + * @param int $roleId + * @return \Illuminate\Database\Eloquent\Builder + */ + public function scopeForRole($query, $roleId) + { + return $query->join('permission_roles', function ($join) use ($roleId) { + $join->on('permission_roles.permission_id', 'permissions.id') + ->where('role_id', $roleId); + }); + } + + /** + * Query scope to get all permissions NOT associated with a specific role. + * + * @param \Illuminate\Database\Eloquent\Builder $query + * @param int $roleId + * @return \Illuminate\Database\Eloquent\Builder + */ + public function scopeNotForRole($query, $roleId) + { + return $query->join('permission_roles', function ($join) use ($roleId) { + $join->on('permission_roles.permission_id', 'permissions.id') + ->where('role_id', '!=', $roleId); + }); + } + + /** + * Get a list of users who have this permission, along with a list of roles through which each user has the permission. + * + * @return \UserFrosting\Sprinkle\Core\Database\Relations\BelongsToManyThrough + */ + public function users() + { + /** @var UserFrosting\Sprinkle\Core\Util\ClassMapper $classMapper */ + $classMapper = static::$ci->classMapper; + + return $this->belongsToManyThrough( + $classMapper->getClassMapping('user'), + $classMapper->getClassMapping('role'), + 'permission_roles', + 'permission_id', + 'role_id', + 'role_users', + 'role_id', + 'user_id' + ); + } +} diff --git a/main/app/sprinkles/account/src/Database/Models/Role.php b/main/app/sprinkles/account/src/Database/Models/Role.php new file mode 100755 index 0000000..ce9cb8c --- /dev/null +++ b/main/app/sprinkles/account/src/Database/Models/Role.php @@ -0,0 +1,105 @@ +permissions()->detach(); + + // Remove all user associations + $this->users()->detach(); + + // Delete the role + $result = parent::delete(); + + return $result; + } + + /** + * Get a list of default roles. + */ + public static function getDefaultSlugs() + { + /** @var UserFrosting\Config $config */ + $config = static::$ci->config; + + return array_map('trim', array_keys($config['site.registration.user_defaults.roles'], true)); + } + + /** + * Get a list of permissions assigned to this role. + */ + public function permissions() + { + /** @var UserFrosting\Sprinkle\Core\Util\ClassMapper $classMapper */ + $classMapper = static::$ci->classMapper; + + return $this->belongsToMany($classMapper->getClassMapping('permission'), 'permission_roles', 'role_id', 'permission_id')->withTimestamps(); + } + + /** + * Query scope to get all roles assigned to a specific user. + * + * @param \Illuminate\Database\Eloquent\Builder $query + * @param int $userId + * @return \Illuminate\Database\Eloquent\Builder + */ + public function scopeForUser($query, $userId) + { + return $query->join('role_users', function ($join) use ($userId) { + $join->on('role_users.role_id', 'roles.id') + ->where('user_id', $userId); + }); + } + + /** + * Get a list of users who have this role. + */ + public function users() + { + /** @var UserFrosting\Sprinkle\Core\Util\ClassMapper $classMapper */ + $classMapper = static::$ci->classMapper; + + return $this->belongsToMany($classMapper->getClassMapping('user'), 'role_users', 'role_id', 'user_id'); + } +} diff --git a/main/app/sprinkles/account/src/Database/Models/User.php b/main/app/sprinkles/account/src/Database/Models/User.php new file mode 100755 index 0000000..235f2ef --- /dev/null +++ b/main/app/sprinkles/account/src/Database/Models/User.php @@ -0,0 +1,493 @@ +lastActivityTime('sign_in'); + } elseif ($name == 'avatar') { + // Use Gravatar as the user avatar + $hash = md5(strtolower(trim( $this->email))); + return 'https://www.gravatar.com/avatar/' . $hash . '?d=mm'; + } else { + return parent::__get($name); + } + } + + /** + * Get all activities for this user. + * + * @return \Illuminate\Database\Eloquent\Relations\HasMany + */ + public function activities() + { + /** @var UserFrosting\Sprinkle\Core\Util\ClassMapper $classMapper */ + $classMapper = static::$ci->classMapper; + + return $this->hasMany($classMapper->getClassMapping('activity'), 'user_id'); + } + + /** + * Delete this user from the database, along with any linked roles and activities. + * + * @param bool $hardDelete Set to true to completely remove the user and all associated objects. + * @return bool true if the deletion was successful, false otherwise. + */ + public function delete($hardDelete = false) + { + /** @var UserFrosting\Sprinkle\Core\Util\ClassMapper $classMapper */ + $classMapper = static::$ci->classMapper; + + if ($hardDelete) { + // Remove all role associations + $this->roles()->detach(); + + // Remove all user activities + $classMapper->staticMethod('activity', 'where', 'user_id', $this->id)->delete(); + + // Remove all user tokens + $classMapper->staticMethod('password_reset', 'where', 'user_id', $this->id)->delete(); + $classMapper->staticMethod('verification', 'where', 'user_id', $this->id)->delete(); + + // TODO: remove any persistences + + // Delete the user + $result = parent::forceDelete(); + } else { + // Soft delete the user, leaving all associated records alone + $result = parent::delete(); + } + + return $result; + } + + /** + * Determines whether a user exists, including checking soft-deleted records + * + * @deprecated since 4.1.7 This method conflicts with and overrides the Builder::exists() method. Use Model::findUnique instead. + * @param mixed $value + * @param string $identifier + * @param bool $checkDeleted set to true to include soft-deleted records + * @return User|null + */ + public static function exists($value, $identifier = 'user_name', $checkDeleted = true) + { + return static::findUnique($value, $identifier, $checkDeleted); + } + + /** + * Return a cache instance specific to that user + * + * @return \Illuminate\Contracts\Cache\Store + */ + public function getCache() + { + return static::$ci->cache->tags('_u'.$this->id); + } + + /** + * Allows you to get the full name of the user using `$user->full_name` + * + * @return string + */ + public function getFullNameAttribute() + { + return $this->first_name . ' ' . $this->last_name; + } + + /** + * Retrieve the cached permissions dictionary for this user. + * + * @return array + */ + public function getCachedPermissions() + { + if (!isset($this->cachedPermissions)) { + $this->reloadCachedPermissions(); + } + + return $this->cachedPermissions; + } + + /** + * Retrieve the cached permissions dictionary for this user. + * + * @return User + */ + public function reloadCachedPermissions() + { + $this->cachedPermissions = $this->buildPermissionsDictionary(); + + return $this; + } + + /** + * Get the amount of time, in seconds, that has elapsed since the last activity of a certain time for this user. + * + * @param string $type The type of activity to search for. + * @return int + */ + public function getSecondsSinceLastActivity($type) + { + $time = $this->lastActivityTime($type); + $time = $time ? $time : '0000-00-00 00:00:00'; + $time = new Carbon($time); + + return $time->diffInSeconds(); + } + + /** + * Return this user's group. + * + * @return \Illuminate\Database\Eloquent\Relations\BelongsTo + */ + public function group() + { + /** @var UserFrosting\Sprinkle\Core\Util\ClassMapper $classMapper */ + $classMapper = static::$ci->classMapper; + + return $this->belongsTo($classMapper->getClassMapping('group'), 'group_id'); + } + + /** + * Returns whether or not this user is the master user. + * + * @return bool + */ + public function isMaster() + { + $masterId = static::$ci->config['reserved_user_ids.master']; + + // Need to use loose comparison for now, because some DBs return `id` as a string + return ($this->id == $masterId); + } + + /** + * Get the most recent activity for this user, based on the user's last_activity_id. + * + * @return \Illuminate\Database\Eloquent\Relations\BelongsTo + */ + public function lastActivity() + { + /** @var UserFrosting\Sprinkle\Core\Util\ClassMapper $classMapper */ + $classMapper = static::$ci->classMapper; + + return $this->belongsTo($classMapper->getClassMapping('activity'), 'last_activity_id'); + } + + /** + * Find the most recent activity for this user of a particular type. + * + * @param string $type + * @return \Illuminate\Database\Eloquent\Builder + */ + public function lastActivityOfType($type = null) + { + /** @var UserFrosting\Sprinkle\Core\Util\ClassMapper $classMapper */ + $classMapper = static::$ci->classMapper; + + $query = $this->hasOne($classMapper->getClassMapping('activity'), 'user_id'); + + if ($type) { + $query = $query->where('type', $type); + } + + return $query->latest('occurred_at'); + } + + /** + * Get the most recent time for a specified activity type for this user. + * + * @param string $type + * @return string|null The last activity time, as a SQL formatted time (YYYY-MM-DD HH:MM:SS), or null if an activity of this type doesn't exist. + */ + public function lastActivityTime($type) + { + $result = $this->activities() + ->where('type', $type) + ->max('occurred_at'); + return $result ? $result : null; + } + + /** + * Performs tasks to be done after this user has been successfully authenticated. + * + * By default, adds a new sign-in activity and updates any legacy hash. + * @param mixed[] $params Optional array of parameters used for this event handler. + * @todo Transition to Laravel Event dispatcher to handle this + */ + public function onLogin($params = []) + { + // Add a sign in activity (time is automatically set by database) + static::$ci->userActivityLogger->info("User {$this->user_name} signed in.", [ + 'type' => 'sign_in' + ]); + + // Update password if we had encountered an outdated hash + $passwordType = Password::getHashType($this->password); + + if ($passwordType != 'modern') { + if (!isset($params['password'])) { + Debug::debug('Notice: Unhashed password must be supplied to update to modern password hashing.'); + } else { + // Hash the user's password and update + $passwordHash = Password::hash($params['password']); + if ($passwordHash === null) { + Debug::debug('Notice: outdated password hash could not be updated because the new hashing algorithm is not supported. Are you running PHP >= 5.3.7?'); + } else { + $this->password = $passwordHash; + Debug::debug('Notice: outdated password hash has been automatically updated to modern hashing.'); + } + } + } + + // Save changes + $this->save(); + + return $this; + } + + /** + * Performs tasks to be done after this user has been logged out. + * + * By default, adds a new sign-out activity. + * @param mixed[] $params Optional array of parameters used for this event handler. + * @todo Transition to Laravel Event dispatcher to handle this + */ + public function onLogout($params = []) + { + static::$ci->userActivityLogger->info("User {$this->user_name} signed out.", [ + 'type' => 'sign_out' + ]); + + return $this; + } + + /** + * Get all password reset requests for this user. + * + * @return \Illuminate\Database\Eloquent\Relations\HasMany + */ + public function passwordResets() + { + /** @var UserFrosting\Sprinkle\Core\Util\ClassMapper $classMapper */ + $classMapper = static::$ci->classMapper; + + return $this->hasMany($classMapper->getClassMapping('password_reset'), 'user_id'); + } + + /** + * Get all of the permissions this user has, via its roles. + * + * @return \UserFrosting\Sprinkle\Core\Database\Relations\BelongsToManyThrough + */ + public function permissions() + { + /** @var UserFrosting\Sprinkle\Core\Util\ClassMapper $classMapper */ + $classMapper = static::$ci->classMapper; + + return $this->belongsToManyThrough( + $classMapper->getClassMapping('permission'), + $classMapper->getClassMapping('role'), + 'role_users', + 'user_id', + 'role_id', + 'permission_roles', + 'role_id', + 'permission_id' + ); + } + + /** + * Get all roles to which this user belongs. + * + * @return \Illuminate\Database\Eloquent\Relations\BelongsToMany + */ + public function roles() + { + /** @var UserFrosting\Sprinkle\Core\Util\ClassMapper $classMapper */ + $classMapper = static::$ci->classMapper; + + return $this->belongsToMany($classMapper->getClassMapping('role'), 'role_users', 'user_id', 'role_id')->withTimestamps(); + } + + /** + * Query scope to get all users who have a specific role. + * + * @param \Illuminate\Database\Eloquent\Builder $query + * @param int $roleId + * @return \Illuminate\Database\Eloquent\Builder + */ + public function scopeForRole($query, $roleId) + { + return $query->join('role_users', function ($join) use ($roleId) { + $join->on('role_users.user_id', 'users.id') + ->where('role_id', $roleId); + }); + } + + /** + * Joins the user's most recent activity directly, so we can do things like sort, search, paginate, etc. + * + * @param \Illuminate\Database\Eloquent\Builder $query + * @return \Illuminate\Database\Eloquent\Builder + */ + public function scopeJoinLastActivity($query) + { + $query = $query->select('users.*'); + + $query = $query->leftJoin('activities', 'activities.id', '=', 'users.last_activity_id'); + + return $query; + } + + /** + * Loads permissions for this user into a cached dictionary of slugs -> arrays of permissions, + * so we don't need to keep requerying the DB for every call of checkAccess. + * + * @return array + */ + protected function buildPermissionsDictionary() + { + $permissions = $this->permissions()->get(); + $cachedPermissions = []; + + foreach ($permissions as $permission) { + $cachedPermissions[$permission->slug][] = $permission; + } + + return $cachedPermissions; + } +} diff --git a/main/app/sprinkles/account/src/Database/Models/Verification.php b/main/app/sprinkles/account/src/Database/Models/Verification.php new file mode 100755 index 0000000..cd5166d --- /dev/null +++ b/main/app/sprinkles/account/src/Database/Models/Verification.php @@ -0,0 +1,70 @@ +token; + } + + public function setToken($value) + { + $this->token = $value; + return $this; + } + + /** + * Get the user associated with this verification request. + */ + public function user() + { + /** @var UserFrosting\Sprinkle\Core\Util\ClassMapper $classMapper */ + $classMapper = static::$ci->classMapper; + + return $this->belongsTo($classMapper->getClassMapping('user'), 'user_id'); + } +} diff --git a/main/app/sprinkles/account/src/Error/Handler/AuthCompromisedExceptionHandler.php b/main/app/sprinkles/account/src/Error/Handler/AuthCompromisedExceptionHandler.php new file mode 100755 index 0000000..330ca65 --- /dev/null +++ b/main/app/sprinkles/account/src/Error/Handler/AuthCompromisedExceptionHandler.php @@ -0,0 +1,34 @@ +ci->view->getEnvironment()->loadTemplate('pages/error/compromised.html.twig'); + + return $this->response + ->withStatus($this->statusCode) + ->withHeader('Content-type', $this->contentType) + ->write($template->render()); + } +} diff --git a/main/app/sprinkles/account/src/Error/Handler/AuthExpiredExceptionHandler.php b/main/app/sprinkles/account/src/Error/Handler/AuthExpiredExceptionHandler.php new file mode 100755 index 0000000..c651f77 --- /dev/null +++ b/main/app/sprinkles/account/src/Error/Handler/AuthExpiredExceptionHandler.php @@ -0,0 +1,50 @@ +writeAlerts(); + + $response = $this->response; + + // For non-AJAX requests, we forward the user to the login page. + if (!$this->request->isXhr()) { + $uri = $this->request->getUri(); + $path = $uri->getPath(); + $query = $uri->getQuery(); + $fragment = $uri->getFragment(); + + $path = $path + . ($query ? '?' . $query : '') + . ($fragment ? '#' . $fragment : ''); + + $loginPage = $this->ci->router->pathFor('login', [], [ + 'redirect' => $path + ]); + + $response = $response->withRedirect($loginPage); + } + + return $response; + } +} diff --git a/main/app/sprinkles/account/src/Error/Handler/ForbiddenExceptionHandler.php b/main/app/sprinkles/account/src/Error/Handler/ForbiddenExceptionHandler.php new file mode 100755 index 0000000..e22f02b --- /dev/null +++ b/main/app/sprinkles/account/src/Error/Handler/ForbiddenExceptionHandler.php @@ -0,0 +1,31 @@ +classMapper->createInstance($this->modelName, $record['extra']); + $log->save(); + + if (isset($record['extra']['user_id'])) { + $user = $this->classMapper->staticMethod('user', 'find', $record['extra']['user_id']); + $user->last_activity_id = $log->id; + $user->save(); + } + } +} diff --git a/main/app/sprinkles/account/src/Log/UserActivityProcessor.php b/main/app/sprinkles/account/src/Log/UserActivityProcessor.php new file mode 100755 index 0000000..2575270 --- /dev/null +++ b/main/app/sprinkles/account/src/Log/UserActivityProcessor.php @@ -0,0 +1,45 @@ +userId = $userId; + } + + public function __invoke(array $record) + { + $additionalFields = [ + 'ip_address' => $_SERVER['REMOTE_ADDR'], + 'user_id' => $this->userId, + 'occurred_at' => $record['datetime'], + 'description' => $record['message'] + ]; + + $record['extra'] = array_replace_recursive($record['extra'], $additionalFields, $record['context']); + + return $record; + } +} diff --git a/main/app/sprinkles/account/src/Repository/PasswordResetRepository.php b/main/app/sprinkles/account/src/Repository/PasswordResetRepository.php new file mode 100755 index 0000000..2dcffd3 --- /dev/null +++ b/main/app/sprinkles/account/src/Repository/PasswordResetRepository.php @@ -0,0 +1,34 @@ +password = Password::hash($args['password']); + // TODO: generate user activity? or do this in controller? + $user->save(); + } +} diff --git a/main/app/sprinkles/account/src/Repository/TokenRepository.php b/main/app/sprinkles/account/src/Repository/TokenRepository.php new file mode 100755 index 0000000..a299439 --- /dev/null +++ b/main/app/sprinkles/account/src/Repository/TokenRepository.php @@ -0,0 +1,230 @@ +classMapper = $classMapper; + $this->algorithm = $algorithm; + } + + /** + * Cancels a specified token by removing it from the database. + * + * @param int $token The token to remove. + * @return Model|false + */ + public function cancel($token) + { + // Hash the password reset token for the stored version + $hash = hash($this->algorithm, $token); + + // Find an incomplete reset request for the specified hash + $model = $this->classMapper + ->staticMethod($this->modelIdentifier, 'where', 'hash', $hash) + ->where('completed', false) + ->first(); + + if ($model === null) { + return false; + } + + $model->delete(); + + return $model; + } + + /** + * Completes a token-based process, invoking updateUser() in the child object to do the actual action. + * + * @param int $token The token to complete. + * @param mixed[] $userParams An optional list of parameters to pass to updateUser(). + * @return Model|false + */ + public function complete($token, $userParams = []) + { + // Hash the token for the stored version + $hash = hash($this->algorithm, $token); + + // Find an unexpired, incomplete token for the specified hash + $model = $this->classMapper + ->staticMethod($this->modelIdentifier, 'where', 'hash', $hash) + ->where('completed', false) + ->where('expires_at', '>', Carbon::now()) + ->first(); + + if ($model === null) { + return false; + } + + // Fetch user for this token + $user = $this->classMapper->staticMethod('user', 'find', $model->user_id); + + if (is_null($user)) { + return false; + } + + $this->updateUser($user, $userParams); + + $model->fill([ + 'completed' => true, + 'completed_at' => Carbon::now() + ]); + + $model->save(); + + return $model; + } + + /** + * Create a new token for a specified user. + * + * @param User $user The user object to associate with this token. + * @param int $timeout The time, in seconds, after which this token should expire. + * @return Model The model (PasswordReset, Verification, etc) object that stores the token. + */ + public function create(User $user, $timeout) + { + // Remove any previous tokens for this user + $this->removeExisting($user); + + // Compute expiration time + $expiresAt = Carbon::now()->addSeconds($timeout); + + $model = $this->classMapper->createInstance($this->modelIdentifier); + + // Generate a random token + $model->setToken($this->generateRandomToken()); + + // Hash the password reset token for the stored version + $hash = hash($this->algorithm, $model->getToken()); + + $model->fill([ + 'hash' => $hash, + 'completed' => false, + 'expires_at' => $expiresAt + ]); + + $model->user_id = $user->id; + + $model->save(); + + return $model; + } + + /** + * Determine if a specified user has an incomplete and unexpired token. + * + * @param User $user The user object to look up. + * @param int $token Optionally, try to match a specific token. + * @return Model|false + */ + public function exists(User $user, $token = null) + { + $model = $this->classMapper + ->staticMethod($this->modelIdentifier, 'where', 'user_id', $user->id) + ->where('completed', false) + ->where('expires_at', '>', Carbon::now()); + + if ($token) { + // get token hash + $hash = hash($this->algorithm, $token); + $model->where('hash', $hash); + } + + return $model->first() ?: false; + } + + /** + * Delete all existing tokens from the database for a particular user. + * + * @param User $user + * @return int + */ + protected function removeExisting(User $user) + { + return $this->classMapper + ->staticMethod($this->modelIdentifier, 'where', 'user_id', $user->id) + ->delete(); + } + + /** + * Remove all expired tokens from the database. + * + * @return bool|null + */ + public function removeExpired() + { + return $this->classMapper + ->staticMethod($this->modelIdentifier, 'where', 'completed', false) + ->where('expires_at', '<', Carbon::now()) + ->delete(); + } + + /** + * Generate a new random token for this user. + * + * This generates a token to use for verifying a new account, resetting a lost password, etc. + * @param string $gen specify an existing token that, if we happen to generate the same value, we should regenerate on. + * @return string + */ + protected function generateRandomToken($gen = null) + { + do { + $gen = md5(uniqid(mt_rand(), false)); + } while($this->classMapper + ->staticMethod($this->modelIdentifier, 'where', 'hash', hash($this->algorithm, $gen)) + ->first()); + return $gen; + } + + /** + * Modify the user during the token completion process. + * + * This method is called during complete(), and is a way for concrete implementations to modify the user. + * @param User $user the user object to modify. + * @return mixed[] $args the list of parameters that were supplied to the call to `complete()` + */ + abstract protected function updateUser($user, $args); +} diff --git a/main/app/sprinkles/account/src/Repository/VerificationRepository.php b/main/app/sprinkles/account/src/Repository/VerificationRepository.php new file mode 100755 index 0000000..b0cf048 --- /dev/null +++ b/main/app/sprinkles/account/src/Repository/VerificationRepository.php @@ -0,0 +1,32 @@ +flag_verified = 1; + // TODO: generate user activity? or do this in controller? + $user->save(); + } +} diff --git a/main/app/sprinkles/account/src/ServicesProvider/ServicesProvider.php b/main/app/sprinkles/account/src/ServicesProvider/ServicesProvider.php new file mode 100755 index 0000000..4c3ab15 --- /dev/null +++ b/main/app/sprinkles/account/src/ServicesProvider/ServicesProvider.php @@ -0,0 +1,444 @@ +extend('assets', function ($assets, $c) { + + // Register paths for user theme, if a user is logged in + // We catch any authorization-related exceptions, so that error pages can be rendered. + try { + /** @var UserFrosting\Sprinkle\Account\Authenticate\Authenticator $authenticator */ + $authenticator = $c->authenticator; + $currentUser = $c->currentUser; + } catch (\Exception $e) { + return $assets; + } + + if ($authenticator->check()) { + $c->sprinkleManager->addResource('assets', $currentUser->theme); + } + + return $assets; + }); + + /** + * Extend the 'classMapper' service to register model classes. + * + * Mappings added: User, Group, Role, Permission, Activity, PasswordReset, Verification + */ + $container->extend('classMapper', function ($classMapper, $c) { + $classMapper->setClassMapping('user', 'UserFrosting\Sprinkle\Account\Database\Models\User'); + $classMapper->setClassMapping('group', 'UserFrosting\Sprinkle\Account\Database\Models\Group'); + $classMapper->setClassMapping('role', 'UserFrosting\Sprinkle\Account\Database\Models\Role'); + $classMapper->setClassMapping('permission', 'UserFrosting\Sprinkle\Account\Database\Models\Permission'); + $classMapper->setClassMapping('activity', 'UserFrosting\Sprinkle\Account\Database\Models\Activity'); + $classMapper->setClassMapping('password_reset', 'UserFrosting\Sprinkle\Account\Database\Models\PasswordReset'); + $classMapper->setClassMapping('verification', 'UserFrosting\Sprinkle\Account\Database\Models\Verification'); + return $classMapper; + }); + + /** + * Extends the 'errorHandler' service with custom exception handlers. + * + * Custom handlers added: ForbiddenExceptionHandler + */ + $container->extend('errorHandler', function ($handler, $c) { + // Register the ForbiddenExceptionHandler. + $handler->registerHandler('\UserFrosting\Support\Exception\ForbiddenException', '\UserFrosting\Sprinkle\Account\Error\Handler\ForbiddenExceptionHandler'); + // Register the AuthExpiredExceptionHandler + $handler->registerHandler('\UserFrosting\Sprinkle\Account\Authenticate\Exception\AuthExpiredException', '\UserFrosting\Sprinkle\Account\Error\Handler\AuthExpiredExceptionHandler'); + // Register the AuthCompromisedExceptionHandler. + $handler->registerHandler('\UserFrosting\Sprinkle\Account\Authenticate\Exception\AuthCompromisedException', '\UserFrosting\Sprinkle\Account\Error\Handler\AuthCompromisedExceptionHandler'); + return $handler; + }); + + /** + * Extends the 'localePathBuilder' service, adding any locale files from the user theme. + * + */ + $container->extend('localePathBuilder', function ($pathBuilder, $c) { + // Add paths for user theme, if a user is logged in + // We catch any authorization-related exceptions, so that error pages can be rendered. + try { + /** @var UserFrosting\Sprinkle\Account\Authenticate\Authenticator $authenticator */ + $authenticator = $c->authenticator; + $currentUser = $c->currentUser; + } catch (\Exception $e) { + return $pathBuilder; + } + + if ($authenticator->check()) { + // Add paths to locale files for user theme + $themePath = $c->sprinkleManager->addResource('locale', $currentUser->theme); + + // Add user locale + $pathBuilder->addLocales($currentUser->locale); + } + + return $pathBuilder; + }); + + /** + * Extends the 'view' service with the AccountExtension for Twig. + * + * Adds account-specific functions, globals, filters, etc to Twig, and the path to templates for the user theme. + */ + $container->extend('view', function ($view, $c) { + $twig = $view->getEnvironment(); + $extension = new AccountExtension($c); + $twig->addExtension($extension); + + // Add paths for user theme, if a user is logged in + // We catch any authorization-related exceptions, so that error pages can be rendered. + try { + /** @var UserFrosting\Sprinkle\Account\Authenticate\Authenticator $authenticator */ + $authenticator = $c->authenticator; + $currentUser = $c->currentUser; + } catch (\Exception $e) { + return $view; + } + + if ($authenticator->check()) { + $theme = $currentUser->theme; + $themePath = $c->sprinkleManager->addResource('templates', $theme); + if ($themePath) { + $loader = $twig->getLoader(); + $loader->prependPath($themePath); + // Add namespaced path as well + $loader->addPath($themePath, $theme); + } + } + + return $view; + }); + + /** + * Authentication service. + * + * Supports logging in users, remembering their sessions, etc. + */ + $container['authenticator'] = function ($c) { + $classMapper = $c->classMapper; + $config = $c->config; + $session = $c->session; + $cache = $c->cache; + + // Force database connection to boot up + $c->db; + + // Fix RememberMe table name + $config['remember_me.table.tableName'] = Capsule::connection()->getTablePrefix() . $config['remember_me.table.tableName']; + + $authenticator = new Authenticator($classMapper, $session, $config, $cache); + return $authenticator; + }; + + /** + * Sets up the AuthGuard middleware, used to limit access to authenticated users for certain routes. + */ + $container['authGuard'] = function ($c) { + $authenticator = $c->authenticator; + return new AuthGuard($authenticator); + }; + + /** + * Authorization check logging with Monolog. + * + * Extend this service to push additional handlers onto the 'auth' log stack. + */ + $container['authLogger'] = function ($c) { + $logger = new Logger('auth'); + + $logFile = $c->get('locator')->findResource('log://userfrosting.log', true, true); + + $handler = new StreamHandler($logFile); + + $formatter = new MixedFormatter(null, null, true); + + $handler->setFormatter($formatter); + $logger->pushHandler($handler); + + return $logger; + }; + + /** + * Authorization service. + * + * Determines permissions for user actions. Extend this service to add additional access condition callbacks. + */ + $container['authorizer'] = function ($c) { + $config = $c->config; + + // Default access condition callbacks. Add more in your sprinkle by using $container->extend(...) + $callbacks = [ + /** + * Unconditionally grant permission - use carefully! + * @return bool returns true no matter what. + */ + 'always' => function () { + return true; + }, + + /** + * Check if the specified values are identical to one another (strict comparison). + * @param mixed $val1 the first value to compare. + * @param mixed $val2 the second value to compare. + * @return bool true if the values are strictly equal, false otherwise. + */ + 'equals' => function ($val1, $val2) { + return ($val1 === $val2); + }, + + /** + * Check if the specified values are numeric, and if so, if they are equal to each other. + * @param mixed $val1 the first value to compare. + * @param mixed $val2 the second value to compare. + * @return bool true if the values are numeric and equal, false otherwise. + */ + 'equals_num' => function ($val1, $val2) { + if (!is_numeric($val1)) { + return false; + } + if (!is_numeric($val2)) { + return false; + } + + return ($val1 == $val2); + }, + + /** + * Check if the specified user (by user_id) has a particular role. + * + * @param int $user_id the id of the user. + * @param int $role_id the id of the role. + * @return bool true if the user has the role, false otherwise. + */ + 'has_role' => function ($user_id, $role_id) { + return Capsule::table('role_users') + ->where('user_id', $user_id) + ->where('role_id', $role_id) + ->count() > 0; + }, + + /** + * Check if the specified value $needle is in the values of $haystack. + * + * @param mixed $needle the value to look for in $haystack + * @param array[mixed] $haystack the array of values to search. + * @return bool true if $needle is present in the values of $haystack, false otherwise. + */ + 'in' => function ($needle, $haystack) { + return in_array($needle, $haystack); + }, + + /** + * Check if the specified user (by user_id) is in a particular group. + * + * @param int $user_id the id of the user. + * @param int $group_id the id of the group. + * @return bool true if the user is in the group, false otherwise. + */ + 'in_group' => function ($user_id, $group_id) { + $user = User::find($user_id); + return ($user->group_id == $group_id); + }, + + /** + * Check if the specified user (by user_id) is the master user. + * + * @param int $user_id the id of the user. + * @return bool true if the user id is equal to the id of the master account, false otherwise. + */ + 'is_master' => function ($user_id) use ($config) { + // Need to use loose comparison for now, because some DBs return `id` as a string + return ($user_id == $config['reserved_user_ids.master']); + }, + + /** + * Check if all values in the array $needle are present in the values of $haystack. + * + * @param array[mixed] $needle the array whose values we should look for in $haystack + * @param array[mixed] $haystack the array of values to search. + * @return bool true if every value in $needle is present in the values of $haystack, false otherwise. + */ + 'subset' => function ($needle, $haystack) { + return count($needle) == count(array_intersect($needle, $haystack)); + }, + + /** + * Check if all keys of the array $needle are present in the values of $haystack. + * + * This function is useful for whitelisting an array of key-value parameters. + * @param array[mixed] $needle the array whose keys we should look for in $haystack + * @param array[mixed] $haystack the array of values to search. + * @return bool true if every key in $needle is present in the values of $haystack, false otherwise. + */ + 'subset_keys' => function ($needle, $haystack) { + return count($needle) == count(array_intersect(array_keys($needle), $haystack)); + } + ]; + + $authorizer = new AuthorizationManager($c, $callbacks); + return $authorizer; + }; + + /** + * Loads the User object for the currently logged-in user. + */ + $container['currentUser'] = function ($c) { + $authenticator = $c->authenticator; + + return $authenticator->user(); + }; + + $container['passwordHasher'] = function ($c) { + $hasher = new Hasher(); + return $hasher; + }; + + /** + * Returns a callback that forwards to dashboard if user is already logged in. + */ + $container['redirect.onAlreadyLoggedIn'] = function ($c) { + /** + * This method is invoked when a user attempts to perform certain public actions when they are already logged in. + * + * @todo Forward to user's landing page or last visited page + * @param \Psr\Http\Message\ServerRequestInterface $request + * @param \Psr\Http\Message\ResponseInterface $response + * @param array $args + * @return \Psr\Http\Message\ResponseInterface + */ + return function (Request $request, Response $response, array $args) use ($c) { + $redirect = $c->router->pathFor('dashboard'); + + return $response->withRedirect($redirect, 302); + }; + }; + + /** + * Returns a callback that handles setting the `UF-Redirect` header after a successful login. + */ + $container['redirect.onLogin'] = function ($c) { + /** + * This method is invoked when a user completes the login process. + * + * Returns a callback that handles setting the `UF-Redirect` header after a successful login. + * @param \Psr\Http\Message\ServerRequestInterface $request + * @param \Psr\Http\Message\ResponseInterface $response + * @param array $args + * @return \Psr\Http\Message\ResponseInterface + */ + return function (Request $request, Response $response, array $args) use ($c) { + // Backwards compatibility for the deprecated determineRedirectOnLogin service + if ($c->has('determineRedirectOnLogin')) { + $determineRedirectOnLogin = $c->determineRedirectOnLogin; + + return $determineRedirectOnLogin($response)->withStatus(200); + } + + /** @var UserFrosting\Sprinkle\Account\Authorize\AuthorizationManager */ + $authorizer = $c->authorizer; + + $currentUser = $c->authenticator->user(); + + if ($authorizer->checkAccess($currentUser, 'uri_account_settings')) { + return $response->withHeader('UF-Redirect', $c->router->pathFor('settings')); + } else { + return $response->withHeader('UF-Redirect', $c->router->pathFor('index')); + } + }; + }; + + /** + * Repository for password reset requests. + */ + $container['repoPasswordReset'] = function ($c) { + $classMapper = $c->classMapper; + $config = $c->config; + + $repo = new PasswordResetRepository($classMapper, $config['password_reset.algorithm']); + return $repo; + }; + + /** + * Repository for verification requests. + */ + $container['repoVerification'] = function ($c) { + $classMapper = $c->classMapper; + $config = $c->config; + + $repo = new VerificationRepository($classMapper, $config['verification.algorithm']); + return $repo; + }; + + /** + * Logger for logging the current user's activities to the database. + * + * Extend this service to push additional handlers onto the 'userActivity' log stack. + */ + $container['userActivityLogger'] = function ($c) { + $classMapper = $c->classMapper; + $config = $c->config; + $session = $c->session; + + $logger = new Logger('userActivity'); + + $handler = new UserActivityDatabaseHandler($classMapper, 'activity'); + + // Note that we get the user id from the session, not the currentUser service. + // This is because the currentUser service may not reflect the actual user during login/logout requests. + $currentUserIdKey = $config['session.keys.current_user_id']; + $userId = isset($session[$currentUserIdKey]) ? $session[$currentUserIdKey] : $config['reserved_user_ids.guest']; + $processor = new UserActivityProcessor($userId); + + $logger->pushProcessor($processor); + $logger->pushHandler($handler); + + return $logger; + }; + } +} diff --git a/main/app/sprinkles/account/src/Twig/AccountExtension.php b/main/app/sprinkles/account/src/Twig/AccountExtension.php new file mode 100755 index 0000000..12bacba --- /dev/null +++ b/main/app/sprinkles/account/src/Twig/AccountExtension.php @@ -0,0 +1,65 @@ +services = $services; + $this->config = $services->config; + } + + public function getName() + { + return 'userfrosting/account'; + } + + public function getFunctions() + { + return array( + // Add Twig function for checking permissions during dynamic menu rendering + new \Twig_SimpleFunction('checkAccess', function ($slug, $params = []) { + $authorizer = $this->services->authorizer; + $currentUser = $this->services->currentUser; + + return $authorizer->checkAccess($currentUser, $slug, $params); + }), + new \Twig_SimpleFunction('checkAuthenticated', function () { + $authenticator = $this->services->authenticator; + return $authenticator->check(); + }) + ); + } + + public function getGlobals() + { + try { + $currentUser = $this->services->currentUser; + } catch (\Exception $e) { + $currentUser = null; + } + + return [ + 'current_user' => $currentUser + ]; + } +} diff --git a/main/app/sprinkles/account/src/Util/HashFailedException.php b/main/app/sprinkles/account/src/Util/HashFailedException.php new file mode 100755 index 0000000..a0b37d1 --- /dev/null +++ b/main/app/sprinkles/account/src/Util/HashFailedException.php @@ -0,0 +1,21 @@ +staticMethod('user', 'where', 'user_name', $suggestion)->first()) { + return $suggestion; + } + } + } + + return ''; + } + +} diff --git a/main/app/sprinkles/account/templates/forms/settings-account.html.twig b/main/app/sprinkles/account/templates/forms/settings-account.html.twig new file mode 100755 index 0000000..996b27b --- /dev/null +++ b/main/app/sprinkles/account/templates/forms/settings-account.html.twig @@ -0,0 +1,37 @@ +
    +
    +

    {{translate("ACCOUNT.SETTINGS")}}

    +
    +
    + {% include "forms/csrf.html.twig" %} + + + + + {% block settings_account %} +
    + + +
    + {% if page.visibility != "disabled" %} +
    + + +
    +
    + + +
    +
    +
    + + +
    + {% endif %} + {% endblock %} +
    + +
    \ No newline at end of file diff --git a/main/app/sprinkles/account/templates/forms/settings-profile.html.twig b/main/app/sprinkles/account/templates/forms/settings-profile.html.twig new file mode 100755 index 0000000..0b0a788 --- /dev/null +++ b/main/app/sprinkles/account/templates/forms/settings-profile.html.twig @@ -0,0 +1,40 @@ +
    +
    +

    {{translate("PROFILE.SETTINGS")}}

    +
    +
    + {% include "forms/csrf.html.twig" %} + + {% block settings_profile %} + +
    +
    +
    + +
    +
    +
    +
    + +
    +
    +
    + +
    + + +

    {{translate("LOCALE.ACCOUNT")}}.

    +
    + {% endblock %} +
    + +
    diff --git a/main/app/sprinkles/account/templates/mail/password-reset.html.twig b/main/app/sprinkles/account/templates/mail/password-reset.html.twig new file mode 100755 index 0000000..37096ce --- /dev/null +++ b/main/app/sprinkles/account/templates/mail/password-reset.html.twig @@ -0,0 +1,22 @@ +{% block subject %} + {{site.title}} - your password reset request +{% endblock %} + +{% block body %} +

    Dear {{user.first_name}}, +

    +

    +A lost password request has been submitted for your account with {{site.title}} ({{site.uri.public}}) on {{request_date | date('m/d/Y g:i A')}}. +

    +

    +If you or someone you trust sent this request, and you wish to set a new password, please click this link: {{site.uri.public}}/account/set-password/confirm?token={{token}} +

    + +

    +If you did not expect this email, you may click this link to cancel the request: {{site.uri.public}}/account/set-password/deny?token={{token}}, or simply do nothing and the request will expire on its own. +

    +

    +With regards,
    +The {{site.title}} Team +

    +{% endblock %} \ No newline at end of file diff --git a/main/app/sprinkles/account/templates/mail/resend-verification.html.twig b/main/app/sprinkles/account/templates/mail/resend-verification.html.twig new file mode 100755 index 0000000..ba1c243 --- /dev/null +++ b/main/app/sprinkles/account/templates/mail/resend-verification.html.twig @@ -0,0 +1,17 @@ +{% block subject %} + {{site.title}} - verify your account +{% endblock %} + +{% block body %} +

    Dear {{user.first_name}}, +

    +

    +We have received a new verification request for your account with {{site.title}} ({{site.uri.public}}). Please follow the link below to verify your account. If your account is already active, please disregard this message. +

    +{{site.uri.public}}/account/verify?token={{token}} +

    +

    +With regards,
    +The {{site.title}} Team +

    +{% endblock %} diff --git a/main/app/sprinkles/account/templates/mail/verify-account.html.twig b/main/app/sprinkles/account/templates/mail/verify-account.html.twig new file mode 100755 index 0000000..aa342c7 --- /dev/null +++ b/main/app/sprinkles/account/templates/mail/verify-account.html.twig @@ -0,0 +1,21 @@ +{% block subject %} + Welcome to {{site.title}} - please verify your account +{% endblock %} + +{% block body %} +

    Dear {{user.first_name}}, +

    +

    +You are receiving this email because you registered with {{site.title}} ({{site.uri.public}}). +

    +

    +You will need to verify your account before you can login. Please follow the link below to verify your account. +

    +

    +{{site.uri.public}}/account/verify?token={{token}} +

    +

    +With regards,
    +The {{site.title}} Team +

    +{% endblock %} diff --git a/main/app/sprinkles/account/templates/modals/tos.html.twig b/main/app/sprinkles/account/templates/modals/tos.html.twig new file mode 100755 index 0000000..d51d897 --- /dev/null +++ b/main/app/sprinkles/account/templates/modals/tos.html.twig @@ -0,0 +1,16 @@ +{% extends 'modals/modal.html.twig' %} + +{% block modal_title %} + {{translate("TOS_FOR", {title: site.title})}} +{% endblock %} + +{% block modal_body %} +
    + {% include 'pages/partials/legal.html.twig' %} + {% include 'pages/partials/privacy.html.twig' %} +
    +{% endblock %} + +{% block modal_footer %} + +{% endblock %} diff --git a/main/app/sprinkles/account/templates/navigation/main-nav.html.twig b/main/app/sprinkles/account/templates/navigation/main-nav.html.twig new file mode 100755 index 0000000..e44c9c8 --- /dev/null +++ b/main/app/sprinkles/account/templates/navigation/main-nav.html.twig @@ -0,0 +1,13 @@ +{# This extend the same file from core to add a sign-up/sign-in or "my account" link to the "home page" nav menu. #} +{% extends "@core/navigation/main-nav.html.twig" %} + +{% block secondary_nav %} + {{parent()}} + {% if not checkAuthenticated() %} +
  • + {{translate("SIGNIN")}} +
  • + {% else %} + {% include "navigation/user-card.html.twig" %} + {% endif %} +{% endblock %} diff --git a/main/app/sprinkles/account/templates/navigation/user-card.html.twig b/main/app/sprinkles/account/templates/navigation/user-card.html.twig new file mode 100755 index 0000000..47e18f1 --- /dev/null +++ b/main/app/sprinkles/account/templates/navigation/user-card.html.twig @@ -0,0 +1,33 @@ +{% block userCard %} + +{% endblock %} diff --git a/main/app/sprinkles/account/templates/pages/account-settings.html.twig b/main/app/sprinkles/account/templates/pages/account-settings.html.twig new file mode 100755 index 0000000..61cd3d0 --- /dev/null +++ b/main/app/sprinkles/account/templates/pages/account-settings.html.twig @@ -0,0 +1,45 @@ +{% extends forcedLayout ? forcedLayout : "pages/abstract/default.html.twig" %} + +{% set page_active = "account-settings" %} + +{% block stylesheets_page %} + + {{ assets.css('css/form-widgets') | raw }} +{% endblock %} + +{# Overrides blocks in head of base template #} +{% block page_title %}{{translate("ACCOUNT.SETTINGS")}}{% endblock %} + +{% block page_description %}{{translate("ACCOUNT.SETTINGS.DESCRIPTION")}}{% endblock %} + +{% block body_matter %} + +
    +
    + {% block settings_profile_box %} +
    + {% include "forms/settings-profile.html.twig" %} +
    + {% endblock %} +
    +
    + {% block settings_account_box %} +
    + {% include "forms/settings-account.html.twig" %} +
    + {% endblock %} +
    +
    +{% endblock %} +{% block scripts_page %} + + + + + {{ assets.js('js/form-widgets') | raw }} + + + {{ assets.js('js/pages/account-settings') | raw }} +{% endblock %} diff --git a/main/app/sprinkles/account/templates/pages/error/compromised.html.twig b/main/app/sprinkles/account/templates/pages/error/compromised.html.twig new file mode 100755 index 0000000..6048619 --- /dev/null +++ b/main/app/sprinkles/account/templates/pages/error/compromised.html.twig @@ -0,0 +1,11 @@ +{% extends "pages/abstract/error.html.twig" %} + +{% block page_title %}{{ translate('ACCOUNT.SESSION_COMPROMISED.TITLE') }}{% endblock %} + +{% block page_description %}{{ translate('ACCOUNT.SESSION_COMPROMISED.TITLE') }}{% endblock %} + +{% block heading %} + {{ translate('ACCOUNT.SESSION_COMPROMISED.TEXT', { + 'url' : site.uri.public ~ '/account/sign-in' + }) | raw }} +{% endblock %} diff --git a/main/app/sprinkles/account/templates/pages/forgot-password.html.twig b/main/app/sprinkles/account/templates/pages/forgot-password.html.twig new file mode 100755 index 0000000..72b1a2a --- /dev/null +++ b/main/app/sprinkles/account/templates/pages/forgot-password.html.twig @@ -0,0 +1,46 @@ +{% extends "pages/abstract/base.html.twig" %} + +{# Overrides blocks in head of base template #} +{% block page_title %}{{translate("PASSWORD.FORGOTTEN")}}{% endblock %} + +{% block page_description %}{{translate("PASSWORD.FORGET.PAGE")}}{% endblock %} + +{% block body_attributes %} + class="hold-transition login-page" +{% endblock %} + +{% block content %} +