Compare commits
	
		
			316 Commits
		
	
	
		
			2.10.0
			...
			chill_amli
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 9814a9d64f | |||
| 93b754fba2 | |||
| 391bbfa32a | |||
| a2d85d4710 | |||
| add71ea373 | |||
| bd968dadaa | |||
| 1064fc3876 | |||
| 32187cfe06 | |||
| fdd537b18e | |||
| 1786fa838f | |||
| 9f2ecff63e | |||
| 78e00b8eba | |||
| e3764f6f91 | |||
| 95f7622923 | |||
| 2c46886e36 | |||
| 58eb089d1c | |||
| 6eff1962df | |||
| 75713af0e0 | |||
| 6cdb3033db | |||
| 7fe8d0837f | |||
| 753d6ea481 | |||
| 8cf25415ab | |||
| 89bdc76565 | |||
| 743e0b9403 | |||
| b9d4b5650b | |||
| 4fe4f6a877 | |||
| 4631f04da6 | |||
| e487bdf7fd | |||
| 37662a4187 | |||
| 97d6f35605 | |||
| cf9b9b3c75 | |||
| ca6fde934b | |||
| 01fb93e9e0 | |||
| 4e82126bed | |||
| 8b64933565 | |||
| 6c0715669e | |||
| 58ec294023 | |||
| 683717e572 | |||
| f763a1ed9f | |||
| 3c8dbe56fc | |||
| 27306015f4 | |||
| c059b7700e | |||
| 5b3cd9eb20 | |||
| eb112b8a85 | |||
| 390009b395 | |||
| 37dcbe92c0 | |||
| b25a1c3cbb | |||
| d599792de8 | |||
| 42c395ecc9 | |||
| 052c0e1969 | |||
| 59b22dbb6d | |||
| 35a2d08267 | |||
| 5af492b2db | |||
| 70c631b612 | |||
| 1869f44ec6 | |||
| 91af01336a | |||
| 37d49e1123 | |||
| d30ac75995 | |||
| d907357748 | |||
| d285e7f875 | |||
| d716e0c2c2 | |||
| 78ea990189 | |||
| 524123c701 | |||
| 38cb1fe357 | |||
| e379d8adb5 | |||
| 211a80e9be | |||
| 91a5db4c14 | |||
| 5dcd157bd0 | |||
| fb60808dca | |||
| fceab958bb | |||
| f07847e985 | |||
| c7e88b3924 | |||
| 424c9239b7 | |||
| b2e83892a7 | |||
| 478afc893b | |||
| 4cb6e8e564 | |||
| 9b1e464011 | |||
| 6c29638fed | |||
| 18a6a5a7eb | |||
| d85be8a92e | |||
| 42ea1f5813 | |||
| ef7a388f38 | |||
| 02b53e23e5 | |||
| bd45fbc85c | |||
| e488d6dadf | |||
| eafe68973a | |||
| 17494b3e9f | |||
| 661e5458ee | |||
| 307280f595 | |||
| d81afb89f2 | |||
| 0a0a692eae | |||
| 8cf9bf4a5f | |||
| b0d77a1656 | |||
| 967c8c62d4 | |||
| 12c37ddb2c | |||
| 822b96f87f | |||
| ebfb030ba6 | |||
| ff5fab5f50 | |||
| 5f2622d0d2 | |||
| 712c7bc492 | |||
| 5c2b2105b2 | |||
| a817b0bf4c | |||
| f10ec3991d | |||
| e550d64fd4 | |||
| 4aaf75a1a4 | |||
| 1dcff2f23c | |||
| 81359877c4 | |||
| 3851e65777 | |||
| 482abd3980 | |||
| d9e602247e | |||
| 9db43f1de3 | |||
| 674629e2bf | |||
| 1b8b99d5ce | |||
| 32c600c155 | |||
| d3a0c4c283 | |||
| e42d6c2d77 | |||
| 29cc589bf2 | |||
| eedfa60bea | |||
| 1c92f58b3b | |||
| d4f3ec368c | |||
| d245afb009 | |||
| 648ec68cfb | |||
| 3295031bcd | |||
| 75bdc335e5 | |||
| c442529799 | |||
| c4f0ad01d3 | |||
| 8688e4d502 | |||
| 374ad43df6 | |||
| cae9dddac4 | |||
| 436f1d6c61 | |||
| d86b86487f | |||
| bee5336c1d | |||
| 0d35dfc303 | |||
| 0784b00793 | |||
| b1bfb2dd95 | |||
| e42355c0d1 | |||
| 941d7b0352 | |||
| 0a06118ac3 | |||
| 3782cf35ff | |||
| 68c1833584 | |||
| 191b8ecf81 | |||
| e7ba42de1f | |||
| 8b145ebc11 | |||
| 722cf789ec | |||
| cec1588e91 | |||
| 1fff90d3a7 | |||
| 3da26a4d45 | |||
| 5662609c23 | |||
| 57277e5b87 | |||
| f72c0576ef | |||
| c8028b8e60 | |||
| 2400bd48d1 | |||
| c841821ed4 | |||
| 8830b6dc23 | |||
| accee53f5a | |||
| 20b70f9eed | |||
| d375abf593 | |||
| 7d281ea50f | |||
| e5aada5561 | |||
| 2d71ba9078 | |||
| 3df06e1eba | |||
| 26a0ba4756 | |||
| a604902074 | |||
| 8bbd3b01d9 | |||
| 3dcec5d44a | |||
| f513749780 | |||
| 93c5e83454 | |||
| 9c1324e9ec | |||
| 87403e509f | |||
| 0276ec1bc7 | |||
| 2a6974610f | |||
| 014e281d13 | |||
| 18be028a87 | |||
| a5b5eea146 | |||
| 71b6b158ba | |||
| d87c6305fd | |||
| 64d7c1fe99 | |||
| 4aa8436399 | |||
| 54ffa999d8 | |||
| 034269b87c | |||
| 8844e3e64a | |||
| e2634b0b0f | |||
| 9e93e2a3f9 | |||
| adad4313a6 | |||
| 849e7158e4 | |||
| 809a0a38ab | |||
| 13dae00a2c | |||
| ad63df85c7 | |||
| b5d5338002 | |||
| 922c5c5f5c | |||
| a9bc98738e | |||
| 3ea630748a | |||
| 75b2f6419e | |||
| a845fddf2e | |||
| ca44ebccf3 | |||
| d8f80f3d02 | |||
| 6b3b010631 | |||
| 5a5958704b | |||
| 8c99fc0089 | |||
| 6c4f116e85 | |||
| 5c08abc2f6 | |||
| 138d431786 | |||
| 03d64995d9 | |||
| 59e24ff39d | |||
| b7c3300884 | |||
| f987a6b5e0 | |||
| 3c685d50dd | |||
| 28c952521f | |||
| 4c0fef4f44 | |||
| 616be5cc8a | |||
| 9e4fd6183e | |||
| c92077926e | |||
| f149b24802 | |||
| 6e48a042b3 | |||
| 64e07c54fa | |||
| d95d97f8fe | |||
| e75b258e44 | |||
| c329862e96 | |||
| fbf1c4365d | |||
| 168aff81f4 | |||
| fc7d2fcca3 | |||
| 537fefee15 | |||
| dc22d18e6a | |||
| dab9204ec7 | |||
| 543d30acb9 | |||
| c31886fea3 | |||
| 6c3043f6fc | |||
| 3a88ea0b0f | |||
| c804462f15 | |||
| 089c92d0ee | |||
| 4217524e63 | |||
| 4fef8ec46e | |||
| 76b49d22e7 | |||
| 840c7e1084 | |||
| 5dcf7000f1 | |||
| ec536871aa | |||
| 1c79e25579 | |||
| 7c0bdc5abe | |||
| 59a64e9a62 | |||
| 782436ee2e | |||
| f962b7543f | |||
| b22f361368 | |||
| 5ea12a581b | |||
| eb8b8c6939 | |||
| 352b5b41b0 | |||
| 17778ab346 | |||
| 9dba558bef | |||
| 00e87f8c75 | |||
| a5998ce99d | |||
| d8e7a7f1af | |||
| d32a69a657 | |||
| 00ef58301e | |||
| 3fbdcdc431 | |||
| 7dcd5be735 | |||
| a025883a5d | |||
| 38c7c8de60 | |||
| f825c69ce5 | |||
| e2052fe71d | |||
| a349089f3d | |||
| 461fc1c41f | |||
| 2b770036a5 | |||
| 67fad5d764 | |||
| b6e0379583 | |||
| dba0e84781 | |||
| b89edc29af | |||
| b50c51bc2a | |||
| f68c69d443 | |||
| 682dc2174e | |||
| c0f9bf35ac | |||
| 0c61edd0d6 | |||
| 9ceb66e2da | |||
| 8b1271a466 | |||
| d921f6c9e5 | |||
| 85f00fdb28 | |||
| 6144f2439a | |||
| 1eee8c6c49 | |||
| a4caf733f1 | |||
| 4be3efc619 | |||
| 7859439f0b | |||
| c60a54eb39 | |||
| f4b1a25a67 | |||
| 0ec0708807 | |||
| d150a8251b | |||
| dd13e631ac | |||
| c892f7de7b | |||
| e895da31d7 | |||
| ba8a2327be | |||
| 38d828cf36 | |||
| ee4a6e08fb | |||
| d570145385 | |||
| 8abce5ab85 | |||
| 811798e23f | |||
| 9935af0497 | |||
| 5331f1becc | |||
| a7ec843509 | |||
| 0212193940 | |||
| aecdfa6b12 | |||
| 88784b7f7e | |||
| c339f7a828 | |||
| 5ceddc747d | |||
| f8a9cafacb | |||
| ce78177ab7 | |||
| 7c693b7495 | |||
| 6631e77df5 | |||
| c77af0bc4a | |||
| e28d17a131 | |||
| 2b1c3d1aff | |||
| f3444b6a58 | |||
| 901b496030 | |||
| 20e1feebf4 | |||
| d1f87718d6 | |||
| cebdfdaa30 | |||
| 5a1a33b6a4 | |||
| 28c0b8994b | |||
| 9c070cd8ae | |||
| 0f6dc3d997 | 
| @@ -18,10 +18,8 @@ max_line_length = 80 | ||||
|  | ||||
| [COMMIT_EDITMSG] | ||||
| max_line_length = 0 | ||||
| <<<<<<< Updated upstream | ||||
| ======= | ||||
|  | ||||
| [*.{js, vue, ts}] | ||||
| indent_size = 2 | ||||
| indent_style = space | ||||
| >>>>>>> Stashed changes | ||||
|  | ||||
|   | ||||
| @@ -11,6 +11,11 @@ and this project adheres to | ||||
| ## Unreleased | ||||
|  | ||||
| <!-- write down unreleased development here --> | ||||
| * [person][export] Fixed: rename the alias for `accompanying_period` to `acp` in filter associated with person | ||||
| * [activity][export] Feature: improve label for aliases in "Filter by activity type" | ||||
| * [activity][export] DX/Feature: use of an `ActivityTypeRepositoryInterface` instead of the old-style EntityRepository | ||||
| * [person][export] Fixed: some inconsistency with date filter on accompanying courses | ||||
| * [person][export] Fixed: use left join for related entities in accompanying course aggregators | ||||
|  | ||||
| ## Test releases | ||||
|  | ||||
|   | ||||
| @@ -19,10 +19,12 @@ | ||||
|         "graylog2/gelf-php": "^1.5", | ||||
|         "knplabs/knp-menu-bundle": "^3.0", | ||||
|         "knplabs/knp-time-bundle": "^1.12", | ||||
|         "knpuniversity/oauth2-client-bundle": "^2.10", | ||||
|         "league/csv": "^9.7.1", | ||||
|         "nyholm/psr7": "^1.4", | ||||
|         "ocramius/package-versions": "^1.10 || ^2", | ||||
|         "odolbeau/phone-number-bundle": "^3.6", | ||||
|         "ovh/ovh": "^3.0", | ||||
|         "phpoffice/phpspreadsheet": "^1.16", | ||||
|         "ramsey/uuid-doctrine": "^1.7", | ||||
|         "sensio/framework-extra-bundle": "^5.5", | ||||
| @@ -36,6 +38,7 @@ | ||||
|         "symfony/http-foundation": "^4.4", | ||||
|         "symfony/intl": "^4.4", | ||||
|         "symfony/mailer": "^5.4", | ||||
|         "symfony/messenger": "^5.4", | ||||
|         "symfony/mime": "^5.4", | ||||
|         "symfony/monolog-bundle": "^3.5", | ||||
|         "symfony/security-bundle": "^4.4", | ||||
| @@ -49,6 +52,7 @@ | ||||
|         "symfony/webpack-encore-bundle": "^1.11", | ||||
|         "symfony/workflow": "^4.4", | ||||
|         "symfony/yaml": "^4.4", | ||||
|         "thenetworg/oauth2-azure": "^2.0", | ||||
|         "twig/extra-bundle": "^3.0", | ||||
|         "twig/intl-extra": "^3.0", | ||||
|         "twig/markdown-extra": "^3.3", | ||||
|   | ||||
							
								
								
									
										320
									
								
								docs/source/installation/msgraph-configure.rst
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,320 @@ | ||||
|  | ||||
| Configure Chill for calendar sync and SSO with Microsoft Graph (Outlook) | ||||
| ======================================================================== | ||||
|  | ||||
| Chill offers the possibility to: | ||||
|  | ||||
| * authenticate users using Microsoft Graph, with relatively small adaptations; | ||||
| * synchronize calendar in both ways (`see the user manual for a large description of the feature <https://gitea.champs-libres.be/Chill-project/manuals>`_). | ||||
|  | ||||
| Both can be configured separately (synchronising calendars without SSO, or SSO without calendar). When calendar sync is configured without SSL, the user's email address is the key to associate Chill's users with Microsoft's ones. | ||||
|  | ||||
| Configure SSO | ||||
| ------------- | ||||
|  | ||||
| On Azure side | ||||
| ************* | ||||
|  | ||||
| Configure an app with the Azure interface, and give it the name of your choice. | ||||
|  | ||||
| Grab the tenant's ID for your app, which is visible on the main tab "Vue d'ensemble": | ||||
|  | ||||
| .. figure:: ./saml_login_id_general.png | ||||
|  | ||||
| This the variable which will be named :code:`SAML_IDP_APP_UUID`. | ||||
|  | ||||
| Go to the "Single sign-on" ("Authentication unique") section. Choose "SAML" as protocol, and fill those values: | ||||
|  | ||||
| .. figure:: ./saml_login_1.png | ||||
|  | ||||
| 1. The :code:`entityId` seems to be arbitrary. This will be your variable :code:`SAML_ENTITY_ID`; | ||||
| 2. The url response must be your Chill's URL appended by :code:`/saml/acs` | ||||
| 3. The only used attributes is :code:`emailaddress`, which must match the user's email one. | ||||
|  | ||||
| .. figure:: ./saml_login_2.png | ||||
|  | ||||
| You must download the certificate, as base64. The format for the download is :code:`cer`: you will remove the first and last line (the ones with :code:`-----BEGIN CERTIFICATE-----` and :code:`-----END CERTIFICATE-----`), and remove all the return line. The final result should be something as :code:`MIIAbcdef...XyZA=`. | ||||
|  | ||||
| This certificat will be your :code:`SAML_IDP_X509_CERT` variable. | ||||
|  | ||||
| The url login will be filled automatically with your tenant id. | ||||
|  | ||||
| Do not forget to provider user's accesses to your app, using the "Utilisateurs et groupes" tab: | ||||
|  | ||||
| .. figure:: ./saml_login_appro.png | ||||
|  | ||||
|  | ||||
| You must know have gathered all the required variables for SSO: | ||||
|  | ||||
| .. code-block::  | ||||
|  | ||||
|    SAML_BASE_URL=https://test.chill.be # must be | ||||
|    SAML_ENTITY_ID=https://test.chill.be # must match the one entered | ||||
|    SAML_IDP_APP_UUID=42XXXXXX-xxxx-xxxx-xxxx-xxxxxxxxxxxx | ||||
|    SAML_IDP_X509_CERT: MIIC...E8u3bk # truncated | ||||
|  | ||||
| Configure chill app | ||||
| ******************* | ||||
|  | ||||
| * add the bundle :code:`hslavich/oneloginsaml-bundle` | ||||
| * add the configuration file (see example above) | ||||
| * configure the security part (see example above) | ||||
| * add a user SAML factory into your src, and register it | ||||
|  | ||||
|  | ||||
| .. code-block:: yaml | ||||
|  | ||||
|    # config/packages/hslavich_onelogin.yaml | ||||
|  | ||||
|    parameters: | ||||
|      saml_base_url: '%env(resolve:SAML_BASE_URL)%' | ||||
|      saml_entity_id: '%env(resolve:SAML_ENTITY_ID)%' | ||||
|      saml_idp_x509cert: '%env(resolve:SAML_IDP_X509_CERT)%' | ||||
|      saml_idp_app_uuid: '%env(resolve:SAML_IDP_APP_UUID)%' | ||||
|  | ||||
|  | ||||
|    hslavich_onelogin_saml: | ||||
|      # Basic settings | ||||
|      idp: | ||||
|        entityId: 'https://sts.windows.net/%saml_idp_app_uuid%/' | ||||
|        singleSignOnService: | ||||
|          url: 'https://login.microsoftonline.com/%saml_idp_app_uuid%/saml2' | ||||
|          binding: 'urn:oasis:names:tc:SAML:2.0:bindings:HTTP-Redirect' | ||||
|        singleLogoutService: | ||||
|          url: 'https://login.microsoftonline.com/%saml_idp_app_uuid%/saml2' | ||||
|          binding: 'urn:oasis:names:tc:SAML:2.0:bindings:HTTP-Redirect' | ||||
|        x509cert: '%saml_idp_x509cert%' | ||||
|      sp: | ||||
|        entityId: '%saml_entity_id%' | ||||
|        assertionConsumerService: | ||||
|          url: '%saml_base_url%/saml/acs' | ||||
|          binding: 'urn:oasis:names:tc:SAML:2.0:bindings:HTTP-POST' | ||||
|        singleLogoutService: | ||||
|          url: '%saml_base_url%/saml/' | ||||
|          binding: 'urn:oasis:names:tc:SAML:2.0:bindings:HTTP-Redirect' | ||||
|        privateKey: '' | ||||
|      # Optional settings. | ||||
|      baseurl: '%saml_base_url%/saml' | ||||
|      strict: true | ||||
|      debug: true | ||||
|      security: | ||||
|        nameIdEncrypted:       false | ||||
|        authnRequestsSigned:   false | ||||
|        logoutRequestSigned:   false | ||||
|        logoutResponseSigned:  false | ||||
|        wantMessagesSigned:    false | ||||
|        wantAssertionsSigned:  false | ||||
|        wantNameIdEncrypted:   false | ||||
|        requestedAuthnContext: true | ||||
|        signMetadata: false | ||||
|        wantXMLValidation: true | ||||
|        signatureAlgorithm: 'http://www.w3.org/2001/04/xmldsig-more#rsa-sha256' | ||||
|        digestAlgorithm: 'http://www.w3.org/2001/04/xmlenc#sha256' | ||||
|      contactPerson: | ||||
|        technical: | ||||
|          givenName: 'Tech User' | ||||
|          emailAddress: 'techuser@example.com' | ||||
|        support: | ||||
|          givenName: 'Support User' | ||||
|          emailAddress: 'supportuser@example.com' | ||||
|      organization: | ||||
|        en: | ||||
|          name: 'Example' | ||||
|          displayname: 'Example' | ||||
|          url: 'http://example.com' | ||||
|  | ||||
|  | ||||
| .. code-block:: yaml | ||||
|  | ||||
|    # config/security.yaml | ||||
|    # merge this with other existing configurations | ||||
|  | ||||
|    security: | ||||
|  | ||||
|  | ||||
|        providers: | ||||
|            saml_provider: | ||||
|                # Loads user from user repository | ||||
|                entity: | ||||
|                    class: Chill\MainBundle\Entity\User | ||||
|                    property: username | ||||
|  | ||||
|        firewalls: | ||||
|  | ||||
|  | ||||
|            default: | ||||
|                # saml part: | ||||
|                saml: | ||||
|                    username_attribute: http://schemas.xmlsoap.org/ws/2005/05/identity/claims/emailaddress | ||||
|                    # weird behaviour in dev environment... configuration seems different | ||||
|                    # username_attribute: http://schemas.xmlsoap.org/ws/2005/05/identity/claims/name | ||||
|                    # Use the attribute's friendlyName instead of the name | ||||
|                    use_attribute_friendly_name: false | ||||
|                    user_factory: user_from_saml_factory | ||||
|                    persist_user: true | ||||
|                    check_path: saml_acs | ||||
|                    login_path: saml_login | ||||
|                logout: | ||||
|                    path: /saml/logout | ||||
|  | ||||
|  | ||||
| .. code-block:: php | ||||
|  | ||||
|    // src/Security/SamlFactory.php | ||||
|  | ||||
|    namespace App\Security; | ||||
|  | ||||
|    use Chill\MainBundle\Entity\User; | ||||
|    use Hslavich\OneloginSamlBundle\Security\Authentication\Token\SamlTokenInterface; | ||||
|    use Hslavich\OneloginSamlBundle\Security\User\SamlUserFactoryInterface; | ||||
|  | ||||
|    class UserSamlFactory implements SamlUserFactoryInterface | ||||
|    { | ||||
|        public function createUser(SamlTokenInterface $token) | ||||
|        { | ||||
|            $attributes = $token->getAttributes(); | ||||
|            $user = new User(); | ||||
|            $user->setUsername($attributes['http://schemas.microsoft.com/identity/claims/displayname'][0]); | ||||
|            $user->setLabel($attributes['http://schemas.microsoft.com/identity/claims/displayname'][0]); | ||||
|            $user->setPassword(''); | ||||
|            $user->setEmail($attributes['http://schemas.xmlsoap.org/ws/2005/05/identity/claims/emailaddress'][0]); | ||||
|            $user->setAttributes($attributes); | ||||
|  | ||||
|            return $user; | ||||
|        } | ||||
|    } | ||||
|  | ||||
|  | ||||
|  | ||||
| Configure sync | ||||
| -------------- | ||||
|  | ||||
| The sync processe might be configured in the same app, or into a different app. | ||||
|  | ||||
| The synchronization processes use Oauth2.0 for authentication and authorization. | ||||
|  | ||||
| .. note:: | ||||
|  | ||||
|    Two flows are in use: | ||||
|  | ||||
|    * we authenticate "on behalf of a user", to allow users to see their own calendar or other user's calendar into the web interface.  | ||||
|  | ||||
|      Typically, when the page is loaded, Chill first check that an authorization token exists. If not, the user is redirected to Microsoft Azure for authentification and a new token is grabbed (most of the times, this is transparent for users).  | ||||
|  | ||||
|    * Chill also acts "as a machine", to synchronize calendars with a daemon background. | ||||
|  | ||||
| One can access the configuration using this screen (it is quite well hidden into the multiple of tabs): | ||||
|  | ||||
| .. figure:: ./oauth_app_registration.png | ||||
|  | ||||
|    You can find the oauth configuration on the "Securité > Autorisations" tab, and click on "application registration" (not translated). | ||||
|  | ||||
| Add a redirection URI for you authentification: | ||||
|  | ||||
| .. figure:: ./oauth_api_authentification.png | ||||
|  | ||||
|    The URI must be "your chill public url" with :code:`/connect/azure/check` at the end. | ||||
|  | ||||
| Allow some authorizations for your app: | ||||
|  | ||||
| .. figure:: ./oauth_api_autorisees.png | ||||
|  | ||||
| Take care of the separation between autorization "on behalf of a user" (déléguée), or "for a machine" (application). | ||||
|  | ||||
| Some explanation: | ||||
|  | ||||
| * Users must be allowed to read their user profile (:code:`User.Read`), and the profile of other users (:code:`User.ReadBasicAll`); | ||||
| * They must be allowed to read their calendar (:code:`Calendars.Read`), and the calendars shared with them (:code:`Calendars.Read.Shared`); | ||||
|  | ||||
| The sync daemon must have write access: | ||||
|  | ||||
| * the daemon must be allowed to read all users and their profile, to establish a link between them and the Chill's users: (:code:`Users.Read.All`); | ||||
| * it must also be allowed to read and write into the calendars (:code:`Calendars.ReadWrite.All`) | ||||
| * for sending invitation to other users, the permission (:code:`Mail.Send`) must be granted. | ||||
|  | ||||
| At this step, you might choose to accept those permissions for all users, or let them do it by yourself. | ||||
|  | ||||
| Grab your client id: | ||||
|  | ||||
| .. figure:: ./oauth_api_client_id.png | ||||
|  | ||||
| This will be your :code:`OAUTH_AZURE_CLIENT_ID` variable. | ||||
|  | ||||
|  | ||||
| Generate a secret: | ||||
|  | ||||
| .. figure:: ./oauth_api_secret.png | ||||
|  | ||||
| This will be your :code:`OAUTH_AZURE_CLIENT_SECRET` variable. | ||||
|  | ||||
| And get you azure's tenant id, which is the same as the :code:`SAML_IDP_APP_UUID` (see above). | ||||
|  | ||||
| Your variables will be: | ||||
|  | ||||
| .. code-block:: | ||||
|  | ||||
|    OAUTH_AZURE_CLIENT_ID=xxxxxxxx-xxxx-xxxx-xxxx-xxxxxxxxxxxx | ||||
|    OAUTH_AZURE_CLIENT_TENANT=xxxxxxxx-xxxx-xxxx-xxxx-xxxxxxxxxxxx | ||||
|    OAUTH_AZURE_CLIENT_SECRET: 3-xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx | ||||
|  | ||||
| Then, configure chill: | ||||
|  | ||||
| Enable the calendar sync with microsoft azure: | ||||
|  | ||||
| .. code-block:: yaml | ||||
|  | ||||
|    # config/packages/chill_calendar.yaml | ||||
|  | ||||
|    chill_calendar: | ||||
|        remote_calendars_sync: | ||||
|            microsoft_graph: | ||||
|                enabled: true | ||||
|  | ||||
| and configure the oauth client: | ||||
|  | ||||
| .. code-block:: yaml | ||||
|  | ||||
|    # config/packages/knp_oauth2_client.yaml | ||||
|    knpu_oauth2_client: | ||||
|        clients: | ||||
|            azure: | ||||
|                type: azure | ||||
|                client_id: '%env(OAUTH_AZURE_CLIENT_ID)%' | ||||
|                client_secret: '%env(OAUTH_AZURE_CLIENT_SECRET)%' | ||||
|                redirect_route: chill_calendar_remote_connect_azure_check | ||||
|                redirect_params: { } | ||||
|                tenant: '%env(OAUTH_AZURE_CLIENT_TENANT)%' | ||||
|                url_api: 'https://graph.microsoft.com/' | ||||
|                default_end_point_version: '2.0' | ||||
|  | ||||
|  | ||||
| You can now process for the first api authorization on the application side, (unless you did it in the Azure interface), and get a first token, by using : | ||||
|  | ||||
| :code:`bin/console chill:calendar:msgraph-grant-admin-consent` | ||||
|  | ||||
| This will generate a url that you can use to grant your app for your tenant. The redirection may fails in the browser, but this is not relevant: if you get an authorization token in the CLI, the authentication works. | ||||
|  | ||||
| Run the processes to synchronize | ||||
| -------------------------------- | ||||
|  | ||||
| The calendar synchronization is processed using symfony messenger. It seems to be intersting to configure a queue (in the postgresql database it is the most simple way), and to run a worker for synchronization, at least in production. | ||||
|  | ||||
| The association between chill's users and Microsoft's users is done by this cli command: | ||||
|  | ||||
| .. code-block::  | ||||
|  | ||||
|    bin/console chill:calendar:msgraph-user-map-subscribe | ||||
|  | ||||
| This command: | ||||
|  | ||||
| * will associate the Microsoft's user metadata in our database; | ||||
| * and, most important, create a subscription to get notification when the user alter his calendar, to sync chill's event and ranges in sync. | ||||
|  | ||||
| The subscription least at most 3 days. This command should be runned: | ||||
|  | ||||
| * at least each time a user is added; | ||||
| * and, at least, every three days. | ||||
|  | ||||
| In production, we advise to run it at least every day to get the sync working. | ||||
|  | ||||
|  | ||||
							
								
								
									
										
											BIN
										
									
								
								docs/source/installation/oauth_api_authentification.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 61 KiB | 
							
								
								
									
										
											BIN
										
									
								
								docs/source/installation/oauth_api_autorisees.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 149 KiB | 
							
								
								
									
										
											BIN
										
									
								
								docs/source/installation/oauth_api_client_id.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 105 KiB | 
							
								
								
									
										
											BIN
										
									
								
								docs/source/installation/oauth_api_secret.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 91 KiB | 
							
								
								
									
										
											BIN
										
									
								
								docs/source/installation/oauth_app_registration.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 116 KiB | 
							
								
								
									
										27
									
								
								docs/source/installation/prod-calendar-sms-sending.rst
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,27 @@ | ||||
|  | ||||
| Send short messages (SMS) with calendar bundle | ||||
| ============================================== | ||||
|  | ||||
| To activate the sending of messages, you should run this command on a regularly basis (using, for instance, a cronjob): | ||||
|  | ||||
| .. code-block:: bash | ||||
|  | ||||
|    bin/console chill:calendar:send-short-messages | ||||
|  | ||||
| A transporter must be configured for the message to be effectively sent.  | ||||
|  | ||||
| Configure OVH Transporter | ||||
| ------------------------- | ||||
|  | ||||
| Currently, this is the only one transporter available. | ||||
|  | ||||
| For configuring this, simply add this config variable in your environment: | ||||
|  | ||||
| ```env | ||||
| SHORT_MESSAGE_DSN=ovh://applicationKey:applicationSecret@endpoint?consumerKey=xxxx&sender=yyyy&service_name=zzzz | ||||
| ``` | ||||
|  | ||||
| In order to generate the application key, secret, and consumerKey, refers to their `documentation <https://docs.ovh.com/gb/en/api/first-steps-with-ovh-api/>`_. | ||||
|  | ||||
| Before to be able to send your first sms, you must enable your account, grab some credits, and configure a sender. The service_name is an internal configuration generated by OVH. | ||||
|  | ||||
							
								
								
									
										48
									
								
								docs/source/installation/prod.rst
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,48 @@ | ||||
| .. Copyright (C)  2014-2019 Champs Libres Cooperative SCRLFS | ||||
| Permission is granted to copy, distribute and/or modify this document | ||||
| under the terms of the GNU Free Documentation License, Version 1.3 | ||||
| or any later version published by the Free Software Foundation; | ||||
| with no Invariant Sections, no Front-Cover Texts, and no Back-Cover Texts. | ||||
| A copy of the license is included in the section entitled "GNU | ||||
| Free Documentation License". | ||||
|  | ||||
| Installation for production | ||||
| ########################### | ||||
|  | ||||
| An installation use these services, which are deployed using docker containers: | ||||
|  | ||||
| * a php-fpm image, which run the Php and Symfony code for Chill; | ||||
| * a nginx image, which serves the assets, and usually proxy the php requests to the fpm image; | ||||
| * a redis server, which stores the cache, sessions (this is currently hardcoded in the php image), and some useful keys (like wopi locks); | ||||
| * a postgresql database. The use of postgresql is mandatory; | ||||
| * a relatorio service, which transform odt templates to full documents (replacing the placeholders); | ||||
|  | ||||
| Some external services: | ||||
|  | ||||
| * (required) an openstack object store, configured with `temporary url <https://docs.openstack.org/swift/latest/api/temporary_url_middleware.html>` configured (no openstack users is required). This is currently the only way to store documents from chill; | ||||
| * a mailer service (SMTP) | ||||
| * (optional) a service for verifying phone number. Currently, only Twilio is possible; | ||||
| * (optional) a service for sending Short Messages (SMS). Currently, only Ovh is possible; | ||||
|  | ||||
| The `docker-compose.yaml` file of chill app is a basis for a production install. The environment variable in the ```.env``` and ```.env.prod``` should be overriden by environment variables, or ```.env.local``` files. | ||||
|  | ||||
| This should be adapted to your needs: | ||||
|  | ||||
| * The image for php and nginx apps are pre-compiled images, with the default configuration and bundle. If they do not fullfill your needs, you should compile your own images. | ||||
|  | ||||
|    .. TODO: | ||||
|  | ||||
|       As the time of writing (2022-07-03) those images are not published yet. | ||||
|  | ||||
| * Think about how you will backup your database. Some adminsys find easier to store database outside of docker, which might be easier to administrate or replicate. | ||||
|  | ||||
| Tweak symfony messenger | ||||
| ======================= | ||||
|  | ||||
| Calendar sync is processed using symfony messenger. | ||||
|  | ||||
| You can tweak the configuration  | ||||
|  | ||||
| Going further: | ||||
|  | ||||
| * Configure the saml login and synchronisation with Outlook api | ||||
							
								
								
									
										
											BIN
										
									
								
								docs/source/installation/saml_login_1.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 57 KiB | 
							
								
								
									
										
											BIN
										
									
								
								docs/source/installation/saml_login_2.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 69 KiB | 
							
								
								
									
										
											BIN
										
									
								
								docs/source/installation/saml_login_appro.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 92 KiB | 
							
								
								
									
										
											BIN
										
									
								
								docs/source/installation/saml_login_id_general.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 87 KiB | 
							
								
								
									
										63
									
								
								exports_alias_conventions.csv
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,63 @@ | ||||
| Entity,Join,Attribute,Alias | ||||
| AccompanyingPeriod::class,,,acp | ||||
| ,AccompanyingPeriodWork::class,acp.works,acpw | ||||
| ,AccompanyingPeriodParticipation::class,acp.participations,acppart | ||||
| ,Location::class,acp.administrativeLocation,acploc | ||||
| ,ClosingMotive::class,acp.closingMotive,acpmotive | ||||
| ,UserJob::class,acp.job,acpjob | ||||
| ,Origin::class,acp.origin,acporigin | ||||
| ,Scope::class,acp.scopes,acpscope | ||||
| ,SocialIssue::class,acp.socialIssues,acpsocialissue | ||||
| ,User::class,acp.user,acpuser | ||||
| AccompanyingPeriodWork::class,,,acpw | ||||
| ,AccompanyingPeriodWorkEvaluation::class,acpw.accompanyingPeriodWorkEvaluations,workeval | ||||
| ,User::class,acpw.referrers,acpwuser | ||||
| ,SocialAction::class,acpw.socialAction,acpwsocialaction | ||||
| ,Goal::class,acpw.goals,goal | ||||
| ,Result::class,acpw.results,result | ||||
| AccompanyingPeriodParticipation::class,,,acppart | ||||
| ,Person::class,acppart.person,partperson | ||||
| AccompanyingPeriodWorkEvaluation::class,,,workeval | ||||
| ,Evaluation::class,workeval.evaluation,eval | ||||
| Goal::class,,,goal | ||||
| ,Result::class,goal.results,goalresult | ||||
| Person::class,,,person | ||||
| ,Center::class,person.center,center | ||||
| ,HouseholdMember::class,partperson.householdParticipations,householdmember | ||||
| ,MaritalStatus::class,person.maritalStatus,personmarital | ||||
| ,VendeePerson::class,,vp | ||||
| ,VendeePersonMineur::class,,vpm | ||||
| ResidentialAddress::class,,,resaddr | ||||
| ,ThirdParty::class,resaddr.hostThirdParty,tparty | ||||
| ThirdParty::class,,,tparty | ||||
| ,ThirdPartyCategory::class,tparty.categories,tpartycat | ||||
| HouseholdMember::class,,,householdmember | ||||
| ,Household::class,householdmember.household,household | ||||
| ,Person::class,householdmember.person,memberperson | ||||
| ,,memberperson.center,membercenter | ||||
| Household::class,,,household | ||||
| ,HouseholdComposition::class,household.compositions,composition | ||||
| Activity::class,,,activity | ||||
| ,Person::class,activity.person,actperson | ||||
| ,AccompanyingPeriod::class,activity.accompanyingPeriod,acp | ||||
| ,Person::class,activity_person_having_activity.person,person_person_having_activity | ||||
| ,ActivityReason::class,activity_person_having_activity.reasons,reasons_person_having_activity | ||||
| ,ActivityType::class,activity.activityType,acttype | ||||
| ,Location::class,activity.location,actloc | ||||
| ,SocialAction::class,activity.socialActions,actsocialaction | ||||
| ,SocialIssue::class,activity.socialIssues,actsocialssue | ||||
| ,ThirdParty::class,activity.thirdParties,acttparty | ||||
| ,User::class,activity.user,actuser | ||||
| ,User::class,activity.users,actusers | ||||
| ,ActivityReason::class,activity.reasons,actreasons | ||||
| ,Center::class,actperson.center,actcenter | ||||
| ActivityReason::class,,,actreasons | ||||
| ,ActivityReasonCategory::class,actreason.category,actreasoncat | ||||
| Calendar::class,,,cal | ||||
| ,CancelReason::class,cal.cancelReason,calcancel | ||||
| ,Location::class,cal.location,calloc | ||||
| ,User::class,cal.user,caluser | ||||
| VendeePerson::class,,,vp | ||||
| ,SituationProfessionelle::class,vp.situationProfessionelle,vpprof | ||||
| ,StatutLogement::class,vp.statutLogement,vplog | ||||
| ,TempsDeTravail::class,vp.tempsDeTravail,vptt | ||||
| 
 | 
							
								
								
									
										71
									
								
								exports_alias_conventions.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,71 @@ | ||||
| # Export conventions | ||||
|  | ||||
|  | ||||
| Add condition with distinct alias on each export join clauses (Indicators + Filters + Aggregators) | ||||
|  | ||||
| These are alias conventions : | ||||
|  | ||||
| | Entity                                  | Join                                    | Attribute                                  | Alias                             | | ||||
| |:----------------------------------------|:----------------------------------------|:-------------------------------------------|:----------------------------------| | ||||
| | AccompanyingPeriod::class               |                                         |                                            | acp                               | | ||||
| |                                         | AccompanyingPeriodWork::class           | acp.works                                  | acpw                              | | ||||
| |                                         | AccompanyingPeriodParticipation::class  | acp.participations                         | acppart                           | | ||||
| |                                         | Location::class                         | acp.administrativeLocation                 | acploc                            | | ||||
| |                                         | ClosingMotive::class                    | acp.closingMotive                          | acpmotive                         | | ||||
| |                                         | UserJob::class                          | acp.job                                    | acpjob                            | | ||||
| |                                         | Origin::class                           | acp.origin                                 | acporigin                         | | ||||
| |                                         | Scope::class                            | acp.scopes                                 | acpscope                          | | ||||
| |                                         | SocialIssue::class                      | acp.socialIssues                           | acpsocialissue                    | | ||||
| |                                         | User::class                             | acp.user                                   | acpuser                           | | ||||
| | AccompanyingPeriodWork::class           |                                         |                                            | acpw                              | | ||||
| |                                         | AccompanyingPeriodWorkEvaluation::class | acpw.accompanyingPeriodWorkEvaluations     | workeval                          | | ||||
| |                                         | User::class                             | acpw.referrers                             | acpwuser                          | | ||||
| |                                         | SocialAction::class                     | acpw.socialAction                          | acpwsocialaction                  | | ||||
| |                                         | Goal::class                             | acpw.goals                                 | goal                              | | ||||
| |                                         | Result::class                           | acpw.results                               | result                        | | ||||
| | AccompanyingPeriodParticipation::class  |                                         |                                            | acppart                           | | ||||
| |                                         | Person::class                           | acppart.person                             | partperson                        | | ||||
| | AccompanyingPeriodWorkEvaluation::class |                                         |                                            | workeval                          | | ||||
| |                                         | Evaluation::class                       | workeval.evaluation                        | eval                              | | ||||
| | Goal::class                             |                                         |                                            | goal                              | | ||||
| |                                         | Result::class                           | goal.results                               | goalresult                        | | ||||
| | Person::class                           |                                         |                                            | person                            | | ||||
| |                                         | Center::class                           | person.center                              | center                            | | ||||
| |                                         | HouseholdMember::class                  | partperson.householdParticipations         | householdmember                   | | ||||
| |                                         | MaritalStatus::class                    | person.maritalStatus                       | personmarital                     | | ||||
| |                                         | VendeePerson::class                     |                                            | vp                                | | ||||
| |                                         | VendeePersonMineur::class               |                                            | vpm                               | | ||||
| | ResidentialAddress::class               |                                         |                                            | resaddr                           | | ||||
| |                                         | ThirdParty::class                       | resaddr.hostThirdParty                     | tparty                            | | ||||
| | ThirdParty::class                       |                                         |                                            | tparty                            | | ||||
| |                                         | ThirdPartyCategory::class               | tparty.categories                          | tpartycat                         | | ||||
| | HouseholdMember::class                  |                                         |                                            | householdmember                   | | ||||
| |                                         | Household::class                        | householdmember.household                  | household                         | | ||||
| |                                         | Person::class                           | householdmember.person                     | memberperson                      | | ||||
| |                                         |                                         | memberperson.center                        | membercenter                      | | ||||
| | Household::class                        |                                         |                                            | household                         | | ||||
| |                                         | HouseholdComposition::class             | household.compositions                     | composition                       | | ||||
| | Activity::class                         |                                         |                                            | activity                          | | ||||
| |                                         | Person::class                           | activity.person                            | actperson                         | | ||||
| |                                         | AccompanyingPeriod::class               | activity.accompanyingPeriod                | acp                            | | ||||
| |                                         | Person::class                           | activity\_person\_having\_activity.person  | person\_person\_having\_activity  | | ||||
| |                                         | ActivityReason::class                   | activity\_person\_having\_activity.reasons | reasons\_person\_having\_activity | | ||||
| |                                         | ActivityType::class                     | activity.activityType                      | acttype                           | | ||||
| |                                         | Location::class                         | activity.location                          | actloc                            | | ||||
| |                                         | SocialAction::class                     | activity.socialActions                     | actsocialaction                   | | ||||
| |                                         | SocialIssue::class                      | activity.socialIssues                      | actsocialssue                     | | ||||
| |                                         | ThirdParty::class                       | activity.thirdParties                      | acttparty                         | | ||||
| |                                         | User::class                             | activity.user                              | actuser                           | | ||||
| |                                         | User::class                             | activity.users                             | actusers                          | | ||||
| |                                         | ActivityReason::class                   | activity.reasons                           | actreasons                        | | ||||
| |                                         | Center::class                           | actperson.center                           | actcenter                         | | ||||
| | ActivityReason::class                   |                                         |                                            | actreasons                        | | ||||
| |                                         | ActivityReasonCategory::class           | actreason.category                         | actreasoncat                      | | ||||
| | Calendar::class                         |                                         |                                            | cal                               | | ||||
| |                                         | CancelReason::class                     | cal.cancelReason                           | calcancel                         | | ||||
| |                                         | Location::class                         | cal.location                               | calloc                            | | ||||
| |                                         | User::class                             | cal.user                                   | caluser                           | | ||||
| | VendeePerson::class                     |                                         |                                            | vp                                | | ||||
| |                                         | SituationProfessionelle::class          | vp.situationProfessionelle                 | vpprof                            | | ||||
| |                                         | StatutLogement::class                   | vp.statutLogement                          | vplog                             | | ||||
| |                                         | TempsDeTravail::class                   | vp.tempsDeTravail                          | vptt                              | | ||||
| @@ -17,10 +17,11 @@ use Chill\ActivityBundle\Form\ActivityType; | ||||
| use Chill\ActivityBundle\Repository\ActivityACLAwareRepositoryInterface; | ||||
| use Chill\ActivityBundle\Repository\ActivityRepository; | ||||
| use Chill\ActivityBundle\Repository\ActivityTypeCategoryRepository; | ||||
| use Chill\ActivityBundle\Repository\ActivityTypeRepository; | ||||
| use Chill\ActivityBundle\Repository\ActivityTypeRepositoryInterface; | ||||
| use Chill\ActivityBundle\Security\Authorization\ActivityVoter; | ||||
| use Chill\MainBundle\Entity\Embeddable\CommentEmbeddable; | ||||
| use Chill\MainBundle\Repository\LocationRepository; | ||||
| use Chill\MainBundle\Repository\UserRepositoryInterface; | ||||
| use Chill\MainBundle\Security\Resolver\CenterResolverManagerInterface; | ||||
| use Chill\PersonBundle\Entity\AccompanyingPeriod; | ||||
| use Chill\PersonBundle\Entity\Person; | ||||
| @@ -55,7 +56,7 @@ final class ActivityController extends AbstractController | ||||
|  | ||||
|     private ActivityTypeCategoryRepository $activityTypeCategoryRepository; | ||||
|  | ||||
|     private ActivityTypeRepository $activityTypeRepository; | ||||
|     private ActivityTypeRepositoryInterface $activityTypeRepository; | ||||
|  | ||||
|     private CenterResolverManagerInterface $centerResolver; | ||||
|  | ||||
| @@ -73,9 +74,11 @@ final class ActivityController extends AbstractController | ||||
|  | ||||
|     private ThirdPartyRepository $thirdPartyRepository; | ||||
|  | ||||
|     private UserRepositoryInterface $userRepository; | ||||
|  | ||||
|     public function __construct( | ||||
|         ActivityACLAwareRepositoryInterface $activityACLAwareRepository, | ||||
|         ActivityTypeRepository $activityTypeRepository, | ||||
|         ActivityTypeRepositoryInterface $activityTypeRepository, | ||||
|         ActivityTypeCategoryRepository $activityTypeCategoryRepository, | ||||
|         PersonRepository $personRepository, | ||||
|         ThirdPartyRepository $thirdPartyRepository, | ||||
| @@ -86,6 +89,7 @@ final class ActivityController extends AbstractController | ||||
|         EventDispatcherInterface $eventDispatcher, | ||||
|         LoggerInterface $logger, | ||||
|         SerializerInterface $serializer, | ||||
|         UserRepositoryInterface $userRepository, | ||||
|         CenterResolverManagerInterface $centerResolver | ||||
|     ) { | ||||
|         $this->activityACLAwareRepository = $activityACLAwareRepository; | ||||
| @@ -100,6 +104,7 @@ final class ActivityController extends AbstractController | ||||
|         $this->eventDispatcher = $eventDispatcher; | ||||
|         $this->logger = $logger; | ||||
|         $this->serializer = $serializer; | ||||
|         $this->userRepository = $userRepository; | ||||
|         $this->centerResolver = $centerResolver; | ||||
|     } | ||||
|  | ||||
| @@ -371,7 +376,7 @@ final class ActivityController extends AbstractController | ||||
|         if ($request->query->has('activityData')) { | ||||
|             $activityData = $request->query->get('activityData'); | ||||
|  | ||||
|             if (array_key_exists('durationTime', $activityData)) { | ||||
|             if (array_key_exists('durationTime', $activityData) && $activityType->getDurationTimeVisible() > 0) { | ||||
|                 $durationTimeInMinutes = $activityData['durationTime']; | ||||
|                 $hours = floor($durationTimeInMinutes / 60); | ||||
|                 $minutes = $durationTimeInMinutes % 60; | ||||
| @@ -390,26 +395,36 @@ final class ActivityController extends AbstractController | ||||
|                 } | ||||
|             } | ||||
|  | ||||
|             if (array_key_exists('personsId', $activityData)) { | ||||
|             if (array_key_exists('personsId', $activityData) && $activityType->getPersonsVisible() > 0) { | ||||
|                 foreach ($activityData['personsId'] as $personId) { | ||||
|                     $concernedPerson = $this->personRepository->find($personId); | ||||
|                     $entity->addPerson($concernedPerson); | ||||
|                 } | ||||
|             } | ||||
|  | ||||
|             if (array_key_exists('professionalsId', $activityData)) { | ||||
|             if (array_key_exists('professionalsId', $activityData) && $activityType->getThirdPartiesVisible() > 0) { | ||||
|                 foreach ($activityData['professionalsId'] as $professionalsId) { | ||||
|                     $professional = $this->thirdPartyRepository->find($professionalsId); | ||||
|                     $entity->addThirdParty($professional); | ||||
|                 } | ||||
|             } | ||||
|  | ||||
|             if (array_key_exists('location', $activityData)) { | ||||
|             if (array_key_exists('usersId', $activityData) && $activityType->getUsersVisible() > 0) { | ||||
|                 foreach ($activityData['usersId'] as $userId) { | ||||
|                     $user = $this->userRepository->find($userId); | ||||
|  | ||||
|                     if (null !== $user) { | ||||
|                         $entity->addUser($user); | ||||
|                     } | ||||
|                 } | ||||
|             } | ||||
|  | ||||
|             if (array_key_exists('location', $activityData) && $activityType->getLocationVisible() > 0) { | ||||
|                 $location = $this->locationRepository->find($activityData['location']); | ||||
|                 $entity->setLocation($location); | ||||
|             } | ||||
|  | ||||
|             if (array_key_exists('comment', $activityData)) { | ||||
|             if (array_key_exists('comment', $activityData) && $activityType->getCommentVisible() > 0) { | ||||
|                 $comment = new CommentEmbeddable(); | ||||
|                 $comment->setComment($activityData['comment']); | ||||
|                 $comment->setUserId($this->getUser()->getid()); | ||||
|   | ||||
| @@ -516,6 +516,11 @@ class ActivityType | ||||
|         return $this->userVisible; | ||||
|     } | ||||
|  | ||||
|     public function hasCategory(): bool | ||||
|     { | ||||
|         return null !== $this->getCategory(); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Is active | ||||
|      * return true if the type is active. | ||||
|   | ||||
| @@ -33,26 +33,19 @@ class BySocialActionAggregator implements AggregatorInterface | ||||
|         $this->actionRepository = $actionRepository; | ||||
|     } | ||||
|  | ||||
|     public function addRole() | ||||
|     public function addRole(): ?string | ||||
|     { | ||||
|         return null; | ||||
|     } | ||||
|  | ||||
|     public function alterQuery(QueryBuilder $qb, $data) | ||||
|     { | ||||
|         if (!in_array('socialaction', $qb->getAllAliases(), true)) { | ||||
|             $qb->join('activity.socialActions', 'socialaction'); | ||||
|         if (!in_array('actsocialaction', $qb->getAllAliases(), true)) { | ||||
|             $qb->leftJoin('activity.socialActions', 'actsocialaction'); | ||||
|         } | ||||
|  | ||||
|         $qb->addSelect('socialaction.id AS socialaction_aggregator'); | ||||
|  | ||||
|         $groupBy = $qb->getDQLPart('groupBy'); | ||||
|  | ||||
|         if (!empty($groupBy)) { | ||||
|             $qb->addGroupBy('socialaction_aggregator'); | ||||
|         } else { | ||||
|             $qb->groupBy('socialaction_aggregator'); | ||||
|         } | ||||
|         $qb->addSelect('actsocialaction.id AS socialaction_aggregator'); | ||||
|         $qb->addGroupBy('socialaction_aggregator'); | ||||
|     } | ||||
|  | ||||
|     public function applyOn(): string | ||||
| @@ -72,6 +65,10 @@ class BySocialActionAggregator implements AggregatorInterface | ||||
|                 return 'Social action'; | ||||
|             } | ||||
|  | ||||
|             if (null === $value) { | ||||
|                 return ''; | ||||
|             } | ||||
|  | ||||
|             $sa = $this->actionRepository->find($value); | ||||
|  | ||||
|             return $this->actionRender->renderString($sa, []); | ||||
|   | ||||
| @@ -33,26 +33,19 @@ class BySocialIssueAggregator implements AggregatorInterface | ||||
|         $this->issueRender = $issueRender; | ||||
|     } | ||||
|  | ||||
|     public function addRole() | ||||
|     public function addRole(): ?string | ||||
|     { | ||||
|         return null; | ||||
|     } | ||||
|  | ||||
|     public function alterQuery(QueryBuilder $qb, $data) | ||||
|     { | ||||
|         if (!in_array('socialissue', $qb->getAllAliases(), true)) { | ||||
|             $qb->join('activity.socialIssues', 'socialissue'); | ||||
|         if (!in_array('actsocialissue', $qb->getAllAliases(), true)) { | ||||
|             $qb->leftJoin('activity.socialIssues', 'actsocialissue'); | ||||
|         } | ||||
|  | ||||
|         $qb->addSelect('socialissue.id AS socialissue_aggregator'); | ||||
|  | ||||
|         $groupBy = $qb->getDQLPart('groupBy'); | ||||
|  | ||||
|         if (!empty($groupBy)) { | ||||
|             $qb->addGroupBy('socialissue_aggregator'); | ||||
|         } else { | ||||
|             $qb->groupBy('socialissue_aggregator'); | ||||
|         } | ||||
|         $qb->addSelect('actsocialissue.id AS socialissue_aggregator'); | ||||
|         $qb->addGroupBy('socialissue_aggregator'); | ||||
|     } | ||||
|  | ||||
|     public function applyOn(): string | ||||
| @@ -72,6 +65,10 @@ class BySocialIssueAggregator implements AggregatorInterface | ||||
|                 return 'Social issues'; | ||||
|             } | ||||
|  | ||||
|             if (null === $value) { | ||||
|                 return ''; | ||||
|             } | ||||
|  | ||||
|             $i = $this->issueRepository->find($value); | ||||
|  | ||||
|             return $this->issueRender->renderString($i, []); | ||||
|   | ||||
| @@ -33,26 +33,19 @@ class ByThirdpartyAggregator implements AggregatorInterface | ||||
|         $this->thirdPartyRender = $thirdPartyRender; | ||||
|     } | ||||
|  | ||||
|     public function addRole() | ||||
|     public function addRole(): ?string | ||||
|     { | ||||
|         return null; | ||||
|     } | ||||
|  | ||||
|     public function alterQuery(QueryBuilder $qb, $data) | ||||
|     { | ||||
|         if (!in_array('thirdparty', $qb->getAllAliases(), true)) { | ||||
|             $qb->join('activity.thirdParties', 'thirdparty'); | ||||
|         if (!in_array('acttparty', $qb->getAllAliases(), true)) { | ||||
|             $qb->leftJoin('activity.thirdParties', 'acttparty'); | ||||
|         } | ||||
|  | ||||
|         $qb->addSelect('thirdparty.id AS thirdparty_aggregator'); | ||||
|  | ||||
|         $groupBy = $qb->getDQLPart('groupBy'); | ||||
|  | ||||
|         if (!empty($groupBy)) { | ||||
|             $qb->addGroupBy('thirdparty_aggregator'); | ||||
|         } else { | ||||
|             $qb->groupBy('thirdparty_aggregator'); | ||||
|         } | ||||
|         $qb->addSelect('acttparty.id AS thirdparty_aggregator'); | ||||
|         $qb->addGroupBy('thirdparty_aggregator'); | ||||
|     } | ||||
|  | ||||
|     public function applyOn(): string | ||||
| @@ -72,6 +65,10 @@ class ByThirdpartyAggregator implements AggregatorInterface | ||||
|                 return 'Accepted thirdparty'; | ||||
|             } | ||||
|  | ||||
|             if (null === $value) { | ||||
|                 return ''; | ||||
|             } | ||||
|  | ||||
|             $tp = $this->thirdPartyRepository->find($value); | ||||
|  | ||||
|             return $this->thirdPartyRender->renderString($tp, []); | ||||
|   | ||||
| @@ -33,26 +33,19 @@ class ByUserAggregator implements AggregatorInterface | ||||
|         $this->userRender = $userRender; | ||||
|     } | ||||
|  | ||||
|     public function addRole() | ||||
|     public function addRole(): ?string | ||||
|     { | ||||
|         return null; | ||||
|     } | ||||
|  | ||||
|     public function alterQuery(QueryBuilder $qb, $data) | ||||
|     { | ||||
|         if (!in_array('user', $qb->getAllAliases(), true)) { | ||||
|             $qb->join('activity.users', 'user'); | ||||
|         if (!in_array('actusers', $qb->getAllAliases(), true)) { | ||||
|             $qb->leftJoin('activity.users', 'actusers'); | ||||
|         } | ||||
|  | ||||
|         $qb->addSelect('user.id AS users_aggregator'); | ||||
|  | ||||
|         $groupBy = $qb->getDQLPart('groupBy'); | ||||
|  | ||||
|         if (!empty($groupBy)) { | ||||
|             $qb->addGroupBy('users_aggregator'); | ||||
|         } else { | ||||
|             $qb->groupBy('users_aggregator'); | ||||
|         } | ||||
|         $qb->addSelect('actusers.id AS users_aggregator'); | ||||
|         $qb->addGroupBy('users_aggregator'); | ||||
|     } | ||||
|  | ||||
|     public function applyOn(): string | ||||
| @@ -72,6 +65,10 @@ class ByUserAggregator implements AggregatorInterface | ||||
|                 return 'Accepted users'; | ||||
|             } | ||||
|  | ||||
|             if (null === $value) { | ||||
|                 return ''; | ||||
|             } | ||||
|  | ||||
|             $u = $this->userRepository->find($value); | ||||
|  | ||||
|             return $this->userRender->renderString($u, []); | ||||
|   | ||||
| @@ -13,7 +13,6 @@ namespace Chill\ActivityBundle\Export\Aggregator\ACPAggregators; | ||||
|  | ||||
| use Chill\ActivityBundle\Export\Declarations; | ||||
| use Chill\MainBundle\Export\AggregatorInterface; | ||||
| use DateTime; | ||||
| use Doctrine\ORM\QueryBuilder; | ||||
| use RuntimeException; | ||||
| use Symfony\Component\Form\Extension\Core\Type\ChoiceType; | ||||
| @@ -38,7 +37,7 @@ class DateAggregator implements AggregatorInterface | ||||
|         $this->translator = $translator; | ||||
|     } | ||||
|  | ||||
|     public function addRole() | ||||
|     public function addRole(): ?string | ||||
|     { | ||||
|         return null; | ||||
|     } | ||||
| @@ -49,41 +48,27 @@ class DateAggregator implements AggregatorInterface | ||||
|  | ||||
|         switch ($data['frequency']) { | ||||
|             case 'month': | ||||
|                 $fmt = 'MM'; | ||||
|                 $fmt = 'YYYY-MM'; | ||||
|  | ||||
| break; | ||||
|                 break; | ||||
|  | ||||
|             case 'week': | ||||
|                 $fmt = 'IW'; | ||||
|                 $fmt = 'YYYY-IW'; | ||||
|  | ||||
| break; | ||||
|                 break; | ||||
|  | ||||
|             case 'year': | ||||
|                 $fmt = 'YYYY'; $order = 'DESC'; | ||||
|  | ||||
| break; | ||||
|                 break; // order DESC does not works ! | ||||
|  | ||||
|             default: | ||||
|                 throw new RuntimeException(sprintf("The frequency data '%s' is invalid.", $data['frequency'])); | ||||
|         } | ||||
|  | ||||
|         $qb->addSelect(sprintf("TO_CHAR(activity.date, '%s') AS date_aggregator", $fmt)); | ||||
|  | ||||
|         $groupBy = $qb->getDQLPart('groupBy'); | ||||
|  | ||||
|         if (!empty($groupBy)) { | ||||
|             $qb->addGroupBy('date_aggregator'); | ||||
|         } else { | ||||
|             $qb->groupBy('date_aggregator'); | ||||
|         } | ||||
|  | ||||
|         $orderBy = $qb->getDQLPart('orderBy'); | ||||
|  | ||||
|         if (!empty($orderBy)) { | ||||
|             $qb->addOrderBy('date_aggregator', $order); | ||||
|         } else { | ||||
|             $qb->orderBy('date_aggregator', $order); | ||||
|         } | ||||
|         $qb->addGroupBy('date_aggregator'); | ||||
|         $qb->addOrderBy('date_aggregator', $order); | ||||
|     } | ||||
|  | ||||
|     public function applyOn(): string | ||||
| @@ -109,16 +94,12 @@ break; | ||||
|                 return 'by ' . $data['frequency']; | ||||
|             } | ||||
|  | ||||
|             if (null === $value) { | ||||
|                 return ''; | ||||
|             } | ||||
|  | ||||
|             switch ($data['frequency']) { | ||||
|                 case 'month': | ||||
|                     $month = DateTime::createFromFormat('!m', $value); | ||||
|  | ||||
|                     return sprintf( | ||||
|                         '%02d (%s)', | ||||
|                         $value, | ||||
|                         $month->format('M') | ||||
|                     ); | ||||
|  | ||||
|                 case 'week': | ||||
|                     //return $this->translator->trans('for week') .' '. $value ; | ||||
|  | ||||
|   | ||||
| @@ -33,26 +33,19 @@ class LocationTypeAggregator implements AggregatorInterface | ||||
|         $this->translatableStringHelper = $translatableStringHelper; | ||||
|     } | ||||
|  | ||||
|     public function addRole() | ||||
|     public function addRole(): ?string | ||||
|     { | ||||
|         return null; | ||||
|     } | ||||
|  | ||||
|     public function alterQuery(QueryBuilder $qb, $data) | ||||
|     { | ||||
|         if (!in_array('location', $qb->getAllAliases(), true)) { | ||||
|             $qb->join('activity.location', 'location'); | ||||
|         if (!in_array('actloc', $qb->getAllAliases(), true)) { | ||||
|             $qb->leftJoin('activity.location', 'actloc'); | ||||
|         } | ||||
|  | ||||
|         $qb->addSelect('IDENTITY(location.locationType) AS locationtype_aggregator'); | ||||
|  | ||||
|         $groupBy = $qb->getDQLPart('groupBy'); | ||||
|  | ||||
|         if (!empty($groupBy)) { | ||||
|             $qb->addGroupBy('locationtype_aggregator'); | ||||
|         } else { | ||||
|             $qb->groupBy('locationtype_aggregator'); | ||||
|         } | ||||
|         $qb->addSelect('IDENTITY(actloc.locationType) AS locationtype_aggregator'); | ||||
|         $qb->addGroupBy('locationtype_aggregator'); | ||||
|     } | ||||
|  | ||||
|     public function applyOn(): string | ||||
| @@ -72,6 +65,10 @@ class LocationTypeAggregator implements AggregatorInterface | ||||
|                 return 'Accepted locationtype'; | ||||
|             } | ||||
|  | ||||
|             if (null === $value) { | ||||
|                 return ''; | ||||
|             } | ||||
|  | ||||
|             $lt = $this->locationTypeRepository->find($value); | ||||
|  | ||||
|             return $this->translatableStringHelper->localize( | ||||
|   | ||||
| @@ -33,26 +33,19 @@ class UserScopeAggregator implements AggregatorInterface | ||||
|         $this->translatableStringHelper = $translatableStringHelper; | ||||
|     } | ||||
|  | ||||
|     public function addRole() | ||||
|     public function addRole(): ?string | ||||
|     { | ||||
|         return null; | ||||
|     } | ||||
|  | ||||
|     public function alterQuery(QueryBuilder $qb, $data) | ||||
|     { | ||||
|         if (!in_array('user', $qb->getAllAliases(), true)) { | ||||
|             $qb->join('activity.user', 'user'); | ||||
|         if (!in_array('actuser', $qb->getAllAliases(), true)) { | ||||
|             $qb->leftJoin('activity.user', 'actuser'); | ||||
|         } | ||||
|  | ||||
|         $qb->addSelect('IDENTITY(user.mainScope) AS userscope_aggregator'); | ||||
|  | ||||
|         $groupBy = $qb->getDQLPart('groupBy'); | ||||
|  | ||||
|         if (!empty($groupBy)) { | ||||
|             $qb->addGroupBy('userscope_aggregator'); | ||||
|         } else { | ||||
|             $qb->groupBy('userscope_aggregator'); | ||||
|         } | ||||
|         $qb->addSelect('IDENTITY(actuser.mainScope) AS userscope_aggregator'); | ||||
|         $qb->addGroupBy('userscope_aggregator'); | ||||
|     } | ||||
|  | ||||
|     public function applyOn(): string | ||||
| @@ -72,6 +65,10 @@ class UserScopeAggregator implements AggregatorInterface | ||||
|                 return 'Scope'; | ||||
|             } | ||||
|  | ||||
|             if (null === $value) { | ||||
|                 return ''; | ||||
|             } | ||||
|  | ||||
|             $s = $this->scopeRepository->find($value); | ||||
|  | ||||
|             return $this->translatableStringHelper->localize( | ||||
|   | ||||
| @@ -12,53 +12,43 @@ declare(strict_types=1); | ||||
| namespace Chill\ActivityBundle\Export\Aggregator; | ||||
|  | ||||
| use Chill\ActivityBundle\Export\Declarations; | ||||
| use Chill\ActivityBundle\Repository\ActivityTypeRepository; | ||||
| use Chill\ActivityBundle\Security\Authorization\ActivityStatsVoter; | ||||
| use Chill\ActivityBundle\Repository\ActivityTypeRepositoryInterface; | ||||
| use Chill\MainBundle\Export\AggregatorInterface; | ||||
| use Chill\MainBundle\Templating\TranslatableStringHelperInterface; | ||||
| use Closure; | ||||
| use Doctrine\ORM\Query\Expr\Join; | ||||
| use Doctrine\ORM\QueryBuilder; | ||||
| use Symfony\Component\Form\FormBuilderInterface; | ||||
| use Symfony\Component\Security\Core\Role\Role; | ||||
| use function in_array; | ||||
|  | ||||
| class ActivityTypeAggregator implements AggregatorInterface | ||||
| { | ||||
|     public const KEY = 'activity_type_aggregator'; | ||||
|  | ||||
|     protected ActivityTypeRepository $activityTypeRepository; | ||||
|     protected ActivityTypeRepositoryInterface $activityTypeRepository; | ||||
|  | ||||
|     protected TranslatableStringHelperInterface $translatableStringHelper; | ||||
|  | ||||
|     public function __construct( | ||||
|         ActivityTypeRepository $activityTypeRepository, | ||||
|         ActivityTypeRepositoryInterface $activityTypeRepository, | ||||
|         TranslatableStringHelperInterface $translatableStringHelper | ||||
|     ) { | ||||
|         $this->activityTypeRepository = $activityTypeRepository; | ||||
|         $this->translatableStringHelper = $translatableStringHelper; | ||||
|     } | ||||
|  | ||||
|     public function addRole() | ||||
|     public function addRole(): ?string | ||||
|     { | ||||
|         return new Role(ActivityStatsVoter::STATS); | ||||
|         return null; | ||||
|     } | ||||
|  | ||||
|     public function alterQuery(QueryBuilder $qb, $data) | ||||
|     { | ||||
|         if (!in_array('type', $qb->getAllAliases(), true)) { | ||||
|             $qb->join('activity.activityType', 'type'); | ||||
|         if (!in_array('acttype', $qb->getAllAliases(), true)) { | ||||
|             $qb->join('activity.activityType', 'acttype'); | ||||
|         } | ||||
|  | ||||
|         $qb->addSelect(sprintf('IDENTITY(activity.activityType) AS %s', self::KEY)); | ||||
|  | ||||
|         $groupby = $qb->getDQLPart('groupBy'); | ||||
|  | ||||
|         if (!empty($groupBy)) { | ||||
|             $qb->addGroupBy(self::KEY); | ||||
|         } else { | ||||
|             $qb->groupBy(self::KEY); | ||||
|         } | ||||
|         $qb->addGroupBy(self::KEY); | ||||
|     } | ||||
|  | ||||
|     public function applyOn(): string | ||||
| @@ -81,6 +71,10 @@ class ActivityTypeAggregator implements AggregatorInterface | ||||
|                 return 'Activity type'; | ||||
|             } | ||||
|  | ||||
|             if (null === $value) { | ||||
|                 return ''; | ||||
|             } | ||||
|  | ||||
|             $t = $this->activityTypeRepository->find($value); | ||||
|  | ||||
|             return $this->translatableStringHelper->localize($t->getName()); | ||||
| @@ -96,23 +90,4 @@ class ActivityTypeAggregator implements AggregatorInterface | ||||
|     { | ||||
|         return 'Aggregate by activity type'; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Check if a join between Activity and another alias. | ||||
|      * | ||||
|      * @param Join[] $joins | ||||
|      * @param string $alias the alias to search for | ||||
|      * | ||||
|      * @return bool | ||||
|      */ | ||||
|     private function checkJoinAlreadyDefined(array $joins, $alias) | ||||
|     { | ||||
|         foreach ($joins as $join) { | ||||
|             if ($join->getAlias() === $alias) { | ||||
|                 return true; | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         return false; | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -12,14 +12,12 @@ declare(strict_types=1); | ||||
| namespace Chill\ActivityBundle\Export\Aggregator; | ||||
|  | ||||
| use Chill\ActivityBundle\Export\Declarations; | ||||
| use Chill\ActivityBundle\Security\Authorization\ActivityStatsVoter; | ||||
| use Chill\MainBundle\Export\AggregatorInterface; | ||||
| use Chill\MainBundle\Repository\UserRepository; | ||||
| use Chill\MainBundle\Templating\Entity\UserRender; | ||||
| use Closure; | ||||
| use Doctrine\ORM\QueryBuilder; | ||||
| use Symfony\Component\Form\FormBuilderInterface; | ||||
| use Symfony\Component\Security\Core\Role\Role; | ||||
|  | ||||
| class ActivityUserAggregator implements AggregatorInterface | ||||
| { | ||||
| @@ -37,9 +35,9 @@ class ActivityUserAggregator implements AggregatorInterface | ||||
|         $this->userRender = $userRender; | ||||
|     } | ||||
|  | ||||
|     public function addRole() | ||||
|     public function addRole(): ?string | ||||
|     { | ||||
|         return new Role(ActivityStatsVoter::STATS); | ||||
|         return null; | ||||
|     } | ||||
|  | ||||
|     public function alterQuery(QueryBuilder $qb, $data) | ||||
| @@ -63,14 +61,15 @@ class ActivityUserAggregator implements AggregatorInterface | ||||
|  | ||||
|     public function getLabels($key, $values, $data): Closure | ||||
|     { | ||||
|         // preload users at once | ||||
|         $this->userRepository->findBy(['id' => $values]); | ||||
|  | ||||
|         return function ($value) { | ||||
|             if ('_header' === $value) { | ||||
|                 return 'Activity user'; | ||||
|             } | ||||
|  | ||||
|             if (null === $value) { | ||||
|                 return ''; | ||||
|             } | ||||
|  | ||||
|             $u = $this->userRepository->find($value); | ||||
|  | ||||
|             return $this->userRender->renderString($u, []); | ||||
|   | ||||
| @@ -14,7 +14,6 @@ namespace Chill\ActivityBundle\Export\Aggregator\PersonAggregators; | ||||
| use Chill\ActivityBundle\Export\Declarations; | ||||
| use Chill\ActivityBundle\Repository\ActivityReasonCategoryRepository; | ||||
| use Chill\ActivityBundle\Repository\ActivityReasonRepository; | ||||
| use Chill\ActivityBundle\Security\Authorization\ActivityStatsVoter; | ||||
| use Chill\MainBundle\Export\AggregatorInterface; | ||||
| use Chill\MainBundle\Export\ExportElementValidatedInterface; | ||||
| use Chill\MainBundle\Templating\TranslatableStringHelper; | ||||
| @@ -24,11 +23,10 @@ use Doctrine\ORM\QueryBuilder; | ||||
| use RuntimeException; | ||||
| use Symfony\Component\Form\Extension\Core\Type\ChoiceType; | ||||
| use Symfony\Component\Form\FormBuilderInterface; | ||||
| use Symfony\Component\Security\Core\Role\Role; | ||||
| use Symfony\Component\Validator\Context\ExecutionContextInterface; | ||||
|  | ||||
| use function array_key_exists; | ||||
| use function count; | ||||
| use function in_array; | ||||
|  | ||||
| class ActivityReasonAggregator implements AggregatorInterface, ExportElementValidatedInterface | ||||
| { | ||||
| @@ -48,19 +46,19 @@ class ActivityReasonAggregator implements AggregatorInterface, ExportElementVali | ||||
|         $this->translatableStringHelper = $translatableStringHelper; | ||||
|     } | ||||
|  | ||||
|     public function addRole() | ||||
|     public function addRole(): ?string | ||||
|     { | ||||
|         return new Role(ActivityStatsVoter::STATS); | ||||
|         return null; | ||||
|     } | ||||
|  | ||||
|     public function alterQuery(QueryBuilder $qb, $data) | ||||
|     { | ||||
|         // add select element | ||||
|         if ('reasons' === $data['level']) { | ||||
|             $elem = 'reasons.id'; | ||||
|             $elem = 'actreasons.id'; | ||||
|             $alias = 'activity_reasons_id'; | ||||
|         } elseif ('categories' === $data['level']) { | ||||
|             $elem = 'category.id'; | ||||
|             $elem = 'actreasoncat.id'; | ||||
|             $alias = 'activity_categories_id'; | ||||
|         } else { | ||||
|             throw new RuntimeException('The data provided are not recognized.'); | ||||
| @@ -69,29 +67,15 @@ class ActivityReasonAggregator implements AggregatorInterface, ExportElementVali | ||||
|         $qb->addSelect($elem . ' as ' . $alias); | ||||
|  | ||||
|         // make a jointure only if needed | ||||
|         $join = $qb->getDQLPart('join'); | ||||
|  | ||||
|         if ( | ||||
|             ( | ||||
|                 array_key_exists('activity', $join) | ||||
|             && !$this->checkJoinAlreadyDefined($join['activity'], 'reasons') | ||||
|             ) | ||||
|             || (!array_key_exists('activity', $join)) | ||||
|         ) { | ||||
|             $qb->add( | ||||
|                 'join', | ||||
|                 [ | ||||
|                     'activity' => new Join(Join::INNER_JOIN, 'activity.reasons', 'reasons'), | ||||
|                 ], | ||||
|                 true | ||||
|             ); | ||||
|         if (!in_array('actreasons', $qb->getAllAliases(), true)) { | ||||
|             $qb->innerJoin('activity.reasons', 'actreasons'); | ||||
|         } | ||||
|  | ||||
|         // join category if necessary | ||||
|         if ('activity_categories_id' === $alias) { | ||||
|             // add join only if needed | ||||
|             if (!$this->checkJoinAlreadyDefined($qb->getDQLPart('join')['activity'], 'category')) { | ||||
|                 $qb->join('reasons.category', 'category'); | ||||
|             if (!in_array('actreasoncat', $qb->getAllAliases(), true)) { | ||||
|                 $qb->join('actreasons.category', 'actreasoncat'); | ||||
|             } | ||||
|         } | ||||
|  | ||||
| @@ -195,23 +179,4 @@ class ActivityReasonAggregator implements AggregatorInterface, ExportElementVali | ||||
|                 ->addViolation(); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Check if a join between Activity and another alias. | ||||
|      * | ||||
|      * @param Join[] $joins | ||||
|      * @param string $alias the alias to search for | ||||
|      * | ||||
|      * @return bool | ||||
|      */ | ||||
|     private function checkJoinAlreadyDefined(array $joins, $alias) | ||||
|     { | ||||
|         foreach ($joins as $join) { | ||||
|             if ($join->getAlias() === $alias) { | ||||
|                 return true; | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         return false; | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -17,11 +17,12 @@ use Chill\ActivityBundle\Security\Authorization\ActivityStatsVoter; | ||||
| use Chill\MainBundle\Export\ExportInterface; | ||||
| use Chill\MainBundle\Export\FormatterInterface; | ||||
| use Chill\MainBundle\Export\GroupedExportInterface; | ||||
| use Chill\PersonBundle\Export\Declarations as PersonDeclarations; | ||||
| use Doctrine\ORM\EntityManagerInterface; | ||||
| use Doctrine\ORM\EntityRepository; | ||||
| use Doctrine\ORM\Query; | ||||
| use LogicException; | ||||
| use Symfony\Component\Form\FormBuilderInterface; | ||||
| use Symfony\Component\Security\Core\Role\Role; | ||||
|  | ||||
| class AvgActivityDuration implements ExportInterface, GroupedExportInterface | ||||
| { | ||||
| @@ -35,7 +36,6 @@ class AvgActivityDuration implements ExportInterface, GroupedExportInterface | ||||
|  | ||||
|     public function buildForm(FormBuilderInterface $builder) | ||||
|     { | ||||
|         // TODO: Implement buildForm() method. | ||||
|     } | ||||
|  | ||||
|     public function getAllowedFormattersTypes(): array | ||||
| @@ -84,17 +84,19 @@ class AvgActivityDuration implements ExportInterface, GroupedExportInterface | ||||
|  | ||||
|     public function initiateQuery(array $requiredModifiers, array $acl, array $data = []) | ||||
|     { | ||||
|         $qb = $this->repository->createQueryBuilder('activity') | ||||
|             ->join('activity.accompanyingPeriod', 'acp'); | ||||
|         $qb = $this->repository->createQueryBuilder('activity'); | ||||
|  | ||||
|         $qb->select('AVG(activity.durationTime) as export_avg_activity_duration'); | ||||
|         $qb | ||||
|             ->join('activity.accompanyingPeriod', 'acp') | ||||
|             ->select('AVG(activity.durationTime) as export_avg_activity_duration') | ||||
|             ->andWhere($qb->expr()->isNotNull('activity.durationTime')); | ||||
|  | ||||
|         return $qb; | ||||
|     } | ||||
|  | ||||
|     public function requiredRole(): Role | ||||
|     public function requiredRole(): string | ||||
|     { | ||||
|         return new Role(ActivityStatsVoter::STATS); | ||||
|         return ActivityStatsVoter::STATS; | ||||
|     } | ||||
|  | ||||
|     public function supportsModifiers(): array | ||||
| @@ -102,7 +104,7 @@ class AvgActivityDuration implements ExportInterface, GroupedExportInterface | ||||
|         return [ | ||||
|             Declarations::ACTIVITY, | ||||
|             Declarations::ACTIVITY_ACP, | ||||
|             //PersonDeclarations::ACP_TYPE, | ||||
|             PersonDeclarations::ACP_TYPE, | ||||
|         ]; | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -17,11 +17,12 @@ use Chill\ActivityBundle\Security\Authorization\ActivityStatsVoter; | ||||
| use Chill\MainBundle\Export\ExportInterface; | ||||
| use Chill\MainBundle\Export\FormatterInterface; | ||||
| use Chill\MainBundle\Export\GroupedExportInterface; | ||||
| use Chill\PersonBundle\Export\Declarations as PersonDeclarations; | ||||
| use Doctrine\ORM\EntityManagerInterface; | ||||
| use Doctrine\ORM\EntityRepository; | ||||
| use Doctrine\ORM\Query; | ||||
| use LogicException; | ||||
| use Symfony\Component\Form\FormBuilderInterface; | ||||
| use Symfony\Component\Security\Core\Role\Role; | ||||
|  | ||||
| class AvgActivityVisitDuration implements ExportInterface, GroupedExportInterface | ||||
| { | ||||
| @@ -84,17 +85,19 @@ class AvgActivityVisitDuration implements ExportInterface, GroupedExportInterfac | ||||
|  | ||||
|     public function initiateQuery(array $requiredModifiers, array $acl, array $data = []) | ||||
|     { | ||||
|         $qb = $this->repository->createQueryBuilder('activity') | ||||
|             ->join('activity.accompanyingPeriod', 'acp'); | ||||
|         $qb = $this->repository->createQueryBuilder('activity'); | ||||
|  | ||||
|         $qb->select('AVG(activity.travelTime) as export_avg_activity_visit_duration'); | ||||
|         $qb | ||||
|             ->join('activity.accompanyingPeriod', 'acp') | ||||
|             ->select('AVG(activity.travelTime) as export_avg_activity_visit_duration') | ||||
|             ->andWhere($qb->expr()->isNotNull('activity.travelTime')); | ||||
|  | ||||
|         return $qb; | ||||
|     } | ||||
|  | ||||
|     public function requiredRole(): Role | ||||
|     public function requiredRole(): string | ||||
|     { | ||||
|         return new Role(ActivityStatsVoter::STATS); | ||||
|         return ActivityStatsVoter::STATS; | ||||
|     } | ||||
|  | ||||
|     public function supportsModifiers(): array | ||||
| @@ -102,7 +105,7 @@ class AvgActivityVisitDuration implements ExportInterface, GroupedExportInterfac | ||||
|         return [ | ||||
|             Declarations::ACTIVITY, | ||||
|             Declarations::ACTIVITY_ACP, | ||||
|             //PersonDeclarations::ACP_TYPE, | ||||
|             PersonDeclarations::ACP_TYPE, | ||||
|         ]; | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -23,7 +23,7 @@ use Doctrine\ORM\EntityRepository; | ||||
| use Doctrine\ORM\Query; | ||||
| use LogicException; | ||||
| use Symfony\Component\Form\FormBuilderInterface; | ||||
| use Symfony\Component\Security\Core\Role\Role; | ||||
| use function in_array; | ||||
|  | ||||
| class CountActivity implements ExportInterface, GroupedExportInterface | ||||
| { | ||||
| @@ -85,17 +85,20 @@ class CountActivity implements ExportInterface, GroupedExportInterface | ||||
|  | ||||
|     public function initiateQuery(array $requiredModifiers, array $acl, array $data = []) | ||||
|     { | ||||
|         $qb = $this->repository->createQueryBuilder('activity') | ||||
|             ->join('activity.accompanyingPeriod', 'acp'); | ||||
|         $qb = $this->repository->createQueryBuilder('activity'); | ||||
|  | ||||
|         $qb->select('COUNT(activity.id) as export_count_activity'); | ||||
|         if (!in_array('acp', $qb->getAllAliases(), true)) { | ||||
|             $qb->join('activity.accompanyingPeriod', 'acp'); | ||||
|         } | ||||
|  | ||||
|         $qb->select('COUNT(DISTINCT activity.id) as export_count_activity'); | ||||
|  | ||||
|         return $qb; | ||||
|     } | ||||
|  | ||||
|     public function requiredRole(): Role | ||||
|     public function requiredRole(): string | ||||
|     { | ||||
|         return new Role(ActivityStatsVoter::STATS); | ||||
|         return ActivityStatsVoter::STATS; | ||||
|     } | ||||
|  | ||||
|     public function supportsModifiers(): array | ||||
| @@ -103,7 +106,7 @@ class CountActivity implements ExportInterface, GroupedExportInterface | ||||
|         return [ | ||||
|             Declarations::ACTIVITY, | ||||
|             Declarations::ACTIVITY_ACP, | ||||
|             //PersonDeclarations::ACP_TYPE, | ||||
|             PersonDeclarations::ACP_TYPE, | ||||
|         ]; | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -17,11 +17,13 @@ use Chill\ActivityBundle\Security\Authorization\ActivityStatsVoter; | ||||
| use Chill\MainBundle\Export\ExportInterface; | ||||
| use Chill\MainBundle\Export\FormatterInterface; | ||||
| use Chill\MainBundle\Export\GroupedExportInterface; | ||||
| use Chill\PersonBundle\Export\Declarations as PersonDeclarations; | ||||
| use Doctrine\ORM\EntityManagerInterface; | ||||
| use Doctrine\ORM\EntityRepository; | ||||
| use Doctrine\ORM\Query; | ||||
| use LogicException; | ||||
| use Symfony\Component\Form\FormBuilderInterface; | ||||
| use Symfony\Component\Security\Core\Role\Role; | ||||
| use function in_array; | ||||
|  | ||||
| class SumActivityDuration implements ExportInterface, GroupedExportInterface | ||||
| { | ||||
| @@ -84,17 +86,21 @@ class SumActivityDuration implements ExportInterface, GroupedExportInterface | ||||
|  | ||||
|     public function initiateQuery(array $requiredModifiers, array $acl, array $data = []) | ||||
|     { | ||||
|         $qb = $this->repository->createQueryBuilder('activity') | ||||
|             ->join('activity.accompanyingPeriod', 'acp'); | ||||
|         $qb = $this->repository->createQueryBuilder('activity'); | ||||
|  | ||||
|         $qb->select('SUM(activity.durationTime) as export_sum_activity_duration'); | ||||
|         if (!in_array('acp', $qb->getAllAliases(), true)) { | ||||
|             $qb->join('activity.accompanyingPeriod', 'acp'); | ||||
|         } | ||||
|  | ||||
|         $qb->select('SUM(activity.durationTime) as export_sum_activity_duration') | ||||
|             ->andWhere($qb->expr()->isNotNull('activity.durationTime')); | ||||
|  | ||||
|         return $qb; | ||||
|     } | ||||
|  | ||||
|     public function requiredRole(): Role | ||||
|     public function requiredRole(): string | ||||
|     { | ||||
|         return new Role(ActivityStatsVoter::STATS); | ||||
|         return ActivityStatsVoter::STATS; | ||||
|     } | ||||
|  | ||||
|     public function supportsModifiers(): array | ||||
| @@ -102,7 +108,7 @@ class SumActivityDuration implements ExportInterface, GroupedExportInterface | ||||
|         return [ | ||||
|             Declarations::ACTIVITY, | ||||
|             Declarations::ACTIVITY_ACP, | ||||
|             //PersonDeclarations::ACP_TYPE, | ||||
|             PersonDeclarations::ACP_TYPE, | ||||
|         ]; | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -17,11 +17,13 @@ use Chill\ActivityBundle\Security\Authorization\ActivityStatsVoter; | ||||
| use Chill\MainBundle\Export\ExportInterface; | ||||
| use Chill\MainBundle\Export\FormatterInterface; | ||||
| use Chill\MainBundle\Export\GroupedExportInterface; | ||||
| use Chill\PersonBundle\Export\Declarations as PersonDeclarations; | ||||
| use Doctrine\ORM\EntityManagerInterface; | ||||
| use Doctrine\ORM\EntityRepository; | ||||
| use Doctrine\ORM\Query; | ||||
| use LogicException; | ||||
| use Symfony\Component\Form\FormBuilderInterface; | ||||
| use Symfony\Component\Security\Core\Role\Role; | ||||
| use function in_array; | ||||
|  | ||||
| class SumActivityVisitDuration implements ExportInterface, GroupedExportInterface | ||||
| { | ||||
| @@ -84,17 +86,21 @@ class SumActivityVisitDuration implements ExportInterface, GroupedExportInterfac | ||||
|  | ||||
|     public function initiateQuery(array $requiredModifiers, array $acl, array $data = []) | ||||
|     { | ||||
|         $qb = $this->repository->createQueryBuilder('activity') | ||||
|             ->join('activity.accompanyingPeriod', 'acp'); | ||||
|         $qb = $this->repository->createQueryBuilder('activity'); | ||||
|  | ||||
|         $qb->select('SUM(activity.travelTime) as export_sum_activity_visit_duration'); | ||||
|         if (!in_array('acp', $qb->getAllAliases(), true)) { | ||||
|             $qb->join('activity.accompanyingPeriod', 'acp'); | ||||
|         } | ||||
|  | ||||
|         $qb->select('SUM(activity.travelTime) as export_sum_activity_visit_duration') | ||||
|             ->andWhere($qb->expr()->isNotNull('activity.travelTime')); | ||||
|  | ||||
|         return $qb; | ||||
|     } | ||||
|  | ||||
|     public function requiredRole(): Role | ||||
|     public function requiredRole(): string | ||||
|     { | ||||
|         return new Role(ActivityStatsVoter::STATS); | ||||
|         return ActivityStatsVoter::STATS; | ||||
|     } | ||||
|  | ||||
|     public function supportsModifiers(): array | ||||
| @@ -102,7 +108,7 @@ class SumActivityVisitDuration implements ExportInterface, GroupedExportInterfac | ||||
|         return [ | ||||
|             Declarations::ACTIVITY, | ||||
|             Declarations::ACTIVITY_ACP, | ||||
|             //PersonDeclarations::ACP_TYPE, | ||||
|             PersonDeclarations::ACP_TYPE, | ||||
|         ]; | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -21,7 +21,7 @@ use Chill\PersonBundle\Export\Declarations as PersonDeclarations; | ||||
| use Doctrine\ORM\Query; | ||||
| use LogicException; | ||||
| use Symfony\Component\Form\FormBuilderInterface; | ||||
| use Symfony\Component\Security\Core\Role\Role; | ||||
| use function in_array; | ||||
|  | ||||
| class CountActivity implements ExportInterface, GroupedExportInterface | ||||
| { | ||||
| @@ -85,8 +85,11 @@ class CountActivity implements ExportInterface, GroupedExportInterface | ||||
|     { | ||||
|         $centers = array_map(static fn ($el) => $el['center'], $acl); | ||||
|  | ||||
|         $qb = $this->activityRepository->createQueryBuilder('activity') | ||||
|             ->join('activity.person', 'person'); | ||||
|         $qb = $this->activityRepository->createQueryBuilder('activity'); | ||||
|  | ||||
|         if (!in_array('person', $qb->getAllAliases(), true)) { | ||||
|             $qb->join('activity.person', 'person'); | ||||
|         } | ||||
|  | ||||
|         $qb->select('COUNT(activity.id) as export_count_activity'); | ||||
|  | ||||
| @@ -97,9 +100,9 @@ class CountActivity implements ExportInterface, GroupedExportInterface | ||||
|         return $qb; | ||||
|     } | ||||
|  | ||||
|     public function requiredRole() | ||||
|     public function requiredRole(): string | ||||
|     { | ||||
|         return new Role(ActivityStatsVoter::STATS); | ||||
|         return ActivityStatsVoter::STATS; | ||||
|     } | ||||
|  | ||||
|     public function supportsModifiers() | ||||
| @@ -107,7 +110,7 @@ class CountActivity implements ExportInterface, GroupedExportInterface | ||||
|         return [ | ||||
|             Declarations::ACTIVITY, | ||||
|             Declarations::ACTIVITY_PERSON, | ||||
|             //PersonDeclarations::PERSON_TYPE, | ||||
|             PersonDeclarations::PERSON_TYPE, | ||||
|         ]; | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -26,7 +26,6 @@ use Doctrine\ORM\EntityManagerInterface; | ||||
| use Doctrine\ORM\Query; | ||||
| use Symfony\Component\Form\Extension\Core\Type\ChoiceType; | ||||
| use Symfony\Component\Form\FormBuilderInterface; | ||||
| use Symfony\Component\Security\Core\Role\Role; | ||||
| use Symfony\Component\Validator\Constraints\Callback; | ||||
| use Symfony\Component\Validator\Context\ExecutionContextInterface; | ||||
| use Symfony\Contracts\Translation\TranslatorInterface; | ||||
| @@ -212,8 +211,8 @@ class ListActivity implements ListInterface, GroupedExportInterface | ||||
|         $qb | ||||
|             ->from('ChillActivityBundle:Activity', 'activity') | ||||
|             ->join('activity.person', 'person') | ||||
|             ->join('person.center', 'center') | ||||
|             ->andWhere('center IN (:authorized_centers)') | ||||
|             ->join('actperson.center', 'actcenter') | ||||
|             ->andWhere('actcenter IN (:authorized_centers)') | ||||
|             ->setParameter('authorized_centers', $centers); | ||||
|  | ||||
|         foreach ($this->fields as $f) { | ||||
| @@ -240,8 +239,8 @@ class ListActivity implements ListInterface, GroupedExportInterface | ||||
|                         break; | ||||
|  | ||||
|                     case 'user_username': | ||||
|                         $qb->join('activity.user', 'user'); | ||||
|                         $qb->addSelect('user.username AS user_username'); | ||||
|                         $qb->join('activity.user', 'actuser'); | ||||
|                         $qb->addSelect('actuser.username AS user_username'); | ||||
|  | ||||
|                         break; | ||||
|  | ||||
| @@ -275,9 +274,9 @@ class ListActivity implements ListInterface, GroupedExportInterface | ||||
|         return $qb; | ||||
|     } | ||||
|  | ||||
|     public function requiredRole() | ||||
|     public function requiredRole(): string | ||||
|     { | ||||
|         return new Role(ActivityStatsVoter::LISTS); | ||||
|         return ActivityStatsVoter::LISTS; | ||||
|     } | ||||
|  | ||||
|     public function supportsModifiers() | ||||
|   | ||||
| @@ -22,7 +22,6 @@ use Chill\PersonBundle\Export\Declarations as PersonDeclarations; | ||||
| use Doctrine\ORM\Query; | ||||
| use LogicException; | ||||
| use Symfony\Component\Form\FormBuilderInterface; | ||||
| use Symfony\Component\Security\Core\Role\Role; | ||||
|  | ||||
| /** | ||||
|  * This export allow to compute stats on activity duration. | ||||
| @@ -122,14 +121,14 @@ class StatActivityDuration implements ExportInterface, GroupedExportInterface | ||||
|  | ||||
|         return $qb->select($select) | ||||
|             ->join('activity.person', 'person') | ||||
|             ->join('person.center', 'center') | ||||
|             ->where($qb->expr()->in('center', ':centers')) | ||||
|             ->join('actperson.center', 'actcenter') | ||||
|             ->where($qb->expr()->in('actcenter', ':centers')) | ||||
|             ->setParameter(':centers', $centers); | ||||
|     } | ||||
|  | ||||
|     public function requiredRole() | ||||
|     public function requiredRole(): string | ||||
|     { | ||||
|         return new Role(ActivityStatsVoter::STATS); | ||||
|         return ActivityStatsVoter::STATS; | ||||
|     } | ||||
|  | ||||
|     public function supportsModifiers() | ||||
|   | ||||
| @@ -0,0 +1,102 @@ | ||||
| <?php | ||||
|  | ||||
| /** | ||||
|  * Chill is a software for social workers. | ||||
|  * | ||||
|  * For the full copyright and license information, please view | ||||
|  * the LICENSE file that was distributed with this source code. | ||||
|  */ | ||||
|  | ||||
| declare(strict_types=1); | ||||
|  | ||||
| /* | ||||
|  * Chill is a software for social workers | ||||
|  * | ||||
|  * For the full copyright and license information, please view | ||||
|  * the LICENSE file that was distributed with this source code. | ||||
|  */ | ||||
|  | ||||
| namespace Chill\ActivityBundle\Export\Filter\ACPFilters; | ||||
|  | ||||
| use Chill\ActivityBundle\Entity\Activity; | ||||
| use Chill\ActivityBundle\Entity\ActivityType; | ||||
| use Chill\ActivityBundle\Repository\ActivityTypeRepositoryInterface; | ||||
| use Chill\MainBundle\Export\FilterInterface; | ||||
| use Chill\MainBundle\Templating\TranslatableStringHelperInterface; | ||||
| use Chill\PersonBundle\Export\Declarations; | ||||
| use Doctrine\ORM\Query\Expr; | ||||
| use Doctrine\ORM\QueryBuilder; | ||||
| use Symfony\Bridge\Doctrine\Form\Type\EntityType; | ||||
| use Symfony\Component\Form\FormBuilderInterface; | ||||
| use function in_array; | ||||
|  | ||||
| class ActivityTypeFilter implements FilterInterface | ||||
| { | ||||
|     private ActivityTypeRepositoryInterface $activityTypeRepository; | ||||
|  | ||||
|     private TranslatableStringHelperInterface $translatableStringHelper; | ||||
|  | ||||
|     public function __construct( | ||||
|         ActivityTypeRepositoryInterface $activityTypeRepository, | ||||
|         TranslatableStringHelperInterface $translatableStringHelper | ||||
|     ) { | ||||
|         $this->activityTypeRepository = $activityTypeRepository; | ||||
|         $this->translatableStringHelper = $translatableStringHelper; | ||||
|     } | ||||
|  | ||||
|     public function addRole(): ?string | ||||
|     { | ||||
|         return null; | ||||
|     } | ||||
|  | ||||
|     public function alterQuery(QueryBuilder $qb, $data) | ||||
|     { | ||||
|         if (!in_array('activity', $qb->getAllAliases(), true)) { | ||||
|             $qb->join(Activity::class, 'activity', Expr\Join::WITH, 'activity.accompanyingPeriod = acp'); | ||||
|         } | ||||
|  | ||||
|         $clause = $qb->expr()->in('activity.activityType', ':selected_activity_types'); | ||||
|  | ||||
|         $qb->andWhere($clause); | ||||
|         $qb->setParameter('selected_activity_types', $data['types']); | ||||
|     } | ||||
|  | ||||
|     public function applyOn() | ||||
|     { | ||||
|         return Declarations::ACP_TYPE; | ||||
|     } | ||||
|  | ||||
|     public function buildForm(FormBuilderInterface $builder) | ||||
|     { | ||||
|         $builder->add('accepted_activitytypes', EntityType::class, [ | ||||
|             'class' => ActivityType::class, | ||||
|             'choices' => $this->activityTypeRepository->findAllActive(), | ||||
|             'choice_label' => function (ActivityType $aty) { | ||||
|                 return | ||||
|                     ($aty->hasCategory() ? $this->translatableStringHelper->localize($aty->getCategory()->getName()) . ' > ' : '') | ||||
|                     . | ||||
|                     $this->translatableStringHelper->localize($aty->getName()); | ||||
|             }, | ||||
|             'multiple' => true, | ||||
|             'expanded' => true, | ||||
|         ]); | ||||
|     } | ||||
|  | ||||
|     public function describeAction($data, $format = 'string'): array | ||||
|     { | ||||
|         $types = []; | ||||
|  | ||||
|         foreach ($data['accepted_activitytypes'] as $aty) { | ||||
|             $types[] = $this->translatableStringHelper->localize($aty->getName()); | ||||
|         } | ||||
|  | ||||
|         return ['Filtered by activity types: only %activitytypes%', [ | ||||
|             '%activitytypes%' => implode(', ', $types), | ||||
|         ]]; | ||||
|     } | ||||
|  | ||||
|     public function getTitle(): string | ||||
|     { | ||||
|         return 'Filter accompanying course by activity type'; | ||||
|     } | ||||
| } | ||||
| @@ -30,7 +30,7 @@ class BySocialActionFilter implements FilterInterface | ||||
|         $this->actionRender = $actionRender; | ||||
|     } | ||||
|  | ||||
|     public function addRole() | ||||
|     public function addRole(): ?string | ||||
|     { | ||||
|         return null; | ||||
|     } | ||||
| @@ -39,11 +39,11 @@ class BySocialActionFilter implements FilterInterface | ||||
|     { | ||||
|         $where = $qb->getDQLPart('where'); | ||||
|  | ||||
|         if (!in_array('socialaction', $qb->getAllAliases(), true)) { | ||||
|             $qb->join('activity.socialActions', 'socialaction'); | ||||
|         if (!in_array('actsocialaction', $qb->getAllAliases(), true)) { | ||||
|             $qb->join('activity.socialActions', 'actsocialaction'); | ||||
|         } | ||||
|  | ||||
|         $clause = $qb->expr()->in('socialaction.id', ':socialactions'); | ||||
|         $clause = $qb->expr()->in('actsocialaction.id', ':socialactions'); | ||||
|  | ||||
|         if ($where instanceof Andx) { | ||||
|             $where->add($clause); | ||||
|   | ||||
| @@ -30,7 +30,7 @@ class BySocialIssueFilter implements FilterInterface | ||||
|         $this->issueRender = $issueRender; | ||||
|     } | ||||
|  | ||||
|     public function addRole() | ||||
|     public function addRole(): ?string | ||||
|     { | ||||
|         return null; | ||||
|     } | ||||
| @@ -39,11 +39,11 @@ class BySocialIssueFilter implements FilterInterface | ||||
|     { | ||||
|         $where = $qb->getDQLPart('where'); | ||||
|  | ||||
|         if (!in_array('socialissue', $qb->getAllAliases(), true)) { | ||||
|             $qb->join('activity.socialIssues', 'socialissue'); | ||||
|         if (!in_array('actsocialissue', $qb->getAllAliases(), true)) { | ||||
|             $qb->join('activity.socialIssues', 'actsocialissue'); | ||||
|         } | ||||
|  | ||||
|         $clause = $qb->expr()->in('socialissue.id', ':socialissues'); | ||||
|         $clause = $qb->expr()->in('actsocialissue.id', ':socialissues'); | ||||
|  | ||||
|         if ($where instanceof Andx) { | ||||
|             $where->add($clause); | ||||
|   | ||||
| @@ -30,7 +30,7 @@ class ByUserFilter implements FilterInterface | ||||
|         $this->userRender = $userRender; | ||||
|     } | ||||
|  | ||||
|     public function addRole() | ||||
|     public function addRole(): ?string | ||||
|     { | ||||
|         return null; | ||||
|     } | ||||
| @@ -39,11 +39,11 @@ class ByUserFilter implements FilterInterface | ||||
|     { | ||||
|         $where = $qb->getDQLPart('where'); | ||||
|  | ||||
|         if (!in_array('user', $qb->getAllAliases(), true)) { | ||||
|             $qb->join('activity.users', 'user'); | ||||
|         if (!in_array('actusers', $qb->getAllAliases(), true)) { | ||||
|             $qb->join('activity.users', 'actusers'); | ||||
|         } | ||||
|  | ||||
|         $clause = $qb->expr()->in('user.id', ':users'); | ||||
|         $clause = $qb->expr()->in('actusers.id', ':users'); | ||||
|  | ||||
|         if ($where instanceof Andx) { | ||||
|             $where->add($clause); | ||||
|   | ||||
| @@ -35,7 +35,7 @@ class EmergencyFilter implements FilterInterface | ||||
|         $this->translator = $translator; | ||||
|     } | ||||
|  | ||||
|     public function addRole() | ||||
|     public function addRole(): ?string | ||||
|     { | ||||
|         return null; | ||||
|     } | ||||
|   | ||||
| @@ -30,19 +30,19 @@ class LocationTypeFilter implements FilterInterface | ||||
|         $this->translatableStringHelper = $translatableStringHelper; | ||||
|     } | ||||
|  | ||||
|     public function addRole() | ||||
|     public function addRole(): ?string | ||||
|     { | ||||
|         return null; | ||||
|     } | ||||
|  | ||||
|     public function alterQuery(QueryBuilder $qb, $data) | ||||
|     { | ||||
|         if (!in_array('location', $qb->getAllAliases(), true)) { | ||||
|             $qb->join('activity.location', 'location'); | ||||
|         if (!in_array('actloc', $qb->getAllAliases(), true)) { | ||||
|             $qb->join('activity.location', 'actloc'); | ||||
|         } | ||||
|  | ||||
|         $where = $qb->getDQLPart('where'); | ||||
|         $clause = $qb->expr()->in('location.locationType', ':locationtype'); | ||||
|         $clause = $qb->expr()->in('actloc.locationType', ':locationtype'); | ||||
|  | ||||
|         if ($where instanceof Andx) { | ||||
|             $where->add($clause); | ||||
|   | ||||
| @@ -36,7 +36,7 @@ class SentReceivedFilter implements FilterInterface | ||||
|         $this->translator = $translator; | ||||
|     } | ||||
|  | ||||
|     public function addRole() | ||||
|     public function addRole(): ?string | ||||
|     { | ||||
|         return null; | ||||
|     } | ||||
|   | ||||
| @@ -29,7 +29,7 @@ class UserFilter implements FilterInterface | ||||
|         $this->userRender = $userRender; | ||||
|     } | ||||
|  | ||||
|     public function addRole() | ||||
|     public function addRole(): ?string | ||||
|     { | ||||
|         return null; | ||||
|     } | ||||
|   | ||||
| @@ -30,20 +30,20 @@ class UserScopeFilter implements FilterInterface | ||||
|         $this->translatableStringHelper = $translatableStringHelper; | ||||
|     } | ||||
|  | ||||
|     public function addRole() | ||||
|     public function addRole(): ?string | ||||
|     { | ||||
|         return null; | ||||
|     } | ||||
|  | ||||
|     public function alterQuery(QueryBuilder $qb, $data) | ||||
|     { | ||||
|         if (!in_array('user', $qb->getAllAliases(), true)) { | ||||
|             $qb->join('activity.user', 'user'); | ||||
|         if (!in_array('actuser', $qb->getAllAliases(), true)) { | ||||
|             $qb->join('activity.user', 'actuser'); | ||||
|         } | ||||
|  | ||||
|         $where = $qb->getDQLPart('where'); | ||||
|  | ||||
|         $clause = $qb->expr()->in('user.mainScope', ':userscope'); | ||||
|         $clause = $qb->expr()->in('actuser.mainScope', ':userscope'); | ||||
|  | ||||
|         if ($where instanceof Andx) { | ||||
|             $where->add($clause); | ||||
|   | ||||
| @@ -33,7 +33,7 @@ class ActivityDateFilter implements FilterInterface | ||||
|         $this->translator = $translator; | ||||
|     } | ||||
|  | ||||
|     public function addRole() | ||||
|     public function addRole(): ?string | ||||
|     { | ||||
|         return null; | ||||
|     } | ||||
|   | ||||
| @@ -13,52 +13,41 @@ namespace Chill\ActivityBundle\Export\Filter; | ||||
|  | ||||
| use Chill\ActivityBundle\Entity\ActivityType; | ||||
| use Chill\ActivityBundle\Export\Declarations; | ||||
| use Chill\ActivityBundle\Repository\ActivityTypeRepository; | ||||
| use Chill\ActivityBundle\Security\Authorization\ActivityStatsVoter; | ||||
| use Chill\ActivityBundle\Repository\ActivityTypeRepositoryInterface; | ||||
| use Chill\MainBundle\Export\ExportElementValidatedInterface; | ||||
| use Chill\MainBundle\Export\FilterInterface; | ||||
| use Chill\MainBundle\Templating\TranslatableStringHelperInterface; | ||||
| use Doctrine\ORM\Query\Expr; | ||||
| use Doctrine\ORM\Query\Expr\Join; | ||||
| use Doctrine\ORM\QueryBuilder; | ||||
| use Symfony\Bridge\Doctrine\Form\Type\EntityType; | ||||
| use Symfony\Component\Form\FormBuilderInterface; | ||||
| use Symfony\Component\Security\Core\Role\Role; | ||||
| use Symfony\Component\Validator\Context\ExecutionContextInterface; | ||||
|  | ||||
| use function count; | ||||
|  | ||||
| class ActivityTypeFilter implements ExportElementValidatedInterface, FilterInterface | ||||
| { | ||||
|     protected ActivityTypeRepository $activityTypeRepository; | ||||
|     protected ActivityTypeRepositoryInterface $activityTypeRepository; | ||||
|  | ||||
|     protected TranslatableStringHelperInterface $translatableStringHelper; | ||||
|  | ||||
|     public function __construct( | ||||
|         TranslatableStringHelperInterface $translatableStringHelper, | ||||
|         ActivityTypeRepository $activityTypeRepository | ||||
|         ActivityTypeRepositoryInterface $activityTypeRepository | ||||
|     ) { | ||||
|         $this->translatableStringHelper = $translatableStringHelper; | ||||
|         $this->activityTypeRepository = $activityTypeRepository; | ||||
|     } | ||||
|  | ||||
|     public function addRole() | ||||
|     public function addRole(): ?string | ||||
|     { | ||||
|         return new Role(ActivityStatsVoter::STATS); | ||||
|         return null; | ||||
|     } | ||||
|  | ||||
|     public function alterQuery(QueryBuilder $qb, $data) | ||||
|     { | ||||
|         $where = $qb->getDQLPart('where'); | ||||
|         $clause = $qb->expr()->in('activity.activityType', ':selected_activity_types'); | ||||
|  | ||||
|         if ($where instanceof Expr\Andx) { | ||||
|             $where->add($clause); | ||||
|         } else { | ||||
|             $where = $qb->expr()->andX($clause); | ||||
|         } | ||||
|  | ||||
|         $qb->add('where', $where); | ||||
|         $qb->andWhere($clause); | ||||
|         $qb->setParameter('selected_activity_types', $data['types']); | ||||
|     } | ||||
|  | ||||
| @@ -70,11 +59,26 @@ class ActivityTypeFilter implements ExportElementValidatedInterface, FilterInter | ||||
|     public function buildForm(FormBuilderInterface $builder) | ||||
|     { | ||||
|         $builder->add('types', EntityType::class, [ | ||||
|             'choices' => $this->activityTypeRepository->findAllActive(), | ||||
|             'class' => ActivityType::class, | ||||
|             'choice_label' => fn (ActivityType $type) => $this->translatableStringHelper->localize($type->getName()), | ||||
|             'group_by' => fn (ActivityType $type) => $this->translatableStringHelper->localize($type->getCategory()->getName()), | ||||
|             'choice_label' => function (ActivityType $aty) { | ||||
|                 return | ||||
|                     ($aty->hasCategory() ? $this->translatableStringHelper->localize($aty->getCategory()->getName()) . ' > ' : '') | ||||
|                     . | ||||
|                     $this->translatableStringHelper->localize($aty->getName()); | ||||
|             }, | ||||
|             'group_by' => function (ActivityType $type) { | ||||
|                 if (!$type->hasCategory()) { | ||||
|                     return null; | ||||
|                 } | ||||
|  | ||||
|                 return $this->translatableStringHelper->localize($type->getCategory()->getName()); | ||||
|             }, | ||||
|             'multiple' => true, | ||||
|             'expanded' => false, | ||||
|             'attr' => [ | ||||
|                 'class' => 'select2', | ||||
|             ], | ||||
|         ]); | ||||
|     } | ||||
|  | ||||
| @@ -104,23 +108,4 @@ class ActivityTypeFilter implements ExportElementValidatedInterface, FilterInter | ||||
|                 ->addViolation(); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Check if a join between Activity and Reason is already defined. | ||||
|      * | ||||
|      * @param Join[] $joins | ||||
|      * @param mixed $alias | ||||
|      * | ||||
|      * @return bool | ||||
|      */ | ||||
|     private function checkJoinAlreadyDefined(array $joins, $alias) | ||||
|     { | ||||
|         foreach ($joins as $join) { | ||||
|             if ($join->getAlias() === $alias) { | ||||
|                 return true; | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         return false; | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -14,21 +14,18 @@ namespace Chill\ActivityBundle\Export\Filter\PersonFilters; | ||||
| use Chill\ActivityBundle\Entity\ActivityReason; | ||||
| use Chill\ActivityBundle\Export\Declarations; | ||||
| use Chill\ActivityBundle\Repository\ActivityReasonRepository; | ||||
| use Chill\ActivityBundle\Security\Authorization\ActivityStatsVoter; | ||||
| use Chill\MainBundle\Export\ExportElementValidatedInterface; | ||||
| use Chill\MainBundle\Export\FilterInterface; | ||||
| use Chill\MainBundle\Templating\TranslatableStringHelper; | ||||
| use Chill\MainBundle\Templating\TranslatableStringHelperInterface; | ||||
| use Doctrine\ORM\Query\Expr; | ||||
| use Doctrine\ORM\Query\Expr\Join; | ||||
| use Doctrine\ORM\QueryBuilder; | ||||
| use Symfony\Bridge\Doctrine\Form\Type\EntityType; | ||||
| use Symfony\Component\Form\FormBuilderInterface; | ||||
| use Symfony\Component\Security\Core\Role\Role; | ||||
| use Symfony\Component\Validator\Context\ExecutionContextInterface; | ||||
|  | ||||
| use function array_key_exists; | ||||
| use function count; | ||||
| use function in_array; | ||||
|  | ||||
| class ActivityReasonFilter implements ExportElementValidatedInterface, FilterInterface | ||||
| { | ||||
| @@ -44,9 +41,9 @@ class ActivityReasonFilter implements ExportElementValidatedInterface, FilterInt | ||||
|         $this->activityReasonRepository = $activityReasonRepository; | ||||
|     } | ||||
|  | ||||
|     public function addRole() | ||||
|     public function addRole(): ?string | ||||
|     { | ||||
|         return new Role(ActivityStatsVoter::STATS); | ||||
|         return null; | ||||
|     } | ||||
|  | ||||
|     public function alterQuery(QueryBuilder $qb, $data) | ||||
| @@ -54,20 +51,9 @@ class ActivityReasonFilter implements ExportElementValidatedInterface, FilterInt | ||||
|         $where = $qb->getDQLPart('where'); | ||||
|         $join = $qb->getDQLPart('join'); | ||||
|         $clause = $qb->expr()->in('reasons', ':selected_activity_reasons'); | ||||
|         //dump($join); | ||||
|         // add a join to reasons only if needed | ||||
|         if ( | ||||
|             ( | ||||
|                 array_key_exists('activity', $join) | ||||
|             && !$this->checkJoinAlreadyDefined($join['activity'], 'reasons') | ||||
|             ) | ||||
|             || (!array_key_exists('activity', $join)) | ||||
|         ) { | ||||
|             $qb->add( | ||||
|                 'join', | ||||
|                 ['activity' => new Join(Join::INNER_JOIN, 'activity.reasons', 'reasons')], | ||||
|                 true | ||||
|             ); | ||||
|  | ||||
|         if (!in_array('actreasons', $qb->getAllAliases(), true)) { | ||||
|             $qb->join('activity.reasons', 'actreasons'); | ||||
|         } | ||||
|  | ||||
|         if ($where instanceof Expr\Andx) { | ||||
| @@ -125,21 +111,4 @@ class ActivityReasonFilter implements ExportElementValidatedInterface, FilterInt | ||||
|                 ->addViolation(); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Check if a join between Activity and Reason is already defined. | ||||
|      * | ||||
|      * @param Join[] $joins | ||||
|      * @param mixed $alias | ||||
|      */ | ||||
|     private function checkJoinAlreadyDefined(array $joins, $alias): bool | ||||
|     { | ||||
|         foreach ($joins as $join) { | ||||
|             if ($join->getAlias() === $alias) { | ||||
|                 return true; | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         return false; | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -11,6 +11,7 @@ declare(strict_types=1); | ||||
|  | ||||
| namespace Chill\ActivityBundle\Export\Filter\PersonFilters; | ||||
|  | ||||
| use Chill\ActivityBundle\Entity\Activity; | ||||
| use Chill\ActivityBundle\Entity\ActivityReason; | ||||
| use Chill\ActivityBundle\Repository\ActivityReasonRepository; | ||||
| use Chill\MainBundle\Export\ExportElementValidatedInterface; | ||||
| @@ -52,17 +53,17 @@ class PersonHavingActivityBetweenDateFilter implements ExportElementValidatedInt | ||||
|         $this->translator = $translator; | ||||
|     } | ||||
|  | ||||
|     public function addRole() | ||||
|     public function addRole(): ?string | ||||
|     { | ||||
|         return null; | ||||
|     } | ||||
|  | ||||
|     public function alterQuery(QueryBuilder $qb, $data) | ||||
|     { | ||||
|         // create a query for activity | ||||
|         // create a subquery for activity | ||||
|         $sqb = $qb->getEntityManager()->createQueryBuilder(); | ||||
|         $sqb->select('person_person_having_activity.id') | ||||
|             ->from('ChillActivityBundle:Activity', 'activity_person_having_activity') | ||||
|             ->from(Activity::class, 'activity_person_having_activity') | ||||
|             ->join('activity_person_having_activity.person', 'person_person_having_activity'); | ||||
|  | ||||
|         // add clause between date | ||||
| @@ -197,7 +198,7 @@ class PersonHavingActivityBetweenDateFilter implements ExportElementValidatedInt | ||||
|  | ||||
|     public function getTitle() | ||||
|     { | ||||
|         return 'Filtered by person having an activity in a period'; | ||||
|         return 'Filter by person having an activity in a period'; | ||||
|     } | ||||
|  | ||||
|     public function validateForm($data, ExecutionContextInterface $context) | ||||
|   | ||||
| @@ -12,48 +12,33 @@ declare(strict_types=1); | ||||
| namespace Chill\ActivityBundle\Form\Type; | ||||
|  | ||||
| use Chill\ActivityBundle\Entity\ActivityType; | ||||
| use Chill\ActivityBundle\Repository\ActivityTypeRepository; | ||||
| use Chill\ActivityBundle\Repository\ActivityTypeRepositoryInterface; | ||||
| use Chill\MainBundle\Templating\TranslatableStringHelperInterface; | ||||
| use Doctrine\DBAL\Types\Types; | ||||
| use Doctrine\ORM\QueryBuilder; | ||||
| use Symfony\Bridge\Doctrine\Form\Type\EntityType; | ||||
| use Symfony\Component\Form\AbstractType; | ||||
| use Symfony\Component\Form\FormBuilderInterface; | ||||
| use Symfony\Component\OptionsResolver\OptionsResolver; | ||||
|  | ||||
| class TranslatableActivityType extends AbstractType | ||||
| { | ||||
|     protected ActivityTypeRepository $activityTypeRepository; | ||||
|     protected ActivityTypeRepositoryInterface $activityTypeRepository; | ||||
|  | ||||
|     protected TranslatableStringHelperInterface $translatableStringHelper; | ||||
|  | ||||
|     public function __construct( | ||||
|         TranslatableStringHelperInterface $helper, | ||||
|         ActivityTypeRepository $activityTypeRepository | ||||
|         ActivityTypeRepositoryInterface $activityTypeRepository | ||||
|     ) { | ||||
|         $this->translatableStringHelper = $helper; | ||||
|         $this->activityTypeRepository = $activityTypeRepository; | ||||
|     } | ||||
|  | ||||
|     public function buildForm(FormBuilderInterface $builder, array $options) | ||||
|     { | ||||
|         /** @var QueryBuilder $qb */ | ||||
|         $qb = $options['query_builder']; | ||||
|  | ||||
|         if (true === $options['active_only']) { | ||||
|             $qb->where($qb->expr()->eq('at.active', ':active')); | ||||
|             $qb->setParameter('active', true, Types::BOOLEAN); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     public function configureOptions(OptionsResolver $resolver) | ||||
|     { | ||||
|         $resolver->setDefaults( | ||||
|             [ | ||||
|                 'class' => ActivityType::class, | ||||
|                 'active_only' => true, | ||||
|                 'query_builder' => $this->activityTypeRepository | ||||
|                     ->createQueryBuilder('at'), | ||||
|                 'choices' => $this->activityTypeRepository->findAllActive(), | ||||
|                 'choice_label' => function (ActivityType $type) { | ||||
|                     return $this->translatableStringHelper->localize($type->getName()); | ||||
|                 }, | ||||
|   | ||||
| @@ -12,19 +12,54 @@ declare(strict_types=1); | ||||
| namespace Chill\ActivityBundle\Repository; | ||||
|  | ||||
| use Chill\ActivityBundle\Entity\ActivityType; | ||||
| use Doctrine\Bundle\DoctrineBundle\Repository\ServiceEntityRepository; | ||||
| use Doctrine\Persistence\ManagerRegistry; | ||||
| use Doctrine\ORM\EntityManagerInterface; | ||||
| use Doctrine\ORM\EntityRepository; | ||||
|  | ||||
| /** | ||||
|  * @method ActivityType|null find($id, $lockMode = null, $lockVersion = null) | ||||
|  * @method ActivityType|null findOneBy(array $criteria, array $orderBy = null) | ||||
|  * @method ActivityType[]    findAll() | ||||
|  * @method ActivityType[]    findBy(array $criteria, array $orderBy = null, $limit = null, $offset = null) | ||||
|  */ | ||||
| class ActivityTypeRepository extends ServiceEntityRepository | ||||
| final class ActivityTypeRepository implements ActivityTypeRepositoryInterface | ||||
| { | ||||
|     public function __construct(ManagerRegistry $registry) | ||||
|     private EntityRepository $repository; | ||||
|  | ||||
|     public function __construct(EntityManagerInterface $em) | ||||
|     { | ||||
|         parent::__construct($registry, ActivityType::class); | ||||
|         $this->repository = $em->getRepository(ActivityType::class); | ||||
|     } | ||||
|  | ||||
|     public function find($id): ?ActivityType | ||||
|     { | ||||
|         return $this->repository->find($id); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * @return array|ActivityType[] | ||||
|      */ | ||||
|     public function findAll(): array | ||||
|     { | ||||
|         return $this->repository->findAll(); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * @return array|ActivityType[] | ||||
|      */ | ||||
|     public function findAllActive(): array | ||||
|     { | ||||
|         return $this->findBy(['active' => true]); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * @return array|ActivityType[] | ||||
|      */ | ||||
|     public function findBy(array $criteria, ?array $orderBy = null, ?int $limit = null, ?int $offset = null): array | ||||
|     { | ||||
|         return $this->repository->findBy($criteria, $orderBy, $limit, $offset); | ||||
|     } | ||||
|  | ||||
|     public function findOneBy(array $criteria): ?ActivityType | ||||
|     { | ||||
|         return $this->repository->findOneBy($criteria); | ||||
|     } | ||||
|  | ||||
|     public function getClassName(): string | ||||
|     { | ||||
|         return ActivityType::class; | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -0,0 +1,23 @@ | ||||
| <?php | ||||
|  | ||||
| declare(strict_types=1); | ||||
|  | ||||
| /* | ||||
|  * Chill is a software for social workers | ||||
|  * | ||||
|  * For the full copyright and license information, please view | ||||
|  * the LICENSE file that was distributed with this source code. | ||||
|  */ | ||||
|  | ||||
| namespace Chill\ActivityBundle\Repository; | ||||
|  | ||||
| use Chill\ActivityBundle\Entity\ActivityType; | ||||
| use Doctrine\Persistence\ObjectRepository; | ||||
|  | ||||
| interface ActivityTypeRepositoryInterface extends ObjectRepository | ||||
| { | ||||
|     /** | ||||
|      * @return array|ActivityType[] | ||||
|      */ | ||||
|     public function findAllActive(): array; | ||||
| } | ||||
| @@ -17,7 +17,7 @@ const getLocations = () => fetchResults('/api/1.0/main/location.json'); | ||||
|  | ||||
| const getLocationTypes = () => fetchResults('/api/1.0/main/location-type.json'); | ||||
|  | ||||
| const getUserCurrentLocation =  | ||||
| const getUserCurrentLocation = | ||||
|     () => fetch('/api/1.0/main/user-current-location.json') | ||||
|         .then(response => { | ||||
|             if (response.ok) { return response.json(); } | ||||
| @@ -35,6 +35,13 @@ const getLocationTypeByDefaultFor = (entity) => { | ||||
|     ); | ||||
| }; | ||||
|  | ||||
| /** | ||||
|  * Post a location | ||||
|  * | ||||
|  * **NOTE**: also in use for Calendar | ||||
|  * @param body | ||||
|  * @returns {Promise<T>} | ||||
|  */ | ||||
| const postLocation = (body) => { | ||||
|     const url = `/api/1.0/main/location.json`; | ||||
|     return fetch(url, { | ||||
|   | ||||
| @@ -55,7 +55,7 @@ const makeAccompanyingPeriodLocation = (locationType, store) => { | ||||
|  | ||||
| export default function prepareLocations(store) { | ||||
|  | ||||
| // find the locations | ||||
|   // find the locations | ||||
|     let allLocations = getLocations().then( | ||||
|         (results) => { | ||||
|             store.commit('addAvailableLocationGroup', { | ||||
| @@ -111,7 +111,7 @@ export default function prepareLocations(store) { | ||||
|         if (window.default_location_id) { | ||||
|             for (let group of store.state.availableLocations) { | ||||
|                 let location = group.locations.find((l) => l.id === window.default_location_id); | ||||
|                 if (location !== undefined & store.state.activity.location === null) { | ||||
|                 if (location !== undefined && store.state.activity.location === null) { | ||||
|                     store.dispatch('updateLocation', location); | ||||
|                     break; | ||||
|                 } | ||||
|   | ||||
| @@ -1,3 +1,14 @@ | ||||
| {# | ||||
|     WARNING: this file is in use in both ActivityBundle and CalendarBundle. | ||||
|  | ||||
|     Take care when editing this file. | ||||
|  | ||||
|     Maybe should we think about abstracting this file a bit more ? Moving it to PersonBundle ? | ||||
| #} | ||||
| {% if context == 'calendar_accompanyingCourse' %} | ||||
|     {% import "@ChillCalendar/_invite.html.twig" as invite %} | ||||
| {% endif %} | ||||
|  | ||||
| {% macro href(pathname, key, value) %} | ||||
|     {% set parms = { (key): value } %} | ||||
|     {{ path(pathname, parms) }} | ||||
| @@ -18,7 +29,7 @@ | ||||
| {% endmacro %} | ||||
|  | ||||
| {% set blocks = [] %} | ||||
| {% if entity.activityType.personsVisible %} | ||||
| {% if context == 'calendar_accompanyingCourse' or entity.activityType.personsVisible %} | ||||
|     {% if context == 'person' %} | ||||
|         {% set blocks = blocks|merge([{ | ||||
|             'title': 'Others persons'|trans, | ||||
| @@ -43,7 +54,7 @@ | ||||
|         }]) %} | ||||
|     {% endif %} | ||||
| {% endif %} | ||||
| {% if entity.activityType.thirdPartiesVisible %} | ||||
| {% if context == 'calendar_accompanyingCourse' or entity.activityType.thirdPartiesVisible %} | ||||
|     {% set blocks = blocks|merge([{ | ||||
|         'title': 'Third parties'|trans, | ||||
|         'items': entity.thirdParties, | ||||
| @@ -52,7 +63,7 @@ | ||||
|         'key'  : 'id', | ||||
|     }]) %} | ||||
| {% endif %} | ||||
| {% if entity.activityType.usersVisible %} | ||||
| {% if context == 'calendar_accompanyingCourse' or entity.activityType.usersVisible %} | ||||
|     {% set blocks = blocks|merge([{ | ||||
|         'title': 'Users concerned'|trans, | ||||
|         'items': entity.users, | ||||
| @@ -132,6 +143,12 @@ | ||||
|                             {% if bloc.type == 'user' %} | ||||
|                                 <span class="badge-user"> | ||||
|                                     {{ item|chill_entity_render_box({'render': 'raw', 'addAltNames': false }) }} | ||||
|                                     {%- if context == 'calendar_accompanyingCourse' %} | ||||
|                                         {% set invite = entity.inviteForUser(item) %} | ||||
|                                         {% if invite is not null %} | ||||
|                                         {{ invite.invite_span(invite) }} | ||||
|                                         {% endif %} | ||||
|                                     {%- endif -%} | ||||
|                                 </span> | ||||
|                             {% else %} | ||||
|                                 {{ _self.insert_onthefly(bloc.type, item) }} | ||||
|   | ||||
| @@ -13,10 +13,10 @@ namespace Chill\ActivityBundle\Security\Authorization; | ||||
|  | ||||
| use Chill\MainBundle\Entity\Center; | ||||
| use Chill\MainBundle\Security\Authorization\AbstractChillVoter; | ||||
| use Chill\MainBundle\Security\Authorization\AuthorizationHelper; | ||||
| use Chill\MainBundle\Security\Authorization\VoterHelperFactoryInterface; | ||||
| use Chill\MainBundle\Security\Authorization\VoterHelperInterface; | ||||
| use Chill\MainBundle\Security\ProvideRoleHierarchyInterface; | ||||
|  | ||||
| use function in_array; | ||||
| use Symfony\Component\Security\Core\Authentication\Token\TokenInterface; | ||||
|  | ||||
| class ActivityStatsVoter extends AbstractChillVoter implements ProvideRoleHierarchyInterface | ||||
| { | ||||
| @@ -24,14 +24,14 @@ class ActivityStatsVoter extends AbstractChillVoter implements ProvideRoleHierar | ||||
|  | ||||
|     public const STATS = 'CHILL_ACTIVITY_STATS'; | ||||
|  | ||||
|     /** | ||||
|      * @var AuthorizationHelper | ||||
|      */ | ||||
|     protected $helper; | ||||
|     protected VoterHelperInterface $helper; | ||||
|  | ||||
|     public function __construct(AuthorizationHelper $helper) | ||||
|     public function __construct(VoterHelperFactoryInterface $voterHelperFactory) | ||||
|     { | ||||
|         $this->helper = $helper; | ||||
|         $this->helper = $voterHelperFactory | ||||
|             ->generate(self::class) | ||||
|             ->addCheckFor(Center::class, [self::STATS, self::LISTS]) | ||||
|             ->build(); | ||||
|     } | ||||
|  | ||||
|     public function getRoles(): array | ||||
| @@ -49,30 +49,14 @@ class ActivityStatsVoter extends AbstractChillVoter implements ProvideRoleHierar | ||||
|         return $this->getAttributes(); | ||||
|     } | ||||
|  | ||||
|     protected function getSupportedClasses() | ||||
|     { | ||||
|         return [Center::class]; | ||||
|     } | ||||
|  | ||||
|     protected function isGranted($attribute, $object, $user = null) | ||||
|     { | ||||
|         if (!$user instanceof \Symfony\Component\Security\Core\User\UserInterface) { | ||||
|             return false; | ||||
|         } | ||||
|  | ||||
|         return $this->helper->userHasAccess($user, $object, $attribute); | ||||
|     } | ||||
|  | ||||
|     protected function supports($attribute, $subject) | ||||
|     { | ||||
|         if ( | ||||
|             $subject instanceof Center | ||||
|             && in_array($attribute, $this->getAttributes(), true) | ||||
|         ) { | ||||
|             return true; | ||||
|         } | ||||
|         return $this->helper->supports($attribute, $subject); | ||||
|     } | ||||
|  | ||||
|         return false; | ||||
|     protected function voteOnAttribute($attribute, $subject, TokenInterface $token) | ||||
|     { | ||||
|         return $this->helper->voteOnAttribute($attribute, $subject, $token); | ||||
|     } | ||||
|  | ||||
|     private function getAttributes() | ||||
|   | ||||
| @@ -0,0 +1,69 @@ | ||||
| <?php | ||||
|  | ||||
| /** | ||||
|  * Chill is a software for social workers. | ||||
|  * | ||||
|  * For the full copyright and license information, please view | ||||
|  * the LICENSE file that was distributed with this source code. | ||||
|  */ | ||||
|  | ||||
| declare(strict_types=1); | ||||
|  | ||||
| /* | ||||
|  * Chill is a software for social workers | ||||
|  * | ||||
|  * For the full copyright and license information, please view | ||||
|  * the LICENSE file that was distributed with this source code. | ||||
|  */ | ||||
|  | ||||
| namespace Chill\ActivityBundle\Tests\Export\Aggregator\ACPAggregators; | ||||
|  | ||||
| use Chill\ActivityBundle\Entity\Activity; | ||||
| use Chill\ActivityBundle\Export\Aggregator\ACPAggregators\BySocialActionAggregator; | ||||
| use Chill\MainBundle\Test\Export\AbstractAggregatorTest; | ||||
| use Doctrine\ORM\EntityManagerInterface; | ||||
|  | ||||
| /** | ||||
|  * @internal | ||||
|  * @coversNothing | ||||
|  */ | ||||
| final class BySocialActionAggregatorTest extends AbstractAggregatorTest | ||||
| { | ||||
|     private BySocialActionAggregator $aggregator; | ||||
|  | ||||
|     protected function setUp(): void | ||||
|     { | ||||
|         self::bootKernel(); | ||||
|  | ||||
|         $this->aggregator = self::$container->get('chill.activity.export.bysocialaction_aggregator'); | ||||
|     } | ||||
|  | ||||
|     public function getAggregator() | ||||
|     { | ||||
|         return $this->aggregator; | ||||
|     } | ||||
|  | ||||
|     public function getFormData(): array | ||||
|     { | ||||
|         return [ | ||||
|             [], | ||||
|         ]; | ||||
|     } | ||||
|  | ||||
|     public function getQueryBuilders(): array | ||||
|     { | ||||
|         if (null === self::$kernel) { | ||||
|             self::bootKernel(); | ||||
|         } | ||||
|  | ||||
|         $em = self::$container->get(EntityManagerInterface::class); | ||||
|  | ||||
|         return [ | ||||
|             $em->createQueryBuilder() | ||||
|                 ->select('count(activity.id)') | ||||
|                 ->from(Activity::class, 'activity') | ||||
|                 ->join('activity.accompanyingPeriod', 'acp') | ||||
|                 ->join('activity.socialActions', 'actsocialaction'), | ||||
|         ]; | ||||
|     } | ||||
| } | ||||
| @@ -0,0 +1,69 @@ | ||||
| <?php | ||||
|  | ||||
| /** | ||||
|  * Chill is a software for social workers. | ||||
|  * | ||||
|  * For the full copyright and license information, please view | ||||
|  * the LICENSE file that was distributed with this source code. | ||||
|  */ | ||||
|  | ||||
| declare(strict_types=1); | ||||
|  | ||||
| /* | ||||
|  * Chill is a software for social workers | ||||
|  * | ||||
|  * For the full copyright and license information, please view | ||||
|  * the LICENSE file that was distributed with this source code. | ||||
|  */ | ||||
|  | ||||
| namespace Chill\ActivityBundle\Tests\Export\Aggregator\ACPAggregators; | ||||
|  | ||||
| use Chill\ActivityBundle\Entity\Activity; | ||||
| use Chill\ActivityBundle\Export\Aggregator\ACPAggregators\BySocialIssueAggregator; | ||||
| use Chill\MainBundle\Test\Export\AbstractAggregatorTest; | ||||
| use Doctrine\ORM\EntityManagerInterface; | ||||
|  | ||||
| /** | ||||
|  * @internal | ||||
|  * @coversNothing | ||||
|  */ | ||||
| final class BySocialIssueAggregatorTest extends AbstractAggregatorTest | ||||
| { | ||||
|     private BySocialIssueAggregator $aggregator; | ||||
|  | ||||
|     protected function setUp(): void | ||||
|     { | ||||
|         self::bootKernel(); | ||||
|  | ||||
|         $this->aggregator = self::$container->get('chill.activity.export.bysocialissue_aggregator'); | ||||
|     } | ||||
|  | ||||
|     public function getAggregator() | ||||
|     { | ||||
|         return $this->aggregator; | ||||
|     } | ||||
|  | ||||
|     public function getFormData(): array | ||||
|     { | ||||
|         return [ | ||||
|             [], | ||||
|         ]; | ||||
|     } | ||||
|  | ||||
|     public function getQueryBuilders(): array | ||||
|     { | ||||
|         if (null === self::$kernel) { | ||||
|             self::bootKernel(); | ||||
|         } | ||||
|  | ||||
|         $em = self::$container->get(EntityManagerInterface::class); | ||||
|  | ||||
|         return [ | ||||
|             $em->createQueryBuilder() | ||||
|                 ->select('count(activity.id)') | ||||
|                 ->from(Activity::class, 'activity') | ||||
|                 ->join('activity.accompanyingPeriod', 'acp') | ||||
|                 ->join('activity.socialIssues', 'actsocialissue'), | ||||
|         ]; | ||||
|     } | ||||
| } | ||||
| @@ -0,0 +1,69 @@ | ||||
| <?php | ||||
|  | ||||
| /** | ||||
|  * Chill is a software for social workers. | ||||
|  * | ||||
|  * For the full copyright and license information, please view | ||||
|  * the LICENSE file that was distributed with this source code. | ||||
|  */ | ||||
|  | ||||
| declare(strict_types=1); | ||||
|  | ||||
| /* | ||||
|  * Chill is a software for social workers | ||||
|  * | ||||
|  * For the full copyright and license information, please view | ||||
|  * the LICENSE file that was distributed with this source code. | ||||
|  */ | ||||
|  | ||||
| namespace Chill\ActivityBundle\Tests\Export\Aggregator\ACPAggregators; | ||||
|  | ||||
| use Chill\ActivityBundle\Entity\Activity; | ||||
| use Chill\ActivityBundle\Export\Aggregator\ACPAggregators\ByThirdpartyAggregator; | ||||
| use Chill\MainBundle\Test\Export\AbstractAggregatorTest; | ||||
| use Doctrine\ORM\EntityManagerInterface; | ||||
|  | ||||
| /** | ||||
|  * @internal | ||||
|  * @coversNothing | ||||
|  */ | ||||
| final class ByThirdpartyAggregatorTest extends AbstractAggregatorTest | ||||
| { | ||||
|     private ByThirdpartyAggregator $aggregator; | ||||
|  | ||||
|     protected function setUp(): void | ||||
|     { | ||||
|         self::bootKernel(); | ||||
|  | ||||
|         $this->aggregator = self::$container->get('chill.activity.export.bythirdparty_aggregator'); | ||||
|     } | ||||
|  | ||||
|     public function getAggregator() | ||||
|     { | ||||
|         return $this->aggregator; | ||||
|     } | ||||
|  | ||||
|     public function getFormData(): array | ||||
|     { | ||||
|         return [ | ||||
|             [], | ||||
|         ]; | ||||
|     } | ||||
|  | ||||
|     public function getQueryBuilders(): array | ||||
|     { | ||||
|         if (null === self::$kernel) { | ||||
|             self::bootKernel(); | ||||
|         } | ||||
|  | ||||
|         $em = self::$container->get(EntityManagerInterface::class); | ||||
|  | ||||
|         return [ | ||||
|             $em->createQueryBuilder() | ||||
|                 ->select('count(activity.id)') | ||||
|                 ->from(Activity::class, 'activity') | ||||
|                 ->join('activity.accompanyingPeriod', 'acp') | ||||
|                 ->join('activity.thirdParties', 'acttparty'), | ||||
|         ]; | ||||
|     } | ||||
| } | ||||
| @@ -0,0 +1,69 @@ | ||||
| <?php | ||||
|  | ||||
| /** | ||||
|  * Chill is a software for social workers. | ||||
|  * | ||||
|  * For the full copyright and license information, please view | ||||
|  * the LICENSE file that was distributed with this source code. | ||||
|  */ | ||||
|  | ||||
| declare(strict_types=1); | ||||
|  | ||||
| /* | ||||
|  * Chill is a software for social workers | ||||
|  * | ||||
|  * For the full copyright and license information, please view | ||||
|  * the LICENSE file that was distributed with this source code. | ||||
|  */ | ||||
|  | ||||
| namespace Chill\ActivityBundle\Tests\Export\Aggregator\ACPAggregators; | ||||
|  | ||||
| use Chill\ActivityBundle\Entity\Activity; | ||||
| use Chill\ActivityBundle\Export\Aggregator\ACPAggregators\ByUserAggregator; | ||||
| use Chill\MainBundle\Test\Export\AbstractAggregatorTest; | ||||
| use Doctrine\ORM\EntityManagerInterface; | ||||
|  | ||||
| /** | ||||
|  * @internal | ||||
|  * @coversNothing | ||||
|  */ | ||||
| final class ByUserAggregatorTest extends AbstractAggregatorTest | ||||
| { | ||||
|     private ByUserAggregator $aggregator; | ||||
|  | ||||
|     protected function setUp(): void | ||||
|     { | ||||
|         self::bootKernel(); | ||||
|  | ||||
|         $this->aggregator = self::$container->get('chill.activity.export.byuser_aggregator'); | ||||
|     } | ||||
|  | ||||
|     public function getAggregator() | ||||
|     { | ||||
|         return $this->aggregator; | ||||
|     } | ||||
|  | ||||
|     public function getFormData(): array | ||||
|     { | ||||
|         return [ | ||||
|             [], | ||||
|         ]; | ||||
|     } | ||||
|  | ||||
|     public function getQueryBuilders(): array | ||||
|     { | ||||
|         if (null === self::$kernel) { | ||||
|             self::bootKernel(); | ||||
|         } | ||||
|  | ||||
|         $em = self::$container->get(EntityManagerInterface::class); | ||||
|  | ||||
|         return [ | ||||
|             $em->createQueryBuilder() | ||||
|                 ->select('count(activity.id)') | ||||
|                 ->from(Activity::class, 'activity') | ||||
|                 ->join('activity.accompanyingPeriod', 'acp') | ||||
|                 ->join('activity.users', 'actusers'), | ||||
|         ]; | ||||
|     } | ||||
| } | ||||
| @@ -0,0 +1,76 @@ | ||||
| <?php | ||||
|  | ||||
| /** | ||||
|  * Chill is a software for social workers. | ||||
|  * | ||||
|  * For the full copyright and license information, please view | ||||
|  * the LICENSE file that was distributed with this source code. | ||||
|  */ | ||||
|  | ||||
| declare(strict_types=1); | ||||
|  | ||||
| /* | ||||
|  * Chill is a software for social workers | ||||
|  * | ||||
|  * For the full copyright and license information, please view | ||||
|  * the LICENSE file that was distributed with this source code. | ||||
|  */ | ||||
|  | ||||
| namespace Chill\ActivityBundle\Tests\Export\Aggregator\ACPAggregators; | ||||
|  | ||||
| use Chill\ActivityBundle\Entity\Activity; | ||||
| use Chill\ActivityBundle\Export\Aggregator\ACPAggregators\DateAggregator; | ||||
| use Chill\MainBundle\Test\Export\AbstractAggregatorTest; | ||||
| use Doctrine\ORM\EntityManagerInterface; | ||||
|  | ||||
| /** | ||||
|  * @internal | ||||
|  * @coversNothing | ||||
|  */ | ||||
| final class DateAggregatorTest extends AbstractAggregatorTest | ||||
| { | ||||
|     private DateAggregator $aggregator; | ||||
|  | ||||
|     protected function setUp(): void | ||||
|     { | ||||
|         self::bootKernel(); | ||||
|  | ||||
|         $this->aggregator = self::$container->get('chill.activity.export.date_aggregator'); | ||||
|     } | ||||
|  | ||||
|     public function getAggregator() | ||||
|     { | ||||
|         return $this->aggregator; | ||||
|     } | ||||
|  | ||||
|     public function getFormData(): array | ||||
|     { | ||||
|         return [ | ||||
|             [ | ||||
|                 'frequency' => 'month', | ||||
|             ], | ||||
|             [ | ||||
|                 'frequency' => 'week', | ||||
|             ], | ||||
|             [ | ||||
|                 'frequency' => 'year', | ||||
|             ], | ||||
|         ]; | ||||
|     } | ||||
|  | ||||
|     public function getQueryBuilders(): array | ||||
|     { | ||||
|         if (null === self::$kernel) { | ||||
|             self::bootKernel(); | ||||
|         } | ||||
|  | ||||
|         $em = self::$container->get(EntityManagerInterface::class); | ||||
|  | ||||
|         return [ | ||||
|             $em->createQueryBuilder() | ||||
|                 ->select('count(activity.id)') | ||||
|                 ->from(Activity::class, 'activity') | ||||
|                 ->join('activity.accompanyingPeriod', 'acp'), | ||||
|         ]; | ||||
|     } | ||||
| } | ||||
| @@ -0,0 +1,69 @@ | ||||
| <?php | ||||
|  | ||||
| /** | ||||
|  * Chill is a software for social workers. | ||||
|  * | ||||
|  * For the full copyright and license information, please view | ||||
|  * the LICENSE file that was distributed with this source code. | ||||
|  */ | ||||
|  | ||||
| declare(strict_types=1); | ||||
|  | ||||
| /* | ||||
|  * Chill is a software for social workers | ||||
|  * | ||||
|  * For the full copyright and license information, please view | ||||
|  * the LICENSE file that was distributed with this source code. | ||||
|  */ | ||||
|  | ||||
| namespace Chill\ActivityBundle\Tests\Export\Aggregator\ACPAggregators; | ||||
|  | ||||
| use Chill\ActivityBundle\Entity\Activity; | ||||
| use Chill\ActivityBundle\Export\Aggregator\ACPAggregators\LocationTypeAggregator; | ||||
| use Chill\MainBundle\Test\Export\AbstractAggregatorTest; | ||||
| use Doctrine\ORM\EntityManagerInterface; | ||||
|  | ||||
| /** | ||||
|  * @internal | ||||
|  * @coversNothing | ||||
|  */ | ||||
| final class LocationTypeAggregatorTest extends AbstractAggregatorTest | ||||
| { | ||||
|     private LocationTypeAggregator $aggregator; | ||||
|  | ||||
|     protected function setUp(): void | ||||
|     { | ||||
|         self::bootKernel(); | ||||
|  | ||||
|         $this->aggregator = self::$container->get('chill.activity.export.locationtype_aggregator'); | ||||
|     } | ||||
|  | ||||
|     public function getAggregator() | ||||
|     { | ||||
|         return $this->aggregator; | ||||
|     } | ||||
|  | ||||
|     public function getFormData(): array | ||||
|     { | ||||
|         return [ | ||||
|             [], | ||||
|         ]; | ||||
|     } | ||||
|  | ||||
|     public function getQueryBuilders(): array | ||||
|     { | ||||
|         if (null === self::$kernel) { | ||||
|             self::bootKernel(); | ||||
|         } | ||||
|  | ||||
|         $em = self::$container->get(EntityManagerInterface::class); | ||||
|  | ||||
|         return [ | ||||
|             $em->createQueryBuilder() | ||||
|                 ->select('count(activity.id)') | ||||
|                 ->from(Activity::class, 'activity') | ||||
|                 ->join('activity.accompanyingPeriod', 'acp') | ||||
|                 ->join('activity.location', 'actloc'), | ||||
|         ]; | ||||
|     } | ||||
| } | ||||
| @@ -0,0 +1,69 @@ | ||||
| <?php | ||||
|  | ||||
| /** | ||||
|  * Chill is a software for social workers. | ||||
|  * | ||||
|  * For the full copyright and license information, please view | ||||
|  * the LICENSE file that was distributed with this source code. | ||||
|  */ | ||||
|  | ||||
| declare(strict_types=1); | ||||
|  | ||||
| /* | ||||
|  * Chill is a software for social workers | ||||
|  * | ||||
|  * For the full copyright and license information, please view | ||||
|  * the LICENSE file that was distributed with this source code. | ||||
|  */ | ||||
|  | ||||
| namespace Chill\ActivityBundle\Tests\Export\Aggregator\ACPAggregators; | ||||
|  | ||||
| use Chill\ActivityBundle\Entity\Activity; | ||||
| use Chill\ActivityBundle\Export\Aggregator\ACPAggregators\UserScopeAggregator; | ||||
| use Chill\MainBundle\Test\Export\AbstractAggregatorTest; | ||||
| use Doctrine\ORM\EntityManagerInterface; | ||||
|  | ||||
| /** | ||||
|  * @internal | ||||
|  * @coversNothing | ||||
|  */ | ||||
| final class UserScopeAggregatorTest extends AbstractAggregatorTest | ||||
| { | ||||
|     private UserScopeAggregator $aggregator; | ||||
|  | ||||
|     protected function setUp(): void | ||||
|     { | ||||
|         self::bootKernel(); | ||||
|  | ||||
|         $this->aggregator = self::$container->get('chill.activity.export.userscope_aggregator'); | ||||
|     } | ||||
|  | ||||
|     public function getAggregator() | ||||
|     { | ||||
|         return $this->aggregator; | ||||
|     } | ||||
|  | ||||
|     public function getFormData(): array | ||||
|     { | ||||
|         return [ | ||||
|             [], | ||||
|         ]; | ||||
|     } | ||||
|  | ||||
|     public function getQueryBuilders(): array | ||||
|     { | ||||
|         if (null === self::$kernel) { | ||||
|             self::bootKernel(); | ||||
|         } | ||||
|  | ||||
|         $em = self::$container->get(EntityManagerInterface::class); | ||||
|  | ||||
|         return [ | ||||
|             $em->createQueryBuilder() | ||||
|                 ->select('count(activity.id)') | ||||
|                 ->from(Activity::class, 'activity') | ||||
|                 ->join('activity.accompanyingPeriod', 'acp') | ||||
|                 ->join('activity.user', 'actuser'), | ||||
|         ]; | ||||
|     } | ||||
| } | ||||
| @@ -11,6 +11,7 @@ declare(strict_types=1); | ||||
|  | ||||
| namespace Chill\ActivityBundle\Tests\Export\Aggregator; | ||||
|  | ||||
| use Chill\ActivityBundle\Export\Aggregator\PersonAggregators\ActivityReasonAggregator; | ||||
| use Chill\MainBundle\Test\Export\AbstractAggregatorTest; | ||||
|  | ||||
| /** | ||||
| @@ -21,10 +22,7 @@ use Chill\MainBundle\Test\Export\AbstractAggregatorTest; | ||||
|  */ | ||||
| final class ActivityReasonAggregatorTest extends AbstractAggregatorTest | ||||
| { | ||||
|     /** | ||||
|      * @var \Chill\ActivityBundle\Export\Aggregator\ActivityReasonAggregator | ||||
|      */ | ||||
|     private $aggregator; | ||||
|     private ActivityReasonAggregator $aggregator; | ||||
|  | ||||
|     protected function setUp(): void | ||||
|     { | ||||
|   | ||||
| @@ -11,6 +11,7 @@ declare(strict_types=1); | ||||
|  | ||||
| namespace Chill\ActivityBundle\Tests\Export\Aggregator; | ||||
|  | ||||
| use Chill\ActivityBundle\Export\Aggregator\PersonAggregators\ActivityTypeAggregator; | ||||
| use Chill\MainBundle\Test\Export\AbstractAggregatorTest; | ||||
|  | ||||
| /** | ||||
| @@ -21,10 +22,7 @@ use Chill\MainBundle\Test\Export\AbstractAggregatorTest; | ||||
|  */ | ||||
| final class ActivityTypeAggregatorTest extends AbstractAggregatorTest | ||||
| { | ||||
|     /** | ||||
|      * @var \Chill\ActivityBundle\Export\Aggregator\ActivityReasonAggregator | ||||
|      */ | ||||
|     private $aggregator; | ||||
|     private ActivityTypeAggregator $aggregator; | ||||
|  | ||||
|     protected function setUp(): void | ||||
|     { | ||||
|   | ||||
| @@ -11,6 +11,7 @@ declare(strict_types=1); | ||||
|  | ||||
| namespace Chill\ActivityBundle\Tests\Export\Aggregator; | ||||
|  | ||||
| use Chill\ActivityBundle\Export\Aggregator\ActivityUserAggregator; | ||||
| use Chill\MainBundle\Test\Export\AbstractAggregatorTest; | ||||
|  | ||||
| /** | ||||
| @@ -21,10 +22,7 @@ use Chill\MainBundle\Test\Export\AbstractAggregatorTest; | ||||
|  */ | ||||
| final class ActivityUserAggregatorTest extends AbstractAggregatorTest | ||||
| { | ||||
|     /** | ||||
|      * @var \Chill\ActivityBundle\Export\Aggregator\ActivityUserAggregator | ||||
|      */ | ||||
|     private $aggregator; | ||||
|     private ActivityUserAggregator $aggregator; | ||||
|  | ||||
|     protected function setUp(): void | ||||
|     { | ||||
|   | ||||
| @@ -0,0 +1,75 @@ | ||||
| <?php | ||||
|  | ||||
| /** | ||||
|  * Chill is a software for social workers. | ||||
|  * | ||||
|  * For the full copyright and license information, please view | ||||
|  * the LICENSE file that was distributed with this source code. | ||||
|  */ | ||||
|  | ||||
| declare(strict_types=1); | ||||
|  | ||||
| /* | ||||
|  * Chill is a software for social workers | ||||
|  * | ||||
|  * For the full copyright and license information, please view | ||||
|  * the LICENSE file that was distributed with this source code. | ||||
|  */ | ||||
|  | ||||
| namespace Chill\ActivityBundle\Tests\Export\Aggregator\PersonAggregators; | ||||
|  | ||||
| use Chill\ActivityBundle\Entity\Activity; | ||||
| use Chill\ActivityBundle\Export\Aggregator\PersonAggregators\ActivityReasonAggregator; | ||||
| use Chill\MainBundle\Test\Export\AbstractAggregatorTest; | ||||
| use Doctrine\ORM\EntityManagerInterface; | ||||
|  | ||||
| /** | ||||
|  * @internal | ||||
|  * @coversNothing | ||||
|  */ | ||||
| final class ActivityReasonAggregatorTest extends AbstractAggregatorTest | ||||
| { | ||||
|     private ActivityReasonAggregator $aggregator; | ||||
|  | ||||
|     protected function setUp(): void | ||||
|     { | ||||
|         self::bootKernel(); | ||||
|  | ||||
|         $this->aggregator = self::$container->get('chill.activity.export.reason_aggregator'); | ||||
|     } | ||||
|  | ||||
|     public function getAggregator() | ||||
|     { | ||||
|         return $this->aggregator; | ||||
|     } | ||||
|  | ||||
|     public function getFormData(): array | ||||
|     { | ||||
|         return [ | ||||
|             [ | ||||
|                 'level' => 'reasons', | ||||
|             ], | ||||
|             [ | ||||
|                 'level' => 'categories', | ||||
|             ], | ||||
|         ]; | ||||
|     } | ||||
|  | ||||
|     public function getQueryBuilders(): array | ||||
|     { | ||||
|         if (null === self::$kernel) { | ||||
|             self::bootKernel(); | ||||
|         } | ||||
|  | ||||
|         $em = self::$container->get(EntityManagerInterface::class); | ||||
|  | ||||
|         return [ | ||||
|             $em->createQueryBuilder() | ||||
|                 ->select('count(activity.id)') | ||||
|                 ->from(Activity::class, 'activity') | ||||
|                 ->join('activity.person', 'actperson') | ||||
|                 ->innerJoin('activity.reasons', 'actreasons') | ||||
|                 ->join('actreasons.category', 'actreasoncat'), | ||||
|         ]; | ||||
|     } | ||||
| } | ||||
| @@ -0,0 +1,92 @@ | ||||
| <?php | ||||
|  | ||||
| /** | ||||
|  * Chill is a software for social workers. | ||||
|  * | ||||
|  * For the full copyright and license information, please view | ||||
|  * the LICENSE file that was distributed with this source code. | ||||
|  */ | ||||
|  | ||||
| declare(strict_types=1); | ||||
|  | ||||
| /* | ||||
|  * Chill is a software for social workers | ||||
|  * | ||||
|  * For the full copyright and license information, please view | ||||
|  * the LICENSE file that was distributed with this source code. | ||||
|  */ | ||||
|  | ||||
| namespace Chill\ActivityBundle\Tests\Export\Filter\ACPFilters; | ||||
|  | ||||
| use Chill\ActivityBundle\Entity\Activity; | ||||
| use Chill\ActivityBundle\Entity\ActivityType; | ||||
| use Chill\ActivityBundle\Export\Filter\ACPFilters\ActivityTypeFilter; | ||||
| use Chill\MainBundle\Test\Export\AbstractFilterTest; | ||||
| use Chill\PersonBundle\Entity\AccompanyingPeriod; | ||||
| use Doctrine\ORM\EntityManagerInterface; | ||||
| use Doctrine\ORM\Query\Expr; | ||||
|  | ||||
| /** | ||||
|  * @internal | ||||
|  * @coversNothing | ||||
|  */ | ||||
| final class ActivityTypeFilterTest extends AbstractFilterTest | ||||
| { | ||||
|     private ActivityTypeFilter $filter; | ||||
|  | ||||
|     protected function setUp(): void | ||||
|     { | ||||
|         self::bootKernel(); | ||||
|  | ||||
|         // add a fake request with a default locale (used in translatable string) | ||||
|         $request = $this->prophesize(); | ||||
|  | ||||
|         $request->willExtend(\Symfony\Component\HttpFoundation\Request::class); | ||||
|         $request->getLocale()->willReturn('fr'); | ||||
|  | ||||
|         $this->filter = self::$container->get('chill.activity.export.filter_activitytype'); | ||||
|     } | ||||
|  | ||||
|     public function getFilter() | ||||
|     { | ||||
|         return $this->filter; | ||||
|     } | ||||
|  | ||||
|     public function getFormData(): array | ||||
|     { | ||||
|         $em = self::$container->get(EntityManagerInterface::class); | ||||
|  | ||||
|         $array = $em->createQueryBuilder() | ||||
|             ->from(ActivityType::class, 'at') | ||||
|             ->select('at') | ||||
|             ->getQuery() | ||||
|             ->getResult(); | ||||
|  | ||||
|         $data = []; | ||||
|  | ||||
|         foreach ($array as $a) { | ||||
|             $data[] = [ | ||||
|                 'accepted_activitytypes' => $a, | ||||
|             ]; | ||||
|         } | ||||
|  | ||||
|         return $data; | ||||
|     } | ||||
|  | ||||
|     public function getQueryBuilders(): array | ||||
|     { | ||||
|         if (null === self::$kernel) { | ||||
|             self::bootKernel(); | ||||
|         } | ||||
|  | ||||
|         $em = self::$container->get(EntityManagerInterface::class); | ||||
|  | ||||
|         return [ | ||||
|             $em->createQueryBuilder() | ||||
|                 ->select('count(activity.id)') | ||||
|                 ->from(AccompanyingPeriod::class, 'acp') | ||||
|                 ->join(Activity::class, 'activity', Expr\Join::WITH, 'activity.accompanyingPeriod = acp') | ||||
|                 ->join('activity.activityType', 'acttype'), | ||||
|         ]; | ||||
|     } | ||||
| } | ||||
| @@ -0,0 +1,89 @@ | ||||
| <?php | ||||
|  | ||||
| /** | ||||
|  * Chill is a software for social workers. | ||||
|  * | ||||
|  * For the full copyright and license information, please view | ||||
|  * the LICENSE file that was distributed with this source code. | ||||
|  */ | ||||
|  | ||||
| declare(strict_types=1); | ||||
|  | ||||
| /* | ||||
|  * Chill is a software for social workers | ||||
|  * | ||||
|  * For the full copyright and license information, please view | ||||
|  * the LICENSE file that was distributed with this source code. | ||||
|  */ | ||||
|  | ||||
| namespace Chill\ActivityBundle\Tests\Export\Filter\ACPFilters; | ||||
|  | ||||
| use Chill\ActivityBundle\Entity\Activity; | ||||
| use Chill\ActivityBundle\Export\Filter\ACPFilters\BySocialActionFilter; | ||||
| use Chill\MainBundle\Test\Export\AbstractFilterTest; | ||||
| use Chill\PersonBundle\Entity\SocialWork\SocialAction; | ||||
| use Doctrine\ORM\EntityManagerInterface; | ||||
|  | ||||
| /** | ||||
|  * @internal | ||||
|  * @coversNothing | ||||
|  */ | ||||
| final class BySocialActionFilterTest extends AbstractFilterTest | ||||
| { | ||||
|     private BySocialActionFilter $filter; | ||||
|  | ||||
|     protected function setUp(): void | ||||
|     { | ||||
|         self::bootKernel(); | ||||
|  | ||||
|         // add a fake request with a default locale (used in translatable string) | ||||
|         $request = $this->prophesize(); | ||||
|  | ||||
|         $request->willExtend(\Symfony\Component\HttpFoundation\Request::class); | ||||
|         $request->getLocale()->willReturn('fr'); | ||||
|  | ||||
|         $this->filter = self::$container->get('chill.activity.export.bysocialaction_filter'); | ||||
|     } | ||||
|  | ||||
|     public function getFilter() | ||||
|     { | ||||
|         return $this->filter; | ||||
|     } | ||||
|  | ||||
|     public function getFormData(): array | ||||
|     { | ||||
|         $em = self::$container->get(EntityManagerInterface::class); | ||||
|  | ||||
|         $array = $em->createQueryBuilder() | ||||
|             ->from(SocialAction::class, 'sa') | ||||
|             ->select('sa') | ||||
|             ->getQuery() | ||||
|             ->getResult(); | ||||
|  | ||||
|         $data = []; | ||||
|  | ||||
|         foreach ($array as $a) { | ||||
|             $data[] = [ | ||||
|                 'accepted_socialactions' => $a, | ||||
|             ]; | ||||
|         } | ||||
|  | ||||
|         return $data; | ||||
|     } | ||||
|  | ||||
|     public function getQueryBuilders(): array | ||||
|     { | ||||
|         if (null === self::$kernel) { | ||||
|             self::bootKernel(); | ||||
|         } | ||||
|  | ||||
|         $em = self::$container->get(EntityManagerInterface::class); | ||||
|  | ||||
|         return [ | ||||
|             $em->createQueryBuilder() | ||||
|                 ->select('count(activity.id)') | ||||
|                 ->from(Activity::class, 'activity') | ||||
|                 ->join('activity.socialActions', 'actsocialaction'), | ||||
|         ]; | ||||
|     } | ||||
| } | ||||
| @@ -0,0 +1,89 @@ | ||||
| <?php | ||||
|  | ||||
| /** | ||||
|  * Chill is a software for social workers. | ||||
|  * | ||||
|  * For the full copyright and license information, please view | ||||
|  * the LICENSE file that was distributed with this source code. | ||||
|  */ | ||||
|  | ||||
| declare(strict_types=1); | ||||
|  | ||||
| /* | ||||
|  * Chill is a software for social workers | ||||
|  * | ||||
|  * For the full copyright and license information, please view | ||||
|  * the LICENSE file that was distributed with this source code. | ||||
|  */ | ||||
|  | ||||
| namespace Chill\ActivityBundle\Tests\Export\Filter\ACPFilters; | ||||
|  | ||||
| use Chill\ActivityBundle\Entity\Activity; | ||||
| use Chill\ActivityBundle\Export\Filter\ACPFilters\BySocialIssueFilter; | ||||
| use Chill\MainBundle\Test\Export\AbstractFilterTest; | ||||
| use Chill\PersonBundle\Entity\SocialWork\SocialIssue; | ||||
| use Doctrine\ORM\EntityManagerInterface; | ||||
|  | ||||
| /** | ||||
|  * @internal | ||||
|  * @coversNothing | ||||
|  */ | ||||
| final class BySocialIssueFilterTest extends AbstractFilterTest | ||||
| { | ||||
|     private BySocialIssueFilter $filter; | ||||
|  | ||||
|     protected function setUp(): void | ||||
|     { | ||||
|         self::bootKernel(); | ||||
|  | ||||
|         // add a fake request with a default locale (used in translatable string) | ||||
|         $request = $this->prophesize(); | ||||
|  | ||||
|         $request->willExtend(\Symfony\Component\HttpFoundation\Request::class); | ||||
|         $request->getLocale()->willReturn('fr'); | ||||
|  | ||||
|         $this->filter = self::$container->get('chill.activity.export.bysocialissue_filter'); | ||||
|     } | ||||
|  | ||||
|     public function getFilter() | ||||
|     { | ||||
|         return $this->filter; | ||||
|     } | ||||
|  | ||||
|     public function getFormData(): array | ||||
|     { | ||||
|         $em = self::$container->get(EntityManagerInterface::class); | ||||
|  | ||||
|         $array = $em->createQueryBuilder() | ||||
|             ->from(SocialIssue::class, 'si') | ||||
|             ->select('si') | ||||
|             ->getQuery() | ||||
|             ->getResult(); | ||||
|  | ||||
|         $data = []; | ||||
|  | ||||
|         foreach ($array as $a) { | ||||
|             $data[] = [ | ||||
|                 'accepted_socialissues' => $a, | ||||
|             ]; | ||||
|         } | ||||
|  | ||||
|         return $data; | ||||
|     } | ||||
|  | ||||
|     public function getQueryBuilders(): array | ||||
|     { | ||||
|         if (null === self::$kernel) { | ||||
|             self::bootKernel(); | ||||
|         } | ||||
|  | ||||
|         $em = self::$container->get(EntityManagerInterface::class); | ||||
|  | ||||
|         return [ | ||||
|             $em->createQueryBuilder() | ||||
|                 ->select('count(activity.id)') | ||||
|                 ->from(Activity::class, 'activity') | ||||
|                 ->join('activity.socialIssues', 'actsocialissue'), | ||||
|         ]; | ||||
|     } | ||||
| } | ||||
| @@ -0,0 +1,89 @@ | ||||
| <?php | ||||
|  | ||||
| /** | ||||
|  * Chill is a software for social workers. | ||||
|  * | ||||
|  * For the full copyright and license information, please view | ||||
|  * the LICENSE file that was distributed with this source code. | ||||
|  */ | ||||
|  | ||||
| declare(strict_types=1); | ||||
|  | ||||
| /* | ||||
|  * Chill is a software for social workers | ||||
|  * | ||||
|  * For the full copyright and license information, please view | ||||
|  * the LICENSE file that was distributed with this source code. | ||||
|  */ | ||||
|  | ||||
| namespace Chill\ActivityBundle\Tests\Export\Filter\ACPFilters; | ||||
|  | ||||
| use Chill\ActivityBundle\Entity\Activity; | ||||
| use Chill\ActivityBundle\Export\Filter\ACPFilters\ByUserFilter; | ||||
| use Chill\MainBundle\Entity\User; | ||||
| use Chill\MainBundle\Test\Export\AbstractFilterTest; | ||||
| use Doctrine\ORM\EntityManagerInterface; | ||||
|  | ||||
| /** | ||||
|  * @internal | ||||
|  * @coversNothing | ||||
|  */ | ||||
| final class ByUserFilterTest extends AbstractFilterTest | ||||
| { | ||||
|     private ByUserFilter $filter; | ||||
|  | ||||
|     protected function setUp(): void | ||||
|     { | ||||
|         self::bootKernel(); | ||||
|  | ||||
|         // add a fake request with a default locale (used in translatable string) | ||||
|         $request = $this->prophesize(); | ||||
|  | ||||
|         $request->willExtend(\Symfony\Component\HttpFoundation\Request::class); | ||||
|         $request->getLocale()->willReturn('fr'); | ||||
|  | ||||
|         $this->filter = self::$container->get('chill.activity.export.byuser_filter'); | ||||
|     } | ||||
|  | ||||
|     public function getFilter() | ||||
|     { | ||||
|         return $this->filter; | ||||
|     } | ||||
|  | ||||
|     public function getFormData(): array | ||||
|     { | ||||
|         $em = self::$container->get(EntityManagerInterface::class); | ||||
|  | ||||
|         $array = $em->createQueryBuilder() | ||||
|             ->from(User::class, 'u') | ||||
|             ->select('u') | ||||
|             ->getQuery() | ||||
|             ->getResult(); | ||||
|  | ||||
|         $data = []; | ||||
|  | ||||
|         foreach ($array as $a) { | ||||
|             $data[] = [ | ||||
|                 'accepted_users' => $a, | ||||
|             ]; | ||||
|         } | ||||
|  | ||||
|         return $data; | ||||
|     } | ||||
|  | ||||
|     public function getQueryBuilders(): array | ||||
|     { | ||||
|         if (null === self::$kernel) { | ||||
|             self::bootKernel(); | ||||
|         } | ||||
|  | ||||
|         $em = self::$container->get(EntityManagerInterface::class); | ||||
|  | ||||
|         return [ | ||||
|             $em->createQueryBuilder() | ||||
|                 ->select('count(activity.id)') | ||||
|                 ->from(Activity::class, 'activity') | ||||
|                 ->join('activity.users', 'actusers'), | ||||
|         ]; | ||||
|     } | ||||
| } | ||||
| @@ -0,0 +1,74 @@ | ||||
| <?php | ||||
|  | ||||
| /** | ||||
|  * Chill is a software for social workers. | ||||
|  * | ||||
|  * For the full copyright and license information, please view | ||||
|  * the LICENSE file that was distributed with this source code. | ||||
|  */ | ||||
|  | ||||
| declare(strict_types=1); | ||||
|  | ||||
| /* | ||||
|  * Chill is a software for social workers | ||||
|  * | ||||
|  * For the full copyright and license information, please view | ||||
|  * the LICENSE file that was distributed with this source code. | ||||
|  */ | ||||
|  | ||||
| namespace Chill\ActivityBundle\Tests\Export\Filter\ACPFilters; | ||||
|  | ||||
| use Chill\ActivityBundle\Entity\Activity; | ||||
| use Chill\ActivityBundle\Export\Filter\ACPFilters\EmergencyFilter; | ||||
| use Chill\MainBundle\Test\Export\AbstractFilterTest; | ||||
| use Doctrine\ORM\EntityManagerInterface; | ||||
|  | ||||
| /** | ||||
|  * @internal | ||||
|  * @coversNothing | ||||
|  */ | ||||
| final class EmergencyFilterTest extends AbstractFilterTest | ||||
| { | ||||
|     private EmergencyFilter $filter; | ||||
|  | ||||
|     protected function setUp(): void | ||||
|     { | ||||
|         self::bootKernel(); | ||||
|  | ||||
|         // add a fake request with a default locale (used in translatable string) | ||||
|         $request = $this->prophesize(); | ||||
|  | ||||
|         $request->willExtend(\Symfony\Component\HttpFoundation\Request::class); | ||||
|         $request->getLocale()->willReturn('fr'); | ||||
|  | ||||
|         $this->filter = self::$container->get('chill.activity.export.emergency_filter'); | ||||
|     } | ||||
|  | ||||
|     public function getFilter() | ||||
|     { | ||||
|         return $this->filter; | ||||
|     } | ||||
|  | ||||
|     public function getFormData(): array | ||||
|     { | ||||
|         return [ | ||||
|             ['accepted_emergency' => true], | ||||
|             ['accepted_emergency' => false], | ||||
|         ]; | ||||
|     } | ||||
|  | ||||
|     public function getQueryBuilders(): array | ||||
|     { | ||||
|         if (null === self::$kernel) { | ||||
|             self::bootKernel(); | ||||
|         } | ||||
|  | ||||
|         $em = self::$container->get(EntityManagerInterface::class); | ||||
|  | ||||
|         return [ | ||||
|             $em->createQueryBuilder() | ||||
|                 ->select('count(activity.id)') | ||||
|                 ->from(Activity::class, 'activity'), | ||||
|         ]; | ||||
|     } | ||||
| } | ||||
| @@ -0,0 +1,89 @@ | ||||
| <?php | ||||
|  | ||||
| /** | ||||
|  * Chill is a software for social workers. | ||||
|  * | ||||
|  * For the full copyright and license information, please view | ||||
|  * the LICENSE file that was distributed with this source code. | ||||
|  */ | ||||
|  | ||||
| declare(strict_types=1); | ||||
|  | ||||
| /* | ||||
|  * Chill is a software for social workers | ||||
|  * | ||||
|  * For the full copyright and license information, please view | ||||
|  * the LICENSE file that was distributed with this source code. | ||||
|  */ | ||||
|  | ||||
| namespace Chill\ActivityBundle\Tests\Export\Filter\ACPFilters; | ||||
|  | ||||
| use Chill\ActivityBundle\Entity\Activity; | ||||
| use Chill\ActivityBundle\Export\Filter\ACPFilters\LocationTypeFilter; | ||||
| use Chill\MainBundle\Entity\LocationType; | ||||
| use Chill\MainBundle\Test\Export\AbstractFilterTest; | ||||
| use Doctrine\ORM\EntityManagerInterface; | ||||
|  | ||||
| /** | ||||
|  * @internal | ||||
|  * @coversNothing | ||||
|  */ | ||||
| final class LocationTypeFilterTest extends AbstractFilterTest | ||||
| { | ||||
|     private LocationTypeFilter $filter; | ||||
|  | ||||
|     protected function setUp(): void | ||||
|     { | ||||
|         self::bootKernel(); | ||||
|  | ||||
|         // add a fake request with a default locale (used in translatable string) | ||||
|         $request = $this->prophesize(); | ||||
|  | ||||
|         $request->willExtend(\Symfony\Component\HttpFoundation\Request::class); | ||||
|         $request->getLocale()->willReturn('fr'); | ||||
|  | ||||
|         $this->filter = self::$container->get('chill.activity.export.locationtype_filter'); | ||||
|     } | ||||
|  | ||||
|     public function getFilter() | ||||
|     { | ||||
|         return $this->filter; | ||||
|     } | ||||
|  | ||||
|     public function getFormData(): array | ||||
|     { | ||||
|         $em = self::$container->get(EntityManagerInterface::class); | ||||
|  | ||||
|         $array = $em->createQueryBuilder() | ||||
|             ->from(LocationType::class, 'lt') | ||||
|             ->select('lt') | ||||
|             ->getQuery() | ||||
|             ->getResult(); | ||||
|  | ||||
|         $data = []; | ||||
|  | ||||
|         foreach ($array as $a) { | ||||
|             $data[] = [ | ||||
|                 'accepted_locationtype' => $a, | ||||
|             ]; | ||||
|         } | ||||
|  | ||||
|         return $data; | ||||
|     } | ||||
|  | ||||
|     public function getQueryBuilders(): array | ||||
|     { | ||||
|         if (null === self::$kernel) { | ||||
|             self::bootKernel(); | ||||
|         } | ||||
|  | ||||
|         $em = self::$container->get(EntityManagerInterface::class); | ||||
|  | ||||
|         return [ | ||||
|             $em->createQueryBuilder() | ||||
|                 ->select('count(activity.id)') | ||||
|                 ->from(Activity::class, 'activity') | ||||
|                 ->join('activity.location', 'actloc'), | ||||
|         ]; | ||||
|     } | ||||
| } | ||||
| @@ -0,0 +1,74 @@ | ||||
| <?php | ||||
|  | ||||
| /** | ||||
|  * Chill is a software for social workers. | ||||
|  * | ||||
|  * For the full copyright and license information, please view | ||||
|  * the LICENSE file that was distributed with this source code. | ||||
|  */ | ||||
|  | ||||
| declare(strict_types=1); | ||||
|  | ||||
| /* | ||||
|  * Chill is a software for social workers | ||||
|  * | ||||
|  * For the full copyright and license information, please view | ||||
|  * the LICENSE file that was distributed with this source code. | ||||
|  */ | ||||
|  | ||||
| namespace Chill\ActivityBundle\Tests\Export\Filter\ACPFilters; | ||||
|  | ||||
| use Chill\ActivityBundle\Entity\Activity; | ||||
| use Chill\ActivityBundle\Export\Filter\ACPFilters\SentReceivedFilter; | ||||
| use Chill\MainBundle\Test\Export\AbstractFilterTest; | ||||
| use Doctrine\ORM\EntityManagerInterface; | ||||
|  | ||||
| /** | ||||
|  * @internal | ||||
|  * @coversNothing | ||||
|  */ | ||||
| final class SentReceivedFilterTest extends AbstractFilterTest | ||||
| { | ||||
|     private SentReceivedFilter $filter; | ||||
|  | ||||
|     protected function setUp(): void | ||||
|     { | ||||
|         self::bootKernel(); | ||||
|  | ||||
|         // add a fake request with a default locale (used in translatable string) | ||||
|         $request = $this->prophesize(); | ||||
|  | ||||
|         $request->willExtend(\Symfony\Component\HttpFoundation\Request::class); | ||||
|         $request->getLocale()->willReturn('fr'); | ||||
|  | ||||
|         $this->filter = self::$container->get('chill.activity.export.sentreceived_filter'); | ||||
|     } | ||||
|  | ||||
|     public function getFilter() | ||||
|     { | ||||
|         return $this->filter; | ||||
|     } | ||||
|  | ||||
|     public function getFormData(): array | ||||
|     { | ||||
|         return [ | ||||
|             ['accepted_sentreceived' => Activity::SENTRECEIVED_SENT], | ||||
|             ['accepted_sentreceived' => Activity::SENTRECEIVED_RECEIVED], | ||||
|         ]; | ||||
|     } | ||||
|  | ||||
|     public function getQueryBuilders(): array | ||||
|     { | ||||
|         if (null === self::$kernel) { | ||||
|             self::bootKernel(); | ||||
|         } | ||||
|  | ||||
|         $em = self::$container->get(EntityManagerInterface::class); | ||||
|  | ||||
|         return [ | ||||
|             $em->createQueryBuilder() | ||||
|                 ->select('count(activity.id)') | ||||
|                 ->from(Activity::class, 'activity'), | ||||
|         ]; | ||||
|     } | ||||
| } | ||||
| @@ -0,0 +1,88 @@ | ||||
| <?php | ||||
|  | ||||
| /** | ||||
|  * Chill is a software for social workers. | ||||
|  * | ||||
|  * For the full copyright and license information, please view | ||||
|  * the LICENSE file that was distributed with this source code. | ||||
|  */ | ||||
|  | ||||
| declare(strict_types=1); | ||||
|  | ||||
| /* | ||||
|  * Chill is a software for social workers | ||||
|  * | ||||
|  * For the full copyright and license information, please view | ||||
|  * the LICENSE file that was distributed with this source code. | ||||
|  */ | ||||
|  | ||||
| namespace Chill\ActivityBundle\Tests\Export\Filter\ACPFilters; | ||||
|  | ||||
| use Chill\ActivityBundle\Entity\Activity; | ||||
| use Chill\ActivityBundle\Export\Filter\ACPFilters\UserFilter; | ||||
| use Chill\MainBundle\Entity\User; | ||||
| use Chill\MainBundle\Test\Export\AbstractFilterTest; | ||||
| use Doctrine\ORM\EntityManagerInterface; | ||||
|  | ||||
| /** | ||||
|  * @internal | ||||
|  * @coversNothing | ||||
|  */ | ||||
| final class UserFilterTest extends AbstractFilterTest | ||||
| { | ||||
|     private UserFilter $filter; | ||||
|  | ||||
|     protected function setUp(): void | ||||
|     { | ||||
|         self::bootKernel(); | ||||
|  | ||||
|         // add a fake request with a default locale (used in translatable string) | ||||
|         $request = $this->prophesize(); | ||||
|  | ||||
|         $request->willExtend(\Symfony\Component\HttpFoundation\Request::class); | ||||
|         $request->getLocale()->willReturn('fr'); | ||||
|  | ||||
|         $this->filter = self::$container->get('chill.activity.export.user_filter'); | ||||
|     } | ||||
|  | ||||
|     public function getFilter() | ||||
|     { | ||||
|         return $this->filter; | ||||
|     } | ||||
|  | ||||
|     public function getFormData(): array | ||||
|     { | ||||
|         $em = self::$container->get(EntityManagerInterface::class); | ||||
|  | ||||
|         $array = $em->createQueryBuilder() | ||||
|             ->from(User::class, 'u') | ||||
|             ->select('u') | ||||
|             ->getQuery() | ||||
|             ->getResult(); | ||||
|  | ||||
|         $data = []; | ||||
|  | ||||
|         foreach ($array as $a) { | ||||
|             $data[] = [ | ||||
|                 'accepted_users' => $a, | ||||
|             ]; | ||||
|         } | ||||
|  | ||||
|         return $data; | ||||
|     } | ||||
|  | ||||
|     public function getQueryBuilders(): array | ||||
|     { | ||||
|         if (null === self::$kernel) { | ||||
|             self::bootKernel(); | ||||
|         } | ||||
|  | ||||
|         $em = self::$container->get(EntityManagerInterface::class); | ||||
|  | ||||
|         return [ | ||||
|             $em->createQueryBuilder() | ||||
|                 ->select('count(activity.id)') | ||||
|                 ->from(Activity::class, 'activity'), | ||||
|         ]; | ||||
|     } | ||||
| } | ||||
| @@ -0,0 +1,89 @@ | ||||
| <?php | ||||
|  | ||||
| /** | ||||
|  * Chill is a software for social workers. | ||||
|  * | ||||
|  * For the full copyright and license information, please view | ||||
|  * the LICENSE file that was distributed with this source code. | ||||
|  */ | ||||
|  | ||||
| declare(strict_types=1); | ||||
|  | ||||
| /* | ||||
|  * Chill is a software for social workers | ||||
|  * | ||||
|  * For the full copyright and license information, please view | ||||
|  * the LICENSE file that was distributed with this source code. | ||||
|  */ | ||||
|  | ||||
| namespace Chill\ActivityBundle\Tests\Export\Filter\ACPFilters; | ||||
|  | ||||
| use Chill\ActivityBundle\Entity\Activity; | ||||
| use Chill\ActivityBundle\Export\Filter\ACPFilters\UserScopeFilter; | ||||
| use Chill\MainBundle\Entity\Scope; | ||||
| use Chill\MainBundle\Test\Export\AbstractFilterTest; | ||||
| use Doctrine\ORM\EntityManagerInterface; | ||||
|  | ||||
| /** | ||||
|  * @internal | ||||
|  * @coversNothing | ||||
|  */ | ||||
| final class UserScopeFilterTest extends AbstractFilterTest | ||||
| { | ||||
|     private UserScopeFilter $filter; | ||||
|  | ||||
|     protected function setUp(): void | ||||
|     { | ||||
|         self::bootKernel(); | ||||
|  | ||||
|         // add a fake request with a default locale (used in translatable string) | ||||
|         $request = $this->prophesize(); | ||||
|  | ||||
|         $request->willExtend(\Symfony\Component\HttpFoundation\Request::class); | ||||
|         $request->getLocale()->willReturn('fr'); | ||||
|  | ||||
|         $this->filter = self::$container->get('chill.activity.export.userscope_filter'); | ||||
|     } | ||||
|  | ||||
|     public function getFilter() | ||||
|     { | ||||
|         return $this->filter; | ||||
|     } | ||||
|  | ||||
|     public function getFormData(): array | ||||
|     { | ||||
|         $em = self::$container->get(EntityManagerInterface::class); | ||||
|  | ||||
|         $array = $em->createQueryBuilder() | ||||
|             ->from(Scope::class, 's') | ||||
|             ->select('s') | ||||
|             ->getQuery() | ||||
|             ->getResult(); | ||||
|  | ||||
|         $data = []; | ||||
|  | ||||
|         foreach ($array as $a) { | ||||
|             $data[] = [ | ||||
|                 'accepted_userscope' => $a, | ||||
|             ]; | ||||
|         } | ||||
|  | ||||
|         return $data; | ||||
|     } | ||||
|  | ||||
|     public function getQueryBuilders(): array | ||||
|     { | ||||
|         if (null === self::$kernel) { | ||||
|             self::bootKernel(); | ||||
|         } | ||||
|  | ||||
|         $em = self::$container->get(EntityManagerInterface::class); | ||||
|  | ||||
|         return [ | ||||
|             $em->createQueryBuilder() | ||||
|                 ->select('count(activity.id)') | ||||
|                 ->from(Activity::class, 'activity') | ||||
|                 ->join('activity.user', 'actuser'), | ||||
|         ]; | ||||
|     } | ||||
| } | ||||
| @@ -0,0 +1,77 @@ | ||||
| <?php | ||||
|  | ||||
| /** | ||||
|  * Chill is a software for social workers. | ||||
|  * | ||||
|  * For the full copyright and license information, please view | ||||
|  * the LICENSE file that was distributed with this source code. | ||||
|  */ | ||||
|  | ||||
| declare(strict_types=1); | ||||
|  | ||||
| /* | ||||
|  * Chill is a software for social workers | ||||
|  * | ||||
|  * For the full copyright and license information, please view | ||||
|  * the LICENSE file that was distributed with this source code. | ||||
|  */ | ||||
|  | ||||
| namespace Chill\ActivityBundle\Tests\Export\Filter; | ||||
|  | ||||
| use Chill\ActivityBundle\Entity\Activity; | ||||
| use Chill\ActivityBundle\Export\Filter\ActivityDateFilter; | ||||
| use Chill\MainBundle\Test\Export\AbstractFilterTest; | ||||
| use DateTime; | ||||
| use Doctrine\ORM\EntityManagerInterface; | ||||
|  | ||||
| /** | ||||
|  * @internal | ||||
|  * @coversNothing | ||||
|  */ | ||||
| final class ActivityDateFilterTest extends AbstractFilterTest | ||||
| { | ||||
|     private ActivityDateFilter $filter; | ||||
|  | ||||
|     protected function setUp(): void | ||||
|     { | ||||
|         self::bootKernel(); | ||||
|  | ||||
|         // add a fake request with a default locale (used in translatable string) | ||||
|         $request = $this->prophesize(); | ||||
|  | ||||
|         $request->willExtend(\Symfony\Component\HttpFoundation\Request::class); | ||||
|         $request->getLocale()->willReturn('fr'); | ||||
|  | ||||
|         $this->filter = self::$container->get('chill.activity.export.date_filter'); | ||||
|     } | ||||
|  | ||||
|     public function getFilter() | ||||
|     { | ||||
|         return $this->filter; | ||||
|     } | ||||
|  | ||||
|     public function getFormData(): array | ||||
|     { | ||||
|         return [ | ||||
|             [ | ||||
|                 'date_from' => DateTime::createFromFormat('Y-m-d', '2020-01-01'), | ||||
|                 'date_to' => DateTime::createFromFormat('Y-m-d', '2021-01-01'), | ||||
|             ], | ||||
|         ]; | ||||
|     } | ||||
|  | ||||
|     public function getQueryBuilders(): array | ||||
|     { | ||||
|         if (null === self::$kernel) { | ||||
|             self::bootKernel(); | ||||
|         } | ||||
|  | ||||
|         $em = self::$container->get(EntityManagerInterface::class); | ||||
|  | ||||
|         return [ | ||||
|             $em->createQueryBuilder() | ||||
|                 ->select('count(activity.id)') | ||||
|                 ->from(Activity::class, 'activity'), | ||||
|         ]; | ||||
|     } | ||||
| } | ||||
| @@ -11,6 +11,7 @@ declare(strict_types=1); | ||||
|  | ||||
| namespace Chill\ActivityBundle\Tests\Export\Filter; | ||||
|  | ||||
| use Chill\ActivityBundle\Export\Filter\PersonFilters\ActivityReasonFilter; | ||||
| use Chill\MainBundle\Test\Export\AbstractFilterTest; | ||||
| use Doctrine\Common\Collections\ArrayCollection; | ||||
|  | ||||
| @@ -20,10 +21,7 @@ use Doctrine\Common\Collections\ArrayCollection; | ||||
|  */ | ||||
| final class ActivityReasonFilterTest extends AbstractFilterTest | ||||
| { | ||||
|     /** | ||||
|      * @var \Chill\PersonBundle\Export\Filter\GenderFilter | ||||
|      */ | ||||
|     private $filter; | ||||
|     private ActivityReasonFilter $filter; | ||||
|  | ||||
|     protected function setUp(): void | ||||
|     { | ||||
|   | ||||
| @@ -9,13 +9,13 @@ declare(strict_types=1); | ||||
|  * the LICENSE file that was distributed with this source code. | ||||
|  */ | ||||
| 
 | ||||
| namespace Export\Filter; | ||||
| namespace Chill\ActivityBundle\Tests\Export\Filter; | ||||
| 
 | ||||
| use Chill\ActivityBundle\Entity\Activity; | ||||
| use Chill\ActivityBundle\Entity\ActivityType; | ||||
| use Chill\ActivityBundle\Export\Filter\ActivityTypeFilter; | ||||
| use Chill\MainBundle\Test\Export\AbstractFilterTest; | ||||
| use Chill\PersonBundle\Export\Filter\ActivityTypeFilter; | ||||
| use Doctrine\ORM\EntityManagerInterface; | ||||
| use Symfony\Component\HttpFoundation\Request; | ||||
| 
 | ||||
| /** | ||||
|  * @internal | ||||
| @@ -27,16 +27,15 @@ final class ActivityTypeFilterTest extends AbstractFilterTest | ||||
| 
 | ||||
|     protected function setUp(): void | ||||
|     { | ||||
|         //parent::setUp();
 | ||||
|         self::bootKernel(); | ||||
| 
 | ||||
|         // add a fake request with a default locale (used in translatable string)
 | ||||
|         $request = $this->prophesize(); | ||||
| 
 | ||||
|         $request->willExtend(Request::class); | ||||
|         $request->willExtend(\Symfony\Component\HttpFoundation\Request::class); | ||||
|         $request->getLocale()->willReturn('fr'); | ||||
| 
 | ||||
|         $this->filter = self::$container->get('chill.person.export.filter_activitytype'); | ||||
|         $this->filter = self::$container->get('chill.activity.export.type_filter'); | ||||
|     } | ||||
| 
 | ||||
|     public function getFilter() | ||||
| @@ -56,8 +55,10 @@ final class ActivityTypeFilterTest extends AbstractFilterTest | ||||
| 
 | ||||
|         $data = []; | ||||
| 
 | ||||
|         foreach ($array as $t) { | ||||
|             $data[] = ['accepted_activitytypes' => $t]; | ||||
|         foreach ($array as $a) { | ||||
|             $data[] = [ | ||||
|                 'types' => $a, | ||||
|             ]; | ||||
|         } | ||||
| 
 | ||||
|         return $data; | ||||
| @@ -73,8 +74,8 @@ final class ActivityTypeFilterTest extends AbstractFilterTest | ||||
| 
 | ||||
|         return [ | ||||
|             $em->createQueryBuilder() | ||||
|                 ->from('ChillPersonBundle:AccompanyingPeriod', 'acp') | ||||
|                 ->select('acp.id'), | ||||
|                 ->select('count(activity.id)') | ||||
|                 ->from(Activity::class, 'activity'), | ||||
|         ]; | ||||
|     } | ||||
| } | ||||
| @@ -0,0 +1,89 @@ | ||||
| <?php | ||||
|  | ||||
| /** | ||||
|  * Chill is a software for social workers. | ||||
|  * | ||||
|  * For the full copyright and license information, please view | ||||
|  * the LICENSE file that was distributed with this source code. | ||||
|  */ | ||||
|  | ||||
| declare(strict_types=1); | ||||
|  | ||||
| /* | ||||
|  * Chill is a software for social workers | ||||
|  * | ||||
|  * For the full copyright and license information, please view | ||||
|  * the LICENSE file that was distributed with this source code. | ||||
|  */ | ||||
|  | ||||
| namespace Chill\ActivityBundle\Tests\Export\Filter\PersonFilters; | ||||
|  | ||||
| use Chill\ActivityBundle\Entity\Activity; | ||||
| use Chill\ActivityBundle\Entity\ActivityReason; | ||||
| use Chill\ActivityBundle\Export\Filter\PersonFilters\ActivityReasonFilter; | ||||
| use Chill\MainBundle\Test\Export\AbstractFilterTest; | ||||
| use Doctrine\ORM\EntityManagerInterface; | ||||
|  | ||||
| /** | ||||
|  * @internal | ||||
|  * @coversNothing | ||||
|  */ | ||||
| final class ActivityReasonFilterTest extends AbstractFilterTest | ||||
| { | ||||
|     private ActivityReasonFilter $filter; | ||||
|  | ||||
|     protected function setUp(): void | ||||
|     { | ||||
|         self::bootKernel(); | ||||
|  | ||||
|         // add a fake request with a default locale (used in translatable string) | ||||
|         $request = $this->prophesize(); | ||||
|  | ||||
|         $request->willExtend(\Symfony\Component\HttpFoundation\Request::class); | ||||
|         $request->getLocale()->willReturn('fr'); | ||||
|  | ||||
|         $this->filter = self::$container->get('chill.activity.export.reason_filter'); | ||||
|     } | ||||
|  | ||||
|     public function getFilter() | ||||
|     { | ||||
|         return $this->filter; | ||||
|     } | ||||
|  | ||||
|     public function getFormData(): array | ||||
|     { | ||||
|         $em = self::$container->get(EntityManagerInterface::class); | ||||
|  | ||||
|         $array = $em->createQueryBuilder() | ||||
|             ->from(ActivityReason::class, 'ar') | ||||
|             ->select('ar') | ||||
|             ->getQuery() | ||||
|             ->getResult(); | ||||
|  | ||||
|         $data = []; | ||||
|  | ||||
|         foreach ($array as $a) { | ||||
|             $data[] = [ | ||||
|                 'reasons' => $a, | ||||
|             ]; | ||||
|         } | ||||
|  | ||||
|         return $data; | ||||
|     } | ||||
|  | ||||
|     public function getQueryBuilders(): array | ||||
|     { | ||||
|         if (null === self::$kernel) { | ||||
|             self::bootKernel(); | ||||
|         } | ||||
|  | ||||
|         $em = self::$container->get(EntityManagerInterface::class); | ||||
|  | ||||
|         return [ | ||||
|             $em->createQueryBuilder() | ||||
|                 ->select('count(activity.id)') | ||||
|                 ->from(Activity::class, 'activity') | ||||
|                 ->join('activity.reasons', 'actreasons'), | ||||
|         ]; | ||||
|     } | ||||
| } | ||||
| @@ -0,0 +1,91 @@ | ||||
| <?php | ||||
|  | ||||
| /** | ||||
|  * Chill is a software for social workers. | ||||
|  * | ||||
|  * For the full copyright and license information, please view | ||||
|  * the LICENSE file that was distributed with this source code. | ||||
|  */ | ||||
|  | ||||
| declare(strict_types=1); | ||||
|  | ||||
| /* | ||||
|  * Chill is a software for social workers | ||||
|  * | ||||
|  * For the full copyright and license information, please view | ||||
|  * the LICENSE file that was distributed with this source code. | ||||
|  */ | ||||
|  | ||||
| namespace Chill\ActivityBundle\Tests\Export\Filter\PersonFilters; | ||||
|  | ||||
| use Chill\ActivityBundle\Entity\Activity; | ||||
| use Chill\ActivityBundle\Entity\ActivityReason; | ||||
| use Chill\ActivityBundle\Export\Filter\PersonFilters\PersonHavingActivityBetweenDateFilter; | ||||
| use Chill\MainBundle\Test\Export\AbstractFilterTest; | ||||
| use DateTime; | ||||
| use Doctrine\ORM\EntityManagerInterface; | ||||
|  | ||||
| /** | ||||
|  * @internal | ||||
|  * @coversNothing | ||||
|  */ | ||||
| final class PersonHavingActivityBetweenDateFilterTest extends AbstractFilterTest | ||||
| { | ||||
|     private PersonHavingActivityBetweenDateFilter $filter; | ||||
|  | ||||
|     protected function setUp(): void | ||||
|     { | ||||
|         self::bootKernel(); | ||||
|  | ||||
|         // add a fake request with a default locale (used in translatable string) | ||||
|         $request = $this->prophesize(); | ||||
|  | ||||
|         $request->willExtend(\Symfony\Component\HttpFoundation\Request::class); | ||||
|         $request->getLocale()->willReturn('fr'); | ||||
|  | ||||
|         $this->filter = self::$container->get('chill.activity.export.person_having_an_activity_between_date_filter'); | ||||
|     } | ||||
|  | ||||
|     public function getFilter() | ||||
|     { | ||||
|         return $this->filter; | ||||
|     } | ||||
|  | ||||
|     public function getFormData(): array | ||||
|     { | ||||
|         $em = self::$container->get(EntityManagerInterface::class); | ||||
|  | ||||
|         $array = $em->createQueryBuilder() | ||||
|             ->from(ActivityReason::class, 'ar') | ||||
|             ->select('ar') | ||||
|             ->getQuery() | ||||
|             ->getResult(); | ||||
|  | ||||
|         $data = []; | ||||
|  | ||||
|         foreach ($array as $a) { | ||||
|             $data[] = [ | ||||
|                 'date_from' => DateTime::createFromFormat('Y-m-d', '2021-07-01'), | ||||
|                 'date_to' => DateTime::createFromFormat('Y-m-d', '2022-07-01'), | ||||
|                 'reasons' => $a, | ||||
|             ]; | ||||
|         } | ||||
|  | ||||
|         return $data; | ||||
|     } | ||||
|  | ||||
|     public function getQueryBuilders(): array | ||||
|     { | ||||
|         if (null === self::$kernel) { | ||||
|             self::bootKernel(); | ||||
|         } | ||||
|  | ||||
|         $em = self::$container->get(EntityManagerInterface::class); | ||||
|  | ||||
|         return [ | ||||
|             $em->createQueryBuilder() | ||||
|                 ->select('count(activity.id)') | ||||
|                 ->from(Activity::class, 'activity'), | ||||
|         ]; | ||||
|     } | ||||
| } | ||||
| @@ -11,6 +11,7 @@ declare(strict_types=1); | ||||
|  | ||||
| namespace Chill\ActivityBundle\Tests\Export\Filter; | ||||
|  | ||||
| use Chill\ActivityBundle\Export\Filter\PersonFilters\PersonHavingActivityBetweenDateFilter; | ||||
| use Chill\MainBundle\Test\Export\AbstractFilterTest; | ||||
| use DateTime; | ||||
| use function array_slice; | ||||
| @@ -21,10 +22,7 @@ use function array_slice; | ||||
|  */ | ||||
| final class PersonHavingActivityBetweenDateFilterTest extends AbstractFilterTest | ||||
| { | ||||
|     /** | ||||
|      * @var \Chill\PersonBundle\Export\Filter\PersonHavingActivityBetweenDateFilter | ||||
|      */ | ||||
|     private $filter; | ||||
|     private PersonHavingActivityBetweenDateFilter $filter; | ||||
|  | ||||
|     protected function setUp(): void | ||||
|     { | ||||
|   | ||||
| @@ -67,6 +67,11 @@ services: | ||||
|                 name: chill.export_filter | ||||
|                 alias: 'activity_person_having_ac_bw_date_filter' | ||||
|  | ||||
|     chill.activity.export.filter_activitytype: | ||||
|         class: Chill\ActivityBundle\Export\Filter\ACPFilters\ActivityTypeFilter | ||||
|         tags: | ||||
|             - { name: chill.export_filter, alias: 'accompanyingcourse_activitytype_filter' } | ||||
|  | ||||
|     chill.activity.export.locationtype_filter: | ||||
|         class: Chill\ActivityBundle\Export\Filter\ACPFilters\LocationTypeFilter | ||||
|         tags: | ||||
|   | ||||
| @@ -311,6 +311,6 @@ This is the minimal activity data: Activité n° | ||||
|  | ||||
| docgen: | ||||
|     Activity basic: Echange | ||||
|     A basic context for activity: Contexte pour les échanges | ||||
|     Accompanying period with a list of activities: Parcours d'accompagnement avec liste des échanges | ||||
|     Accompanying period with a list of activities description: Ce contexte reprend les informations du parcours, et tous les échanges pour un parcours. Les échanges ne sont pas filtrés. | ||||
|     A basic context for activity: Contexte pour les activités | ||||
|     Accompanying period with a list of activities: Parcours d'accompagnement avec liste des activités | ||||
|     Accompanying period with a list of activities description: Ce contexte reprend les informations du parcours, et tous les activités pour un parcours. Les activités ne sont pas filtrés. | ||||
|   | ||||
| @@ -231,4 +231,4 @@ This is the minimal activity data: Activité n° | ||||
|  | ||||
| docgen: | ||||
|     Activity basic: Echange | ||||
|     A basic context for activity: Contexte pour les échanges | ||||
|     A basic context for activity: Contexte pour les activités | ||||
|   | ||||
| @@ -11,8 +11,16 @@ declare(strict_types=1); | ||||
|  | ||||
| namespace Chill\CalendarBundle; | ||||
|  | ||||
| use Chill\CalendarBundle\RemoteCalendar\DependencyInjection\RemoteCalendarCompilerPass; | ||||
| use Symfony\Component\DependencyInjection\ContainerBuilder; | ||||
| use Symfony\Component\HttpKernel\Bundle\Bundle; | ||||
|  | ||||
| class ChillCalendarBundle extends Bundle | ||||
| { | ||||
|     public function build(ContainerBuilder $container) | ||||
|     { | ||||
|         parent::build($container); | ||||
|  | ||||
|         $container->addCompilerPass(new RemoteCalendarCompilerPass()); | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -0,0 +1,84 @@ | ||||
| <?php | ||||
|  | ||||
| /** | ||||
|  * Chill is a software for social workers. | ||||
|  * | ||||
|  * For the full copyright and license information, please view | ||||
|  * the LICENSE file that was distributed with this source code. | ||||
|  */ | ||||
|  | ||||
| declare(strict_types=1); | ||||
|  | ||||
| /* | ||||
|  * Chill is a software for social workers | ||||
|  * | ||||
|  * For the full copyright and license information, please view | ||||
|  * the LICENSE file that was distributed with this source code. | ||||
|  */ | ||||
|  | ||||
| namespace Chill\CalendarBundle\Command; | ||||
|  | ||||
| use Chill\CalendarBundle\RemoteCalendar\Connector\MSGraph\MachineTokenStorage; | ||||
| use KnpU\OAuth2ClientBundle\Client\ClientRegistry; | ||||
| use Symfony\Component\Console\Command\Command; | ||||
| use Symfony\Component\Console\Helper\FormatterHelper; | ||||
| use Symfony\Component\Console\Input\InputInterface; | ||||
| use Symfony\Component\Console\Output\OutputInterface; | ||||
| use Symfony\Component\Console\Question\ConfirmationQuestion; | ||||
| use TheNetworg\OAuth2\Client\Provider\Azure; | ||||
|  | ||||
| class AzureGrantAdminConsentAndAcquireToken extends Command | ||||
| { | ||||
|     private Azure $azure; | ||||
|  | ||||
|     private ClientRegistry $clientRegistry; | ||||
|  | ||||
|     private MachineTokenStorage $machineTokenStorage; | ||||
|  | ||||
|     public function __construct(Azure $azure, ClientRegistry $clientRegistry, MachineTokenStorage $machineTokenStorage) | ||||
|     { | ||||
|         parent::__construct('chill:calendar:msgraph-grant-admin-consent'); | ||||
|  | ||||
|         $this->azure = $azure; | ||||
|         $this->clientRegistry = $clientRegistry; | ||||
|         $this->machineTokenStorage = $machineTokenStorage; | ||||
|     } | ||||
|  | ||||
|     protected function execute(InputInterface $input, OutputInterface $output) | ||||
|     { | ||||
|         /** @var FormatterHelper $formatter */ | ||||
|         $formatter = $this->getHelper('formatter'); | ||||
|         $this->azure->scope = ['https://graph.microsoft.com/.default']; | ||||
|         $authorizationUrl = explode('?', $this->azure->getAuthorizationUrl(['prompt' => 'admin_consent'])); | ||||
|  | ||||
|         // replace the first part by the admin consent authorization url | ||||
|         $authorizationUrl[0] = strtr('https://login.microsoftonline.com/{tenant}/adminconsent', ['{tenant}' => $this->azure->tenant]); | ||||
|  | ||||
|         $output->writeln('Go to the url'); | ||||
|         $output->writeln(implode('?', $authorizationUrl)); | ||||
|         $output->writeln('Authenticate as admin, and grant admin consent'); | ||||
|  | ||||
|         // not necessary ? | ||||
|         $helper = $this->getHelper('question'); | ||||
|         $question = new ConfirmationQuestion('Access granted ?'); | ||||
|  | ||||
|         if (!$helper->ask($input, $output, $question)) { | ||||
|             $messages = ['No problem, we will wait for you', 'Grant access and come back here']; | ||||
|             $output->writeln($formatter->formatBlock($messages, 'warning')); | ||||
|  | ||||
|             return 0; | ||||
|         } | ||||
|  | ||||
|         $token = $this->machineTokenStorage->getToken(); | ||||
|  | ||||
|         $messages = ['Token acquired!', 'We could acquire a machine token successfully']; | ||||
|         $output->writeln($formatter->formatBlock($messages, 'success')); | ||||
|  | ||||
|         $output->writeln('Token information:'); | ||||
|         $output->writeln($token->getToken()); | ||||
|         $output->writeln('Expires at: ' . $token->getExpires()); | ||||
|         $output->writeln('To inspect the token content, go to https://jwt.ms/#access_token=' . urlencode($token->getToken())); | ||||
|  | ||||
|         return 0; | ||||
|     } | ||||
| } | ||||
| @@ -0,0 +1,171 @@ | ||||
| <?php | ||||
|  | ||||
| /** | ||||
|  * Chill is a software for social workers. | ||||
|  * | ||||
|  * For the full copyright and license information, please view | ||||
|  * the LICENSE file that was distributed with this source code. | ||||
|  */ | ||||
|  | ||||
| declare(strict_types=1); | ||||
|  | ||||
| /* | ||||
|  * Chill is a software for social workers | ||||
|  * | ||||
|  * For the full copyright and license information, please view | ||||
|  * the LICENSE file that was distributed with this source code. | ||||
|  */ | ||||
|  | ||||
| namespace Chill\CalendarBundle\Command; | ||||
|  | ||||
| use Chill\CalendarBundle\RemoteCalendar\Connector\MSGraph\EventsOnUserSubscriptionCreator; | ||||
| use Chill\CalendarBundle\RemoteCalendar\Connector\MSGraph\MapCalendarToUser; | ||||
| use Chill\CalendarBundle\RemoteCalendar\Connector\MSGraph\MSGraphUserRepository; | ||||
| use DateInterval; | ||||
| use DateTimeImmutable; | ||||
| use Doctrine\ORM\EntityManagerInterface; | ||||
| use Psr\Log\LoggerInterface; | ||||
| use Symfony\Component\Console\Command\Command; | ||||
| use Symfony\Component\Console\Input\InputInterface; | ||||
| use Symfony\Component\Console\Input\InputOption; | ||||
| use Symfony\Component\Console\Output\OutputInterface; | ||||
|  | ||||
| class MapAndSubscribeUserCalendarCommand extends Command | ||||
| { | ||||
|     private EntityManagerInterface $em; | ||||
|  | ||||
|     private EventsOnUserSubscriptionCreator $eventsOnUserSubscriptionCreator; | ||||
|  | ||||
|     private LoggerInterface $logger; | ||||
|  | ||||
|     private MapCalendarToUser $mapCalendarToUser; | ||||
|  | ||||
|     private MSGraphUserRepository $userRepository; | ||||
|  | ||||
|     public function __construct( | ||||
|         EntityManagerInterface $em, | ||||
|         EventsOnUserSubscriptionCreator $eventsOnUserSubscriptionCreator, | ||||
|         LoggerInterface $logger, | ||||
|         MapCalendarToUser $mapCalendarToUser, | ||||
|         MSGraphUserRepository $userRepository | ||||
|     ) { | ||||
|         parent::__construct('chill:calendar:msgraph-user-map-subscribe'); | ||||
|  | ||||
|         $this->em = $em; | ||||
|         $this->eventsOnUserSubscriptionCreator = $eventsOnUserSubscriptionCreator; | ||||
|         $this->logger = $logger; | ||||
|         $this->mapCalendarToUser = $mapCalendarToUser; | ||||
|         $this->userRepository = $userRepository; | ||||
|     } | ||||
|  | ||||
|     public function execute(InputInterface $input, OutputInterface $output): int | ||||
|     { | ||||
|         $this->logger->info(__CLASS__ . ' execute command'); | ||||
|  | ||||
|         $limit = 50; | ||||
|         $offset = 0; | ||||
|         /** @var DateInterval $interval the interval before the end of the expiration */ | ||||
|         $interval = new DateInterval('P1D'); | ||||
|         $expiration = (new DateTimeImmutable('now'))->add(new DateInterval($input->getOption('subscription-duration'))); | ||||
|         $total = $this->userRepository->countByMostOldSubscriptionOrWithoutSubscriptionOrData($interval); | ||||
|         $created = 0; | ||||
|         $renewed = 0; | ||||
|  | ||||
|         $this->logger->info(__CLASS__ . ' the number of user to get - renew', [ | ||||
|             'total' => $total, | ||||
|             'expiration' => $expiration->format(DateTimeImmutable::ATOM), | ||||
|         ]); | ||||
|  | ||||
|         while ($offset < ($total - 1)) { | ||||
|             $users = $this->userRepository->findByMostOldSubscriptionOrWithoutSubscriptionOrData( | ||||
|                 $interval, | ||||
|                 $limit, | ||||
|                 $offset | ||||
|             ); | ||||
|  | ||||
|             foreach ($users as $user) { | ||||
|                 if (!$this->mapCalendarToUser->hasUserId($user)) { | ||||
|                     $this->mapCalendarToUser->writeMetadata($user); | ||||
|                 } | ||||
|  | ||||
|                 if ($this->mapCalendarToUser->hasUserId($user)) { | ||||
|                     // we first try to renew an existing subscription, if any. | ||||
|                     // if not, or if it fails, we try to create a new one | ||||
|                     if ($this->mapCalendarToUser->hasActiveSubscription($user)) { | ||||
|                         $this->logger->debug(__CLASS__ . ' renew a subscription for', [ | ||||
|                             'userId' => $user->getId(), | ||||
|                             'username' => $user->getUsernameCanonical(), | ||||
|                         ]); | ||||
|  | ||||
|                         ['secret' => $secret, 'id' => $id, 'expiration' => $expirationTs] | ||||
|                             = $this->eventsOnUserSubscriptionCreator->renewSubscriptionForUser($user, $expiration); | ||||
|                         $this->mapCalendarToUser->writeSubscriptionMetadata($user, $expirationTs, $id, $secret); | ||||
|  | ||||
|                         if (0 !== $expirationTs) { | ||||
|                             ++$renewed; | ||||
|                         } else { | ||||
|                             $this->logger->warning(__CLASS__ . ' could not renew subscription for a user', [ | ||||
|                                 'userId' => $user->getId(), | ||||
|                                 'username' => $user->getUsernameCanonical(), | ||||
|                             ]); | ||||
|                         } | ||||
|                     } | ||||
|  | ||||
|                     if (!$this->mapCalendarToUser->hasActiveSubscription($user)) { | ||||
|                         $this->logger->debug(__CLASS__ . ' create a subscription for', [ | ||||
|                             'userId' => $user->getId(), | ||||
|                             'username' => $user->getUsernameCanonical(), | ||||
|                         ]); | ||||
|  | ||||
|                         ['secret' => $secret, 'id' => $id, 'expiration' => $expirationTs] | ||||
|                             = $this->eventsOnUserSubscriptionCreator->createSubscriptionForUser($user, $expiration); | ||||
|                         $this->mapCalendarToUser->writeSubscriptionMetadata($user, $expirationTs, $id, $secret); | ||||
|  | ||||
|                         if (0 !== $expirationTs) { | ||||
|                             ++$created; | ||||
|                         } else { | ||||
|                             $this->logger->warning(__CLASS__ . ' could not create subscription for a user', [ | ||||
|                                 'userId' => $user->getId(), | ||||
|                                 'username' => $user->getUsernameCanonical(), | ||||
|                             ]); | ||||
|                         } | ||||
|                     } | ||||
|                 } | ||||
|  | ||||
|                 ++$offset; | ||||
|             } | ||||
|  | ||||
|             $this->em->flush(); | ||||
|             $this->em->clear(); | ||||
|         } | ||||
|  | ||||
|         $this->logger->warning(__CLASS__ . ' process executed', [ | ||||
|             'created' => $created, | ||||
|             'renewed' => $renewed, | ||||
|         ]); | ||||
|  | ||||
|         return 0; | ||||
|     } | ||||
|  | ||||
|     protected function configure() | ||||
|     { | ||||
|         parent::configure(); | ||||
|  | ||||
|         $this | ||||
|             ->setDescription('MSGraph: collect user metadata and create subscription on events for users') | ||||
|             ->addOption( | ||||
|                 'renew-before-end-interval', | ||||
|                 'r', | ||||
|                 InputOption::VALUE_OPTIONAL, | ||||
|                 'delay before renewing subscription', | ||||
|                 'P1D' | ||||
|             ) | ||||
|             ->addOption( | ||||
|                 'subscription-duration', | ||||
|                 's', | ||||
|                 InputOption::VALUE_OPTIONAL, | ||||
|                 'duration for the subscription', | ||||
|                 'PT4230M' | ||||
|             ); | ||||
|     } | ||||
| } | ||||
| @@ -0,0 +1,48 @@ | ||||
| <?php | ||||
|  | ||||
| /** | ||||
|  * Chill is a software for social workers. | ||||
|  * | ||||
|  * For the full copyright and license information, please view | ||||
|  * the LICENSE file that was distributed with this source code. | ||||
|  */ | ||||
|  | ||||
| declare(strict_types=1); | ||||
|  | ||||
| /* | ||||
|  * Chill is a software for social workers | ||||
|  * | ||||
|  * For the full copyright and license information, please view | ||||
|  * the LICENSE file that was distributed with this source code. | ||||
|  */ | ||||
|  | ||||
| namespace Chill\CalendarBundle\Command; | ||||
|  | ||||
| use Chill\CalendarBundle\Service\ShortMessageNotification\BulkCalendarShortMessageSender; | ||||
| use Symfony\Component\Console\Command\Command; | ||||
| use Symfony\Component\Console\Input\InputInterface; | ||||
| use Symfony\Component\Console\Output\OutputInterface; | ||||
|  | ||||
| class SendShortMessageOnEligibleCalendar extends Command | ||||
| { | ||||
|     private BulkCalendarShortMessageSender $messageSender; | ||||
|  | ||||
|     public function __construct(BulkCalendarShortMessageSender $messageSender) | ||||
|     { | ||||
|         parent::__construct(); | ||||
|  | ||||
|         $this->messageSender = $messageSender; | ||||
|     } | ||||
|  | ||||
|     public function getName() | ||||
|     { | ||||
|         return 'chill:calendar:send-short-messages'; | ||||
|     } | ||||
|  | ||||
|     protected function execute(InputInterface $input, OutputInterface $output) | ||||
|     { | ||||
|         $this->messageSender->sendBulkMessageToEligibleCalendars(); | ||||
|  | ||||
|         return 0; | ||||
|     } | ||||
| } | ||||
| @@ -0,0 +1,207 @@ | ||||
| <?php | ||||
|  | ||||
| /** | ||||
|  * Chill is a software for social workers. | ||||
|  * | ||||
|  * For the full copyright and license information, please view | ||||
|  * the LICENSE file that was distributed with this source code. | ||||
|  */ | ||||
|  | ||||
| declare(strict_types=1); | ||||
|  | ||||
| /* | ||||
|  * Chill is a software for social workers | ||||
|  * | ||||
|  * For the full copyright and license information, please view | ||||
|  * the LICENSE file that was distributed with this source code. | ||||
|  */ | ||||
|  | ||||
| namespace Chill\CalendarBundle\Command; | ||||
|  | ||||
| use Chill\CalendarBundle\Entity\Calendar; | ||||
| use Chill\CalendarBundle\Service\ShortMessageNotification\ShortMessageForCalendarBuilderInterface; | ||||
| use Chill\MainBundle\Entity\User; | ||||
| use Chill\MainBundle\Phonenumber\PhoneNumberHelperInterface; | ||||
| use Chill\MainBundle\Repository\UserRepositoryInterface; | ||||
| use Chill\MainBundle\Service\ShortMessage\ShortMessageTransporterInterface; | ||||
| use Chill\PersonBundle\Entity\Person; | ||||
| use Chill\PersonBundle\Repository\PersonRepository; | ||||
| use DateInterval; | ||||
| use DateTimeImmutable; | ||||
| use libphonenumber\PhoneNumber; | ||||
| use libphonenumber\PhoneNumberFormat; | ||||
| use libphonenumber\PhoneNumberType; | ||||
| use libphonenumber\PhoneNumberUtil; | ||||
| use Symfony\Component\Console\Command\Command; | ||||
| use Symfony\Component\Console\Helper\QuestionHelper; | ||||
| use Symfony\Component\Console\Input\InputInterface; | ||||
| use Symfony\Component\Console\Output\OutputInterface; | ||||
| use Symfony\Component\Console\Question\ConfirmationQuestion; | ||||
| use Symfony\Component\Console\Question\Question; | ||||
| use UnexpectedValueException; | ||||
| use function count; | ||||
|  | ||||
| class SendTestShortMessageOnCalendarCommand extends Command | ||||
| { | ||||
|     private ShortMessageForCalendarBuilderInterface $messageForCalendarBuilder; | ||||
|  | ||||
|     private PersonRepository $personRepository; | ||||
|  | ||||
|     private PhoneNumberHelperInterface $phoneNumberHelper; | ||||
|  | ||||
|     private PhoneNumberUtil $phoneNumberUtil; | ||||
|  | ||||
|     private ShortMessageTransporterInterface $transporter; | ||||
|  | ||||
|     private UserRepositoryInterface $userRepository; | ||||
|  | ||||
|     public function __construct( | ||||
|         PersonRepository $personRepository, | ||||
|         PhoneNumberUtil $phoneNumberUtil, | ||||
|         PhoneNumberHelperInterface $phoneNumberHelper, | ||||
|         ShortMessageForCalendarBuilderInterface $messageForCalendarBuilder, | ||||
|         ShortMessageTransporterInterface $transporter, | ||||
|         UserRepositoryInterface $userRepository | ||||
|     ) { | ||||
|         parent::__construct(); | ||||
|  | ||||
|         $this->personRepository = $personRepository; | ||||
|         $this->phoneNumberUtil = $phoneNumberUtil; | ||||
|         $this->phoneNumberHelper = $phoneNumberHelper; | ||||
|         $this->messageForCalendarBuilder = $messageForCalendarBuilder; | ||||
|         $this->transporter = $transporter; | ||||
|         $this->userRepository = $userRepository; | ||||
|     } | ||||
|  | ||||
|     public function getName() | ||||
|     { | ||||
|         return 'chill:calendar:test-send-short-message'; | ||||
|     } | ||||
|  | ||||
|     protected function configure() | ||||
|     { | ||||
|         $this->setDescription('Test sending a SMS for a dummy calendar appointment'); | ||||
|     } | ||||
|  | ||||
|     protected function execute(InputInterface $input, OutputInterface $output): int | ||||
|     { | ||||
|         $calendar = new Calendar(); | ||||
|         $calendar->setSendSMS(true); | ||||
|  | ||||
|         /** @var QuestionHelper $helper */ | ||||
|         $helper = $this->getHelper('question'); | ||||
|  | ||||
|         // start date | ||||
|         $question = new Question('When will start the appointment ? (default: "1 hour") ', '1 hour'); | ||||
|         $startDate = new DateTimeImmutable($helper->ask($input, $output, $question)); | ||||
|  | ||||
|         if (false === $startDate) { | ||||
|             throw new UnexpectedValueException('could not create a date with this date and time'); | ||||
|         } | ||||
|  | ||||
|         $calendar->setStartDate($startDate); | ||||
|  | ||||
|         // end date | ||||
|         $question = new Question('How long will last the appointment ? (default: "PT30M") ', 'PT30M'); | ||||
|         $interval = new DateInterval($helper->ask($input, $output, $question)); | ||||
|  | ||||
|         if (false === $interval) { | ||||
|             throw new UnexpectedValueException('could not create the interval'); | ||||
|         } | ||||
|  | ||||
|         $calendar->setEndDate($calendar->getStartDate()->add($interval)); | ||||
|  | ||||
|         // a person | ||||
|         $question = new Question('Who will participate ? Give an id for a person. '); | ||||
|         $question | ||||
|             ->setValidator(function ($answer): Person { | ||||
|                 if (!is_numeric($answer)) { | ||||
|                     throw new UnexpectedValueException('the answer must be numeric'); | ||||
|                 } | ||||
|  | ||||
|                 if (0 >= (int) $answer) { | ||||
|                     throw new UnexpectedValueException('the answer must be greater than zero'); | ||||
|                 } | ||||
|  | ||||
|                 $person = $this->personRepository->find((int) $answer); | ||||
|  | ||||
|                 if (null === $person) { | ||||
|                     throw new UnexpectedValueException('The person is not found'); | ||||
|                 } | ||||
|  | ||||
|                 return $person; | ||||
|             }); | ||||
|  | ||||
|         $person = $helper->ask($input, $output, $question); | ||||
|         $calendar->addPerson($person); | ||||
|  | ||||
|         // a main user | ||||
|         $question = new Question('Who will be the main user ? Give an id for a user. '); | ||||
|         $question | ||||
|             ->setValidator(function ($answer): User { | ||||
|                 if (!is_numeric($answer)) { | ||||
|                     throw new UnexpectedValueException('the answer must be numeric'); | ||||
|                 } | ||||
|  | ||||
|                 if (0 >= (int) $answer) { | ||||
|                     throw new UnexpectedValueException('the answer must be greater than zero'); | ||||
|                 } | ||||
|  | ||||
|                 $user = $this->userRepository->find((int) $answer); | ||||
|  | ||||
|                 if (null === $user) { | ||||
|                     throw new UnexpectedValueException('The user is not found'); | ||||
|                 } | ||||
|  | ||||
|                 return $user; | ||||
|             }); | ||||
|  | ||||
|         $user = $helper->ask($input, $output, $question); | ||||
|         $calendar->setMainUser($user); | ||||
|  | ||||
|         // phonenumber | ||||
|         $phonenumberFormatted = null !== $person->getMobilenumber() ? | ||||
|             $this->phoneNumberUtil->format($person->getMobilenumber(), PhoneNumberFormat::E164) : ''; | ||||
|         $question = new Question( | ||||
|             sprintf('To which number are we going to send this fake message ? (default to: %s)', $phonenumberFormatted), | ||||
|             $phonenumberFormatted | ||||
|         ); | ||||
|  | ||||
|         $question->setNormalizer(function ($answer): PhoneNumber { | ||||
|             if (null === $answer) { | ||||
|                 throw new UnexpectedValueException('The person is not found'); | ||||
|             } | ||||
|  | ||||
|             $phone = $this->phoneNumberUtil->parse($answer, 'BE'); | ||||
|  | ||||
|             if (!$this->phoneNumberUtil->isPossibleNumberForType($phone, PhoneNumberType::MOBILE)) { | ||||
|                 throw new UnexpectedValueException('Phone number si not a mobile'); | ||||
|             } | ||||
|  | ||||
|             return $phone; | ||||
|         }); | ||||
|  | ||||
|         $phone = $helper->ask($input, $output, $question); | ||||
|  | ||||
|         $question = new ConfirmationQuestion('really send the message to the phone ?'); | ||||
|         $reallySend = (bool) $helper->ask($input, $output, $question); | ||||
|  | ||||
|         $messages = $this->messageForCalendarBuilder->buildMessageForCalendar($calendar); | ||||
|  | ||||
|         if (0 === count($messages)) { | ||||
|             $output->writeln('no message to send to this user'); | ||||
|         } | ||||
|  | ||||
|         foreach ($messages as $key => $message) { | ||||
|             $output->writeln("The short message for SMS {$key} will be: "); | ||||
|             $output->writeln($message->getContent()); | ||||
|             $message->setPhoneNumber($phone); | ||||
|  | ||||
|             if ($reallySend) { | ||||
|                 $this->transporter->send($message); | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         return 0; | ||||
|     } | ||||
| } | ||||
| @@ -11,11 +11,73 @@ declare(strict_types=1); | ||||
|  | ||||
| namespace Chill\CalendarBundle\Controller; | ||||
|  | ||||
| use Chill\CalendarBundle\Repository\CalendarRepository; | ||||
| use Chill\MainBundle\CRUD\Controller\ApiController; | ||||
| use Chill\MainBundle\Entity\User; | ||||
| use Chill\MainBundle\Serializer\Model\Collection; | ||||
| use DateTimeImmutable; | ||||
| use Symfony\Component\HttpFoundation\JsonResponse; | ||||
| use Symfony\Component\HttpFoundation\Request; | ||||
| use Symfony\Component\HttpFoundation\Response; | ||||
| use Symfony\Component\HttpKernel\Exception\BadRequestHttpException; | ||||
| use Symfony\Component\Routing\Annotation\Route; | ||||
|  | ||||
| class CalendarAPIController extends ApiController | ||||
| { | ||||
|     private CalendarRepository $calendarRepository; | ||||
|  | ||||
|     public function __construct(CalendarRepository $calendarRepository) | ||||
|     { | ||||
|         $this->calendarRepository = $calendarRepository; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * @Route("/api/1.0/calendar/calendar/by-user/{id}.{_format}", | ||||
|      *     name="chill_api_single_calendar_list_by-user", | ||||
|      *     requirements={"_format": "json"} | ||||
|      * ) | ||||
|      */ | ||||
|     public function listByUser(User $user, Request $request, string $_format): JsonResponse | ||||
|     { | ||||
|         $this->denyAccessUnlessGranted('ROLE_USER'); | ||||
|  | ||||
|         if (!$request->query->has('dateFrom')) { | ||||
|             throw new BadRequestHttpException('You must provide a dateFrom parameter'); | ||||
|         } | ||||
|  | ||||
|         if (false === $dateFrom = DateTimeImmutable::createFromFormat( | ||||
|             DateTimeImmutable::ATOM, | ||||
|             $request->query->get('dateFrom') | ||||
|         )) { | ||||
|             throw new BadRequestHttpException('dateFrom not parsable'); | ||||
|         } | ||||
|  | ||||
|         if (!$request->query->has('dateTo')) { | ||||
|             throw new BadRequestHttpException('You must provide a dateTo parameter'); | ||||
|         } | ||||
|  | ||||
|         if (false === $dateTo = DateTimeImmutable::createFromFormat( | ||||
|             DateTimeImmutable::ATOM, | ||||
|             $request->query->get('dateTo') | ||||
|         )) { | ||||
|             throw new BadRequestHttpException('dateTo not parsable'); | ||||
|         } | ||||
|  | ||||
|         $total = $this->calendarRepository->countByUser($user, $dateFrom, $dateTo); | ||||
|         $paginator = $this->getPaginatorFactory()->create($total); | ||||
|         $ranges = $this->calendarRepository->findByUser( | ||||
|             $user, | ||||
|             $dateFrom, | ||||
|             $dateTo, | ||||
|             $paginator->getItemsPerPage(), | ||||
|             $paginator->getCurrentPageFirstItemNumber() | ||||
|         ); | ||||
|  | ||||
|         $collection = new Collection($ranges, $paginator); | ||||
|  | ||||
|         return $this->json($collection, Response::HTTP_OK, [], ['groups' => ['calendar:light']]); | ||||
|     } | ||||
|  | ||||
|     protected function customizeQuery(string $action, Request $request, $qb): void | ||||
|     { | ||||
|         if ($request->query->has('main_user')) { | ||||
|   | ||||
| @@ -13,53 +13,67 @@ namespace Chill\CalendarBundle\Controller; | ||||
|  | ||||
| use Chill\CalendarBundle\Entity\Calendar; | ||||
| use Chill\CalendarBundle\Form\CalendarType; | ||||
| use Chill\CalendarBundle\RemoteCalendar\Connector\RemoteCalendarConnectorInterface; | ||||
| use Chill\CalendarBundle\Repository\CalendarACLAwareRepositoryInterface; | ||||
| use Chill\CalendarBundle\Repository\CalendarRepository; | ||||
| use Chill\MainBundle\Entity\User; | ||||
| use Chill\MainBundle\Pagination\PaginatorFactory; | ||||
| use Chill\MainBundle\Security\Authorization\AuthorizationHelper; | ||||
| use Chill\MainBundle\Repository\UserRepository; | ||||
| use Chill\MainBundle\Templating\Listing\FilterOrderHelper; | ||||
| use Chill\MainBundle\Templating\Listing\FilterOrderHelperFactoryInterface; | ||||
| use Chill\PersonBundle\Entity\AccompanyingPeriod; | ||||
| use Chill\PersonBundle\Entity\Person; | ||||
| use Chill\ThirdPartyBundle\Entity\ThirdParty; | ||||
| use DateTimeImmutable; | ||||
| use Exception; | ||||
| use Psr\Log\LoggerInterface; | ||||
| use Symfony\Bundle\FrameworkBundle\Controller\AbstractController; | ||||
| use Symfony\Component\EventDispatcher\EventDispatcherInterface; | ||||
| use Symfony\Component\Form\Extension\Core\Type\SubmitType; | ||||
| use Symfony\Component\Form\Form; | ||||
| use Symfony\Component\Form\FormInterface; | ||||
| use Symfony\Component\HttpFoundation\RedirectResponse; | ||||
| use Symfony\Component\HttpFoundation\Request; | ||||
| use Symfony\Component\HttpFoundation\Response; | ||||
| use Symfony\Component\HttpKernel\Exception\UnauthorizedHttpException; | ||||
| use Symfony\Component\Routing\Annotation\Route; | ||||
| use Symfony\Component\Serializer\SerializerInterface; | ||||
|  | ||||
| class CalendarController extends AbstractController | ||||
| { | ||||
|     protected AuthorizationHelper $authorizationHelper; | ||||
|  | ||||
|     protected EventDispatcherInterface $eventDispatcher; | ||||
|  | ||||
|     protected LoggerInterface $logger; | ||||
|  | ||||
|     protected PaginatorFactory $paginator; | ||||
|  | ||||
|     protected SerializerInterface $serializer; | ||||
|     private CalendarACLAwareRepositoryInterface $calendarACLAwareRepository; | ||||
|  | ||||
|     private CalendarRepository $calendarRepository; | ||||
|  | ||||
|     private FilterOrderHelperFactoryInterface $filterOrderHelperFactory; | ||||
|  | ||||
|     private LoggerInterface $logger; | ||||
|  | ||||
|     private PaginatorFactory $paginator; | ||||
|  | ||||
|     private RemoteCalendarConnectorInterface $remoteCalendarConnector; | ||||
|  | ||||
|     private SerializerInterface $serializer; | ||||
|  | ||||
|     private UserRepository $userRepository; | ||||
|  | ||||
|     public function __construct( | ||||
|         EventDispatcherInterface $eventDispatcher, | ||||
|         AuthorizationHelper $authorizationHelper, | ||||
|         CalendarRepository $calendarRepository, | ||||
|         CalendarACLAwareRepositoryInterface $calendarACLAwareRepository, | ||||
|         FilterOrderHelperFactoryInterface $filterOrderHelperFactory, | ||||
|         LoggerInterface $logger, | ||||
|         SerializerInterface $serializer, | ||||
|         PaginatorFactory $paginator, | ||||
|         CalendarRepository $calendarRepository | ||||
|         RemoteCalendarConnectorInterface $remoteCalendarConnector, | ||||
|         SerializerInterface $serializer, | ||||
|         UserRepository $userRepository | ||||
|     ) { | ||||
|         $this->eventDispatcher = $eventDispatcher; | ||||
|         $this->authorizationHelper = $authorizationHelper; | ||||
|         $this->logger = $logger; | ||||
|         $this->serializer = $serializer; | ||||
|         $this->paginator = $paginator; | ||||
|         $this->calendarRepository = $calendarRepository; | ||||
|         $this->calendarACLAwareRepository = $calendarACLAwareRepository; | ||||
|         $this->filterOrderHelperFactory = $filterOrderHelperFactory; | ||||
|         $this->logger = $logger; | ||||
|         $this->paginator = $paginator; | ||||
|         $this->remoteCalendarConnector = $remoteCalendarConnector; | ||||
|         $this->serializer = $serializer; | ||||
|         $this->userRepository = $userRepository; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
| @@ -67,12 +81,13 @@ class CalendarController extends AbstractController | ||||
|      * | ||||
|      * @Route("/{_locale}/calendar/{id}/delete", name="chill_calendar_calendar_delete") | ||||
|      */ | ||||
|     public function deleteAction(Request $request, int $id) | ||||
|     public function deleteAction(Request $request, Calendar $entity) | ||||
|     { | ||||
|         $view = null; | ||||
|         $em = $this->getDoctrine()->getManager(); | ||||
|  | ||||
|         [$user, $accompanyingPeriod] = $this->getEntity($request); | ||||
|         $accompanyingPeriod = $entity->getAccompanyingPeriod(); | ||||
|         $user = null; // TODO legacy code ? remove it ? | ||||
|  | ||||
|         if ($accompanyingPeriod instanceof AccompanyingPeriod) { | ||||
|             $view = '@ChillCalendar/Calendar/confirm_deleteByAccompanyingCourse.html.twig'; | ||||
| @@ -80,14 +95,7 @@ class CalendarController extends AbstractController | ||||
|             $view = '@ChillCalendar/Calendar/confirm_deleteByUser.html.twig'; | ||||
|         } | ||||
|  | ||||
|         /** @var Calendar $entity */ | ||||
|         $entity = $em->getRepository(\Chill\CalendarBundle\Entity\Calendar::class)->find($id); | ||||
|  | ||||
|         if (!$entity) { | ||||
|             throw $this->createNotFoundException('Unable to find Calendar entity.'); | ||||
|         } | ||||
|  | ||||
|         $form = $this->createDeleteForm($id, $user, $accompanyingPeriod); | ||||
|         $form = $this->createDeleteForm($entity->getId(), $user, $accompanyingPeriod); | ||||
|  | ||||
|         if ($request->getMethod() === Request::METHOD_DELETE) { | ||||
|             $form->handleRequest($request); | ||||
| @@ -106,7 +114,7 @@ class CalendarController extends AbstractController | ||||
|  | ||||
|                 $params = $this->buildParamsToUrl($user, $accompanyingPeriod); | ||||
|  | ||||
|                 return $this->redirectToRoute('chill_calendar_calendar_list', $params); | ||||
|                 return $this->redirectToRoute('chill_calendar_calendar_list_by_period', $params); | ||||
|             } | ||||
|         } | ||||
|  | ||||
| @@ -126,8 +134,12 @@ class CalendarController extends AbstractController | ||||
|      * | ||||
|      * @Route("/{_locale}/calendar/calendar/{id}/edit", name="chill_calendar_calendar_edit") | ||||
|      */ | ||||
|     public function editAction(int $id, Request $request): Response | ||||
|     public function editAction(Calendar $entity, Request $request): Response | ||||
|     { | ||||
|         if (!$this->remoteCalendarConnector->isReady()) { | ||||
|             return $this->remoteCalendarConnector->getMakeReadyResponse($request->getUri()); | ||||
|         } | ||||
|  | ||||
|         $view = null; | ||||
|         $em = $this->getDoctrine()->getManager(); | ||||
|  | ||||
| @@ -136,35 +148,28 @@ class CalendarController extends AbstractController | ||||
|         if ($accompanyingPeriod instanceof AccompanyingPeriod) { | ||||
|             $view = '@ChillCalendar/Calendar/editByAccompanyingCourse.html.twig'; | ||||
|         } elseif ($user instanceof User) { | ||||
|             throw new Exception('to analyze'); | ||||
|             $view = '@ChillCalendar/Calendar/editByUser.html.twig'; | ||||
|         } | ||||
|  | ||||
|         $entity = $em->getRepository(\Chill\CalendarBundle\Entity\Calendar::class)->find($id); | ||||
|  | ||||
|         if (!$entity) { | ||||
|             throw $this->createNotFoundException('Unable to find Calendar entity.'); | ||||
|         } | ||||
|  | ||||
|         $form = $this->createForm(CalendarType::class, $entity, [ | ||||
|             'accompanyingPeriod' => $accompanyingPeriod, | ||||
|         ])->handleRequest($request); | ||||
|         $form = $this->createForm(CalendarType::class, $entity); | ||||
|         $form->handleRequest($request); | ||||
|  | ||||
|         if ($form->isSubmitted() && $form->isValid()) { | ||||
|             $em->persist($entity); | ||||
|             $em->flush(); | ||||
|  | ||||
|             $this->addFlash('success', $this->get('translator')->trans('Success : calendar item updated!')); | ||||
|  | ||||
|             $params = $this->buildParamsToUrl($user, $accompanyingPeriod); | ||||
|  | ||||
|             return $this->redirectToRoute('chill_calendar_calendar_list', $params); | ||||
|             return $this->redirectToRoute('chill_calendar_calendar_list_by_period', $params); | ||||
|         } | ||||
|  | ||||
|         if ($form->isSubmitted() && !$form->isValid()) { | ||||
|             $this->addFlash('error', $this->get('translator')->trans('This form contains errors')); | ||||
|         } | ||||
|  | ||||
|         $deleteForm = $this->createDeleteForm($id, $user, $accompanyingPeriod); | ||||
|         $deleteForm = $this->createDeleteForm($entity->getId(), $user, $accompanyingPeriod); | ||||
|  | ||||
|         if (null === $view) { | ||||
|             throw $this->createNotFoundException('Template not found'); | ||||
| @@ -177,7 +182,7 @@ class CalendarController extends AbstractController | ||||
|             'form' => $form->createView(), | ||||
|             'delete_form' => $deleteForm->createView(), | ||||
|             'accompanyingCourse' => $accompanyingPeriod, | ||||
|             'user' => $user, | ||||
|             //            'user' => $user, | ||||
|             'entity_json' => $entity_array, | ||||
|         ]); | ||||
|     } | ||||
| @@ -185,45 +190,53 @@ class CalendarController extends AbstractController | ||||
|     /** | ||||
|      * Lists all Calendar entities. | ||||
|      * | ||||
|      * @Route("/{_locale}/calendar/calendar/", name="chill_calendar_calendar_list") | ||||
|      * @Route("/{_locale}/calendar/calendar/by-period/{id}", name="chill_calendar_calendar_list_by_period") | ||||
|      */ | ||||
|     public function listAction(Request $request): Response | ||||
|     public function listActionByCourse(AccompanyingPeriod $accompanyingPeriod): Response | ||||
|     { | ||||
|         $view = null; | ||||
|         $filterOrder = $this->buildListFilterOrder(); | ||||
|         ['from' => $from, 'to' => $to] = $filterOrder->getDateRangeData('startDate'); | ||||
|  | ||||
|         [$user, $accompanyingPeriod] = $this->getEntity($request); | ||||
|         $total = $this->calendarACLAwareRepository | ||||
|             ->countByAccompanyingPeriod($accompanyingPeriod, $from, $to); | ||||
|         $paginator = $this->paginator->create($total); | ||||
|         $calendarItems = $this->calendarACLAwareRepository->findByAccompanyingPeriod( | ||||
|             $accompanyingPeriod, | ||||
|             $from, | ||||
|             $to, | ||||
|             ['startDate' => 'DESC'], | ||||
|             $paginator->getCurrentPageFirstItemNumber(), | ||||
|             $paginator->getItemsPerPage() | ||||
|         ); | ||||
|  | ||||
|         if ($user instanceof User) { | ||||
|             $calendarItems = $this->calendarRepository->findByUser($user); | ||||
|         return $this->render('@ChillCalendar/Calendar/listByAccompanyingCourse.html.twig', [ | ||||
|             'calendarItems' => $calendarItems, | ||||
|             'accompanyingCourse' => $accompanyingPeriod, | ||||
|             'paginator' => $paginator, | ||||
|             'filterOrder' => $filterOrder, | ||||
|         ]); | ||||
|     } | ||||
|  | ||||
|             $view = '@ChillCalendar/Calendar/listByUser.html.twig'; | ||||
|     /** | ||||
|      * @Route("/{_locale}/calendar/calendar/my", name="chill_calendar_calendar_list_my") | ||||
|      */ | ||||
|     public function myCalendar(Request $request): Response | ||||
|     { | ||||
|         $this->denyAccessUnlessGranted('ROLE_USER'); | ||||
|  | ||||
|             return $this->render($view, [ | ||||
|                 'calendarItems' => $calendarItems, | ||||
|                 'user' => $user, | ||||
|             ]); | ||||
|         if (!$this->remoteCalendarConnector->isReady()) { | ||||
|             return $this->remoteCalendarConnector->getMakeReadyResponse($request->getUri()); | ||||
|         } | ||||
|  | ||||
|         if ($accompanyingPeriod instanceof AccompanyingPeriod) { | ||||
|             $total = $this->calendarRepository->countByAccompanyingPeriod($accompanyingPeriod); | ||||
|             $paginator = $this->paginator->create($total); | ||||
|             $calendarItems = $this->calendarRepository->findBy( | ||||
|                 ['accompanyingPeriod' => $accompanyingPeriod], | ||||
|                 ['startDate' => 'DESC'], | ||||
|                 $paginator->getItemsPerPage(), | ||||
|                 $paginator->getCurrentPageFirstItemNumber() | ||||
|             ); | ||||
|  | ||||
|             $view = '@ChillCalendar/Calendar/listByAccompanyingCourse.html.twig'; | ||||
|  | ||||
|             return $this->render($view, [ | ||||
|                 'calendarItems' => $calendarItems, | ||||
|                 'accompanyingCourse' => $accompanyingPeriod, | ||||
|                 'paginator' => $paginator, | ||||
|             ]); | ||||
|         if (!$this->getUser() instanceof User) { | ||||
|             throw new UnauthorizedHttpException('you are not an user'); | ||||
|         } | ||||
|  | ||||
|         throw new Exception('Unable to list actions.'); | ||||
|         $view = '@ChillCalendar/Calendar/listByUser.html.twig'; | ||||
|  | ||||
|         return $this->render($view, [ | ||||
|             'user' => $this->getUser(), | ||||
|         ]); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
| @@ -233,6 +246,10 @@ class CalendarController extends AbstractController | ||||
|      */ | ||||
|     public function newAction(Request $request): Response | ||||
|     { | ||||
|         if (!$this->remoteCalendarConnector->isReady()) { | ||||
|             return $this->remoteCalendarConnector->getMakeReadyResponse($request->getUri()); | ||||
|         } | ||||
|  | ||||
|         $view = null; | ||||
|         $em = $this->getDoctrine()->getManager(); | ||||
|  | ||||
| @@ -246,8 +263,10 @@ class CalendarController extends AbstractController | ||||
|         // } | ||||
|  | ||||
|         $entity = new Calendar(); | ||||
|         $entity->setUser($this->getUser()); | ||||
|         $entity->setStatus($entity::STATUS_VALID); | ||||
|  | ||||
|         if ($request->query->has('mainUser')) { | ||||
|             $entity->setMainUser($this->userRepository->find($request->query->getInt('mainUser'))); | ||||
|         } | ||||
|  | ||||
|         // if ($user instanceof User) { | ||||
|         //     $entity->setPerson($user); | ||||
| @@ -257,9 +276,8 @@ class CalendarController extends AbstractController | ||||
|             $entity->setAccompanyingPeriod($accompanyingPeriod); | ||||
|         } | ||||
|  | ||||
|         $form = $this->createForm(CalendarType::class, $entity, [ | ||||
|             'accompanyingPeriod' => $accompanyingPeriod, | ||||
|         ])->handleRequest($request); | ||||
|         $form = $this->createForm(CalendarType::class, $entity); | ||||
|         $form->handleRequest($request); | ||||
|  | ||||
|         if ($form->isSubmitted() && $form->isValid()) { | ||||
|             $em->persist($entity); | ||||
| @@ -269,7 +287,7 @@ class CalendarController extends AbstractController | ||||
|  | ||||
|             $params = $this->buildParamsToUrl($user, $accompanyingPeriod); | ||||
|  | ||||
|             return $this->redirectToRoute('chill_calendar_calendar_list', $params); | ||||
|             return $this->redirectToRoute('chill_calendar_calendar_list_by_period', $params); | ||||
|         } | ||||
|  | ||||
|         if ($form->isSubmitted() && !$form->isValid()) { | ||||
| @@ -349,7 +367,7 @@ class CalendarController extends AbstractController | ||||
|             'professionalsId' => $professionalsId, | ||||
|             'date' => $entity->getStartDate()->format('Y-m-d'), | ||||
|             'durationTime' => $durationTimeInMinutes, | ||||
|             'location' => $entity->getLocation()->getId(), | ||||
|             'location' => $entity->getLocation() ? $entity->getLocation()->getId() : null, | ||||
|             'comment' => $entity->getComment()->getComment(), | ||||
|         ]; | ||||
|  | ||||
| @@ -362,6 +380,58 @@ class CalendarController extends AbstractController | ||||
|         ]); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * @Route("/{_locale}/calendar/calendar/{id}/to-activity", name="chill_calendar_calendar_to_activity") | ||||
|      */ | ||||
|     public function toActivity(Request $request, Calendar $calendar): RedirectResponse | ||||
|     { | ||||
|         $personsId = array_map( | ||||
|             static fn (Person $p): int => $p->getId(), | ||||
|             $calendar->getPersons()->toArray() | ||||
|         ); | ||||
|  | ||||
|         $professionalsId = array_map( | ||||
|             static fn (ThirdParty $thirdParty): ?int => $thirdParty->getId(), | ||||
|             $calendar->getProfessionals()->toArray() | ||||
|         ); | ||||
|  | ||||
|         $usersId = array_map( | ||||
|             static fn (User $user): ?int => $user->getId(), | ||||
|             array_merge($calendar->getUsers()->toArray(), [$calendar->getMainUser()]) | ||||
|         ); | ||||
|  | ||||
|         $durationTime = $calendar->getEndDate()->diff($calendar->getStartDate()); | ||||
|         $durationTimeInMinutes = $durationTime->days * 1440 + $durationTime->h * 60 + $durationTime->i; | ||||
|  | ||||
|         $activityData = [ | ||||
|             'calendarId' => $calendar->getId(), | ||||
|             'personsId' => $personsId, | ||||
|             'professionalsId' => $professionalsId, | ||||
|             'usersId' => $usersId, | ||||
|             'date' => $calendar->getStartDate()->format('Y-m-d'), | ||||
|             'durationTime' => $durationTimeInMinutes, | ||||
|             'location' => $calendar->getLocation() ? $calendar->getLocation()->getId() : null, | ||||
|             'comment' => $calendar->getComment()->getComment(), | ||||
|         ]; | ||||
|  | ||||
|         return $this->redirectToRoute( | ||||
|             'chill_activity_activity_new', | ||||
|             [ | ||||
|                 'accompanying_period_id' => $calendar->getAccompanyingPeriod()->getId(), | ||||
|                 'activityData' => $activityData, | ||||
|                 'returnPath' => $request->query->get('returnPath', null), | ||||
|             ] | ||||
|         ); | ||||
|     } | ||||
|  | ||||
|     private function buildListFilterOrder(): FilterOrderHelper | ||||
|     { | ||||
|         $filterOrder = $this->filterOrderHelperFactory->create(self::class); | ||||
|         $filterOrder->addDateRange('startDate', null, new DateTimeImmutable('3 days ago'), null); | ||||
|  | ||||
|         return $filterOrder->build(); | ||||
|     } | ||||
|  | ||||
|     private function buildParamsToUrl(?User $user, ?AccompanyingPeriod $accompanyingPeriod): array | ||||
|     { | ||||
|         $params = []; | ||||
| @@ -371,7 +441,7 @@ class CalendarController extends AbstractController | ||||
|         } | ||||
|  | ||||
|         if (null !== $accompanyingPeriod) { | ||||
|             $params['accompanying_period_id'] = $accompanyingPeriod->getId(); | ||||
|             $params['id'] = $accompanyingPeriod->getId(); | ||||
|         } | ||||
|  | ||||
|         return $params; | ||||
|   | ||||
| @@ -11,38 +11,72 @@ declare(strict_types=1); | ||||
|  | ||||
| namespace Chill\CalendarBundle\Controller; | ||||
|  | ||||
| use Chill\CalendarBundle\Repository\CalendarRangeRepository; | ||||
| use Chill\MainBundle\CRUD\Controller\ApiController; | ||||
| use Chill\MainBundle\Entity\User; | ||||
| use Chill\MainBundle\Serializer\Model\Collection; | ||||
| use DateTimeImmutable; | ||||
| use Symfony\Component\HttpFoundation\JsonResponse; | ||||
| use Symfony\Component\HttpFoundation\Request; | ||||
| use Symfony\Component\HttpFoundation\Response; | ||||
| use Symfony\Component\Routing\Annotation\Route; | ||||
| use Symfony\Component\HttpKernel\Exception\BadRequestHttpException; | ||||
|  | ||||
| use function count; | ||||
| use Symfony\Component\Routing\Annotation\Route; | ||||
|  | ||||
| class CalendarRangeAPIController extends ApiController | ||||
| { | ||||
|     /** | ||||
|      * @Route("/api/1.0/calendar/calendar-range-available.{_format}", name="chill_api_single_calendar_range_available") | ||||
|      */ | ||||
|     public function availableRanges(Request $request, string $_format): JsonResponse | ||||
|     private CalendarRangeRepository $calendarRangeRepository; | ||||
|  | ||||
|     public function __construct(CalendarRangeRepository $calendarRangeRepository) | ||||
|     { | ||||
|         $em = $this->getDoctrine()->getManager(); | ||||
|         $this->calendarRangeRepository = $calendarRangeRepository; | ||||
|     } | ||||
|  | ||||
|         $sql = 'SELECT c FROM ChillCalendarBundle:CalendarRange c | ||||
|         WHERE NOT EXISTS (SELECT cal.id FROM ChillCalendarBundle:Calendar cal WHERE cal.calendarRange = c.id)'; | ||||
|     /** | ||||
|      * @Route("/api/1.0/calendar/calendar-range-available/{id}.{_format}", | ||||
|      *     name="chill_api_single_calendar_range_available", | ||||
|      *     requirements={"_format": "json"} | ||||
|      * ) | ||||
|      */ | ||||
|     public function availableRanges(User $user, Request $request, string $_format): JsonResponse | ||||
|     { | ||||
|         //return new JsonResponse(['ok' => true], 200, [], false); | ||||
|         $this->denyAccessUnlessGranted('ROLE_USER'); | ||||
|  | ||||
|         if ($request->query->has('user')) { | ||||
|             $user = $request->query->get('user'); | ||||
|             $sql = $sql . ' AND c.user = :user'; | ||||
|             $query = $em->createQuery($sql) | ||||
|                 ->setParameter('user', $user); | ||||
|         } else { | ||||
|             $query = $em->createQuery($sql); | ||||
|         if (!$request->query->has('dateFrom')) { | ||||
|             throw new BadRequestHttpException('You must provide a dateFrom parameter'); | ||||
|         } | ||||
|  | ||||
|         $results = $query->getResult(); | ||||
|         if (false === $dateFrom = DateTimeImmutable::createFromFormat( | ||||
|             DateTimeImmutable::ATOM, | ||||
|             $request->query->get('dateFrom') | ||||
|         )) { | ||||
|             throw new BadRequestHttpException('dateFrom not parsable'); | ||||
|         } | ||||
|  | ||||
|         return $this->json(['count' => count($results), 'results' => $results], Response::HTTP_OK, [], ['groups' => ['read']]); | ||||
|         //TODO use also the paginator, eg return $this->serializeCollection('get', $request, $_format, $paginator, $results); | ||||
|         if (!$request->query->has('dateTo')) { | ||||
|             throw new BadRequestHttpException('You must provide a dateTo parameter'); | ||||
|         } | ||||
|  | ||||
|         if (false === $dateTo = DateTimeImmutable::createFromFormat( | ||||
|             DateTimeImmutable::ATOM, | ||||
|             $request->query->get('dateTo') | ||||
|         )) { | ||||
|             throw new BadRequestHttpException('dateTo not parsable'); | ||||
|         } | ||||
|  | ||||
|         $total = $this->calendarRangeRepository->countByAvailableRangesForUser($user, $dateFrom, $dateTo); | ||||
|         $paginator = $this->getPaginatorFactory()->create($total); | ||||
|         $ranges = $this->calendarRangeRepository->findByAvailableRangesForUser( | ||||
|             $user, | ||||
|             $dateFrom, | ||||
|             $dateTo, | ||||
|             $paginator->getItemsPerPage(), | ||||
|             $paginator->getCurrentPageFirstItemNumber() | ||||
|         ); | ||||
|  | ||||
|         $collection = new Collection($ranges, $paginator); | ||||
|  | ||||
|         return $this->json($collection, Response::HTTP_OK, [], ['groups' => ['read']]); | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -0,0 +1,83 @@ | ||||
| <?php | ||||
|  | ||||
| /** | ||||
|  * Chill is a software for social workers. | ||||
|  * | ||||
|  * For the full copyright and license information, please view | ||||
|  * the LICENSE file that was distributed with this source code. | ||||
|  */ | ||||
|  | ||||
| declare(strict_types=1); | ||||
|  | ||||
| /* | ||||
|  * Chill is a software for social workers | ||||
|  * | ||||
|  * For the full copyright and license information, please view | ||||
|  * the LICENSE file that was distributed with this source code. | ||||
|  */ | ||||
|  | ||||
| namespace Chill\CalendarBundle\Controller; | ||||
|  | ||||
| use Chill\CalendarBundle\Entity\Calendar; | ||||
| use Chill\CalendarBundle\Entity\Invite; | ||||
| use Chill\CalendarBundle\Messenger\Message\InviteUpdateMessage; | ||||
| use Chill\CalendarBundle\Security\Voter\InviteVoter; | ||||
| use Chill\MainBundle\Entity\User; | ||||
| use Doctrine\ORM\EntityManagerInterface; | ||||
| use Symfony\Component\HttpFoundation\JsonResponse; | ||||
| use Symfony\Component\HttpFoundation\Response; | ||||
| use Symfony\Component\HttpKernel\Exception\AccessDeniedHttpException; | ||||
| use Symfony\Component\HttpKernel\Exception\BadRequestHttpException; | ||||
| use Symfony\Component\Messenger\MessageBusInterface; | ||||
| use Symfony\Component\Routing\Annotation\Route; | ||||
| use Symfony\Component\Security\Core\Security; | ||||
| use function in_array; | ||||
|  | ||||
| class InviteApiController | ||||
| { | ||||
|     private EntityManagerInterface $entityManager; | ||||
|  | ||||
|     private MessageBusInterface $messageBus; | ||||
|  | ||||
|     private Security $security; | ||||
|  | ||||
|     public function __construct(EntityManagerInterface $entityManager, MessageBusInterface $messageBus, Security $security) | ||||
|     { | ||||
|         $this->entityManager = $entityManager; | ||||
|         $this->messageBus = $messageBus; | ||||
|         $this->security = $security; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Give an answer to a calendar invite. | ||||
|      * | ||||
|      * @Route("/api/1.0/calendar/calendar/{id}/answer/{answer}.json", methods={"post"}) | ||||
|      */ | ||||
|     public function answer(Calendar $calendar, string $answer): Response | ||||
|     { | ||||
|         $user = $this->security->getUser(); | ||||
|  | ||||
|         if (!$user instanceof User) { | ||||
|             throw new AccessDeniedHttpException('not a regular user'); | ||||
|         } | ||||
|  | ||||
|         if (null === $invite = $calendar->getInviteForUser($user)) { | ||||
|             throw new AccessDeniedHttpException('not invited to this calendar'); | ||||
|         } | ||||
|  | ||||
|         if (!$this->security->isGranted(InviteVoter::ANSWER, $invite)) { | ||||
|             throw new AccessDeniedHttpException('not allowed to answer on this invitation'); | ||||
|         } | ||||
|  | ||||
|         if (!in_array($answer, Invite::STATUSES, true)) { | ||||
|             throw new BadRequestHttpException('answer not valid'); | ||||
|         } | ||||
|  | ||||
|         $invite->setStatus($answer); | ||||
|         $this->entityManager->flush(); | ||||
|  | ||||
|         $this->messageBus->dispatch(new InviteUpdateMessage($invite, $this->security->getUser())); | ||||
|  | ||||
|         return new JsonResponse(null, Response::HTTP_ACCEPTED, [], false); | ||||
|     } | ||||
| } | ||||
| @@ -0,0 +1,76 @@ | ||||
| <?php | ||||
|  | ||||
| /** | ||||
|  * Chill is a software for social workers. | ||||
|  * | ||||
|  * For the full copyright and license information, please view | ||||
|  * the LICENSE file that was distributed with this source code. | ||||
|  */ | ||||
|  | ||||
| declare(strict_types=1); | ||||
|  | ||||
| /* | ||||
|  * Chill is a software for social workers | ||||
|  * | ||||
|  * For the full copyright and license information, please view | ||||
|  * the LICENSE file that was distributed with this source code. | ||||
|  */ | ||||
|  | ||||
| namespace Chill\CalendarBundle\Controller; | ||||
|  | ||||
| use Chill\CalendarBundle\RemoteCalendar\Connector\MSGraph\OnBehalfOfUserTokenStorage; | ||||
| use KnpU\OAuth2ClientBundle\Client\ClientRegistry; | ||||
| use League\OAuth2\Client\Provider\Exception\IdentityProviderException; | ||||
| use Symfony\Component\HttpFoundation\RedirectResponse; | ||||
| use Symfony\Component\HttpFoundation\Request; | ||||
| use Symfony\Component\HttpFoundation\Response; | ||||
| use Symfony\Component\Routing\Annotation\Route; | ||||
| use TheNetworg\OAuth2\Client\Provider\Azure; | ||||
| use TheNetworg\OAuth2\Client\Token\AccessToken; | ||||
|  | ||||
| class RemoteCalendarConnectAzureController | ||||
| { | ||||
|     private ClientRegistry $clientRegistry; | ||||
|  | ||||
|     private OnBehalfOfUserTokenStorage $MSGraphTokenStorage; | ||||
|  | ||||
|     public function __construct( | ||||
|         ClientRegistry $clientRegistry, | ||||
|         OnBehalfOfUserTokenStorage $MSGraphTokenStorage | ||||
|     ) { | ||||
|         $this->clientRegistry = $clientRegistry; | ||||
|         $this->MSGraphTokenStorage = $MSGraphTokenStorage; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * @Route("/{_locale}/connect/azure", name="chill_calendar_remote_connect_azure") | ||||
|      */ | ||||
|     public function connectAzure(Request $request): Response | ||||
|     { | ||||
|         $request->getSession()->set('azure_return_path', $request->query->get('returnPath', '/')); | ||||
|  | ||||
|         return $this->clientRegistry | ||||
|             ->getClient('azure') // key used in config/packages/knpu_oauth2_client.yaml | ||||
|             ->redirect(['https://graph.microsoft.com/.default', 'offline_access'], []); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * @Route("/connect/azure/check", name="chill_calendar_remote_connect_azure_check") | ||||
|      */ | ||||
|     public function connectAzureCheck(Request $request): Response | ||||
|     { | ||||
|         /** @var Azure $client */ | ||||
|         $client = $this->clientRegistry->getClient('azure'); | ||||
|  | ||||
|         try { | ||||
|             /** @var AccessToken $token */ | ||||
|             $token = $client->getAccessToken([]); | ||||
|  | ||||
|             $this->MSGraphTokenStorage->setToken($token); | ||||
|         } catch (IdentityProviderException $e) { | ||||
|             throw $e; | ||||
|         } | ||||
|  | ||||
|         return new RedirectResponse($request->getSession()->remove('azure_return_path', '/')); | ||||
|     } | ||||
| } | ||||
| @@ -0,0 +1,61 @@ | ||||
| <?php | ||||
|  | ||||
| /** | ||||
|  * Chill is a software for social workers. | ||||
|  * | ||||
|  * For the full copyright and license information, please view | ||||
|  * the LICENSE file that was distributed with this source code. | ||||
|  */ | ||||
|  | ||||
| declare(strict_types=1); | ||||
|  | ||||
| /* | ||||
|  * Chill is a software for social workers | ||||
|  * | ||||
|  * For the full copyright and license information, please view | ||||
|  * the LICENSE file that was distributed with this source code. | ||||
|  */ | ||||
|  | ||||
| namespace Chill\CalendarBundle\Controller; | ||||
|  | ||||
| use Chill\CalendarBundle\Messenger\Message\MSGraphChangeNotificationMessage; | ||||
| use JsonException; | ||||
| use Symfony\Component\HttpFoundation\Request; | ||||
| use Symfony\Component\HttpFoundation\Response; | ||||
| use Symfony\Component\HttpKernel\Exception\BadRequestHttpException; | ||||
| use Symfony\Component\Messenger\MessageBusInterface; | ||||
| use Symfony\Component\Routing\Annotation\Route; | ||||
| use const JSON_THROW_ON_ERROR; | ||||
|  | ||||
| class RemoteCalendarMSGraphSyncController | ||||
| { | ||||
|     private MessageBusInterface $messageBus; | ||||
|  | ||||
|     public function __construct(MessageBusInterface $messageBus) | ||||
|     { | ||||
|         $this->messageBus = $messageBus; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * @Route("/public/incoming-hook/calendar/msgraph/events/{userId}", name="chill_calendar_remote_msgraph_incoming_webhook_events", | ||||
|      * methods={"POST"}) | ||||
|      */ | ||||
|     public function webhookCalendarReceiver(int $userId, Request $request): Response | ||||
|     { | ||||
|         if ($request->query->has('validationToken')) { | ||||
|             return new Response($request->query->get('validationToken'), Response::HTTP_OK, [ | ||||
|                 'content-type' => 'text/plain', | ||||
|             ]); | ||||
|         } | ||||
|  | ||||
|         try { | ||||
|             $body = json_decode($request->getContent(), true, 512, JSON_THROW_ON_ERROR); | ||||
|         } catch (JsonException $e) { | ||||
|             throw new BadRequestHttpException('could not decode json', $e); | ||||
|         } | ||||
|  | ||||
|         $this->messageBus->dispatch(new MSGraphChangeNotificationMessage($body, $userId)); | ||||
|  | ||||
|         return new Response('', Response::HTTP_ACCEPTED); | ||||
|     } | ||||
| } | ||||
| @@ -0,0 +1,114 @@ | ||||
| <?php | ||||
|  | ||||
| /** | ||||
|  * Chill is a software for social workers. | ||||
|  * | ||||
|  * For the full copyright and license information, please view | ||||
|  * the LICENSE file that was distributed with this source code. | ||||
|  */ | ||||
|  | ||||
| declare(strict_types=1); | ||||
|  | ||||
| /* | ||||
|  * Chill is a software for social workers | ||||
|  * | ||||
|  * For the full copyright and license information, please view | ||||
|  * the LICENSE file that was distributed with this source code. | ||||
|  */ | ||||
|  | ||||
| namespace Chill\CalendarBundle\Controller; | ||||
|  | ||||
| use Chill\CalendarBundle\RemoteCalendar\Connector\RemoteCalendarConnectorInterface; | ||||
| use Chill\MainBundle\Entity\User; | ||||
| use Chill\MainBundle\Pagination\PaginatorFactory; | ||||
| use Chill\MainBundle\Serializer\Model\Collection; | ||||
| use DateTimeImmutable; | ||||
| use Symfony\Component\HttpFoundation\JsonResponse; | ||||
| use Symfony\Component\HttpFoundation\Request; | ||||
| use Symfony\Component\HttpFoundation\Response; | ||||
| use Symfony\Component\HttpKernel\Exception\BadRequestHttpException; | ||||
| use Symfony\Component\Routing\Annotation\Route; | ||||
| use Symfony\Component\Serializer\SerializerInterface; | ||||
| use function count; | ||||
|  | ||||
| /** | ||||
|  * Contains method to get events (Calendar) from remote calendar. | ||||
|  */ | ||||
| class RemoteCalendarProxyController | ||||
| { | ||||
|     private PaginatorFactory $paginatorFactory; | ||||
|  | ||||
|     private RemoteCalendarConnectorInterface $remoteCalendarConnector; | ||||
|  | ||||
|     private SerializerInterface $serializer; | ||||
|  | ||||
|     public function __construct(PaginatorFactory $paginatorFactory, RemoteCalendarConnectorInterface $remoteCalendarConnector, SerializerInterface $serializer) | ||||
|     { | ||||
|         $this->paginatorFactory = $paginatorFactory; | ||||
|         $this->remoteCalendarConnector = $remoteCalendarConnector; | ||||
|         $this->serializer = $serializer; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * @Route("api/1.0/calendar/proxy/calendar/by-user/{id}/events") | ||||
|      */ | ||||
|     public function listEventForCalendar(User $user, Request $request): Response | ||||
|     { | ||||
|         if (!$request->query->has('dateFrom')) { | ||||
|             throw new BadRequestHttpException('You must provide a dateFrom parameter'); | ||||
|         } | ||||
|  | ||||
|         if (false === $dateFrom = DateTimeImmutable::createFromFormat( | ||||
|             DateTimeImmutable::ATOM, | ||||
|             $request->query->get('dateFrom') | ||||
|         )) { | ||||
|             throw new BadRequestHttpException('dateFrom not parsable'); | ||||
|         } | ||||
|  | ||||
|         if (!$request->query->has('dateTo')) { | ||||
|             throw new BadRequestHttpException('You must provide a dateTo parameter'); | ||||
|         } | ||||
|  | ||||
|         if (false === $dateTo = DateTimeImmutable::createFromFormat( | ||||
|             DateTimeImmutable::ATOM, | ||||
|             $request->query->get('dateTo') | ||||
|         )) { | ||||
|             throw new BadRequestHttpException('dateTo not parsable'); | ||||
|         } | ||||
|  | ||||
|         $total = $this->remoteCalendarConnector->countEventsForUser($user, $dateFrom, $dateTo); | ||||
|         $paginator = $this->paginatorFactory->create($total); | ||||
|  | ||||
|         if (0 === $total) { | ||||
|             return new JsonResponse( | ||||
|                 $this->serializer->serialize(new Collection([], $paginator), 'json'), | ||||
|                 JsonResponse::HTTP_OK, | ||||
|                 [], | ||||
|                 true | ||||
|             ); | ||||
|         } | ||||
|  | ||||
|         $events = $this->remoteCalendarConnector->listEventsForUser( | ||||
|             $user, | ||||
|             $dateFrom, | ||||
|             $dateTo, | ||||
|             $paginator->getCurrentPageFirstItemNumber(), | ||||
|             $paginator->getItemsPerPage() | ||||
|         ); | ||||
|  | ||||
|         // in some case, we cannot paginate: we have to fetch all the items at once. We must avoid | ||||
|         // further requests by forcing the number of items returned. | ||||
|         if (count($events) > $paginator->getItemsPerPage()) { | ||||
|             $paginator->setItemsPerPage(count($events)); | ||||
|         } | ||||
|  | ||||
|         $collection = new Collection($events, $paginator); | ||||
|  | ||||
|         return new JsonResponse( | ||||
|             $this->serializer->serialize($collection, 'json', ['groups' => ['read']]), | ||||
|             JsonResponse::HTTP_OK, | ||||
|             [], | ||||
|             true | ||||
|         ); | ||||
|     } | ||||
| } | ||||
| @@ -12,12 +12,18 @@ declare(strict_types=1); | ||||
| namespace Chill\CalendarBundle\DataFixtures\ORM; | ||||
|  | ||||
| use Chill\CalendarBundle\Entity\CalendarRange; | ||||
| use Chill\MainBundle\Entity\Address; | ||||
| use Chill\MainBundle\Entity\Country; | ||||
| use Chill\MainBundle\Entity\Location; | ||||
| use Chill\MainBundle\Entity\LocationType; | ||||
| use Chill\MainBundle\Entity\PostalCode; | ||||
| use Chill\MainBundle\Repository\UserRepository; | ||||
| use DateTimeImmutable; | ||||
| use Doctrine\Bundle\FixturesBundle\Fixture; | ||||
| use Doctrine\Bundle\FixturesBundle\FixtureGroupInterface; | ||||
| use Doctrine\Common\DataFixtures\OrderedFixtureInterface; | ||||
| use Doctrine\Persistence\ObjectManager; | ||||
| use libphonenumber\PhoneNumberUtil; | ||||
|  | ||||
| class LoadCalendarRange extends Fixture implements FixtureGroupInterface, OrderedFixtureInterface | ||||
| { | ||||
| @@ -49,6 +55,24 @@ class LoadCalendarRange extends Fixture implements FixtureGroupInterface, Ordere | ||||
|  | ||||
|         $users = $this->userRepository->findAll(); | ||||
|  | ||||
|         $location = (new Location()) | ||||
|             ->setAddress($address = new Address()) | ||||
|             ->setName('Centre A') | ||||
|             ->setEmail('centreA@test.chill.social') | ||||
|             ->setLocationType($type = new LocationType()) | ||||
|             ->setPhonenumber1(PhoneNumberUtil::getInstance()->parse('+3287653812')); | ||||
|         $type->setTitle('Service'); | ||||
|         $address->setStreet('Rue des Épaules')->setStreetNumber('14') | ||||
|             ->setPostcode($postCode = new PostalCode()); | ||||
|         $postCode->setCode('4145')->setName('Houte-Si-Plout')->setCountry( | ||||
|             ($country = new Country())->setName(['fr' => 'Pays'])->setCountryCode('ZZ') | ||||
|         ); | ||||
|         $manager->persist($country); | ||||
|         $manager->persist($postCode); | ||||
|         $manager->persist($address); | ||||
|         $manager->persist($type); | ||||
|         $manager->persist($location); | ||||
|  | ||||
|         $days = [ | ||||
|             '2021-08-23', | ||||
|             '2021-08-24', | ||||
| @@ -58,6 +82,8 @@ class LoadCalendarRange extends Fixture implements FixtureGroupInterface, Ordere | ||||
|             '2021-08-31', | ||||
|             '2021-09-01', | ||||
|             '2021-09-02', | ||||
|             (new DateTimeImmutable('tomorrow'))->format('Y-m-d'), | ||||
|             (new DateTimeImmutable('today'))->format('Y-m-d'), | ||||
|         ]; | ||||
|  | ||||
|         $hours = [ | ||||
| @@ -76,7 +102,8 @@ class LoadCalendarRange extends Fixture implements FixtureGroupInterface, Ordere | ||||
|                     $calendarRange = (new CalendarRange()) | ||||
|                         ->setUser($u) | ||||
|                         ->setStartDate($startEvent) | ||||
|                         ->setEndDate($endEvent); | ||||
|                         ->setEndDate($endEvent) | ||||
|                         ->setLocation($location); | ||||
|  | ||||
|                     $manager->persist($calendarRange); | ||||
|                 } | ||||
|   | ||||
| @@ -37,6 +37,15 @@ class ChillCalendarExtension extends Extension implements PrependExtensionInterf | ||||
|         $loader->load('services/fixtures.yml'); | ||||
|         $loader->load('services/form.yml'); | ||||
|         $loader->load('services/event.yml'); | ||||
|         $loader->load('services/remote_calendar.yaml'); | ||||
|  | ||||
|         $container->setParameter('chill_calendar', $config); | ||||
|  | ||||
|         if ($config['short_messages']['enabled']) { | ||||
|             $container->setParameter('chill_calendar.short_messages', $config['short_messages']); | ||||
|         } else { | ||||
|             $container->setParameter('chill_calendar.short_messages', null); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     public function prepend(ContainerBuilder $container) | ||||
|   | ||||