From e6df3212328d2dd92ef7be57d3230ce442011de0 Mon Sep 17 00:00:00 2001 From: Boubaker Khanfir Date: Wed, 16 Oct 2024 16:43:48 +0100 Subject: [PATCH] feat: Add Space Template Management Backoffice - MEED-7607 - Meeds-io/MIPs#150 (#4118) This change will add a new portlet for Space Template Management with its Backend using spring Beans. --- .../service/TranslationService.java | 17 + .../social/core/space/SpaceTemplate.java | 3 + .../space/SpacesAdministrationService.java | 16 +- .../social/core/space/model/Space.java | 4 + .../social/core/space/spi/SpaceService.java | 8 + .../social/core/storage/api/SpaceStorage.java | 8 + .../meeds/social/space/constant/Priority.java | 25 + .../space/constant/PublicSiteVisibility.java | 25 + .../social/space/constant/Registration.java | 25 + .../social/space/constant/Visibility.java | 25 + .../space/template/dao/SpaceTemplateDAO.java | 36 + .../template/entity/SpaceTemplateEntity.java | 93 + .../space/template/model/SpaceTemplate.java | 67 + .../template/model/SpaceTemplateFilter.java | 38 + .../SpaceTemplateBannerAttachmentPlugin.java | 88 + .../SpaceTemplateTranslationPlugin.java | 83 + .../service/SpaceTemplateService.java | 220 ++ .../storage/SpaceTemplateStorage.java | 87 + .../space/template/utils/EntityMapper.java | 79 + .../service/TranslationServiceImpl.java | 48 +- .../jpa/storage/EntityConverterUtils.java | 23 +- .../jpa/storage/RDBMSSpaceStorageImpl.java | 49 +- .../social/core/jpa/storage/dao/SpaceDAO.java | 9 + .../jpa/storage/dao/jpa/SpaceDAOImpl.java | 31 +- .../dao/jpa/query/SpaceQueryBuilder.java | 5 +- .../core/jpa/storage/entity/SpaceEntity.java | 124 +- .../impl/DefaultSpaceApplicationHandler.java | 2 +- .../core/space/impl/SpaceServiceImpl.java | 88 +- .../impl/SpacesAdministrationServiceImpl.java | 15 + .../social/core/storage/cache/CacheType.java | 5 +- .../storage/cache/CachedSpaceStorage.java | 381 +-- .../cache/SocialStorageCacheService.java | 10 +- .../social-rdbms.db.changelog-1.0.0.xml | 35 + .../social-rdbms.db.changelog-master.xml | 29 - .../core/src/main/resources/jpa-entities.idx | 1 + ...aceTemplateBannerAttachmentPluginTest.java | 113 + .../SpaceTemplateTranslationPluginTest.java | 102 + .../service/SpaceTemplateServiceTest.java | 246 ++ .../storage/SpaceTemplateStorageTest.java | 184 ++ .../core/jpa/storage/dao/SpaceDAOTest.java | 9 +- .../jpa/storage/dao/SpaceMemberDAOTest.java | 9 +- .../core/test/NoContainerTestSuite.java | 8 + .../template/rest/SpaceTemplateRest.java | 164 + .../social/rest/impl/space/SpaceRest.java | 13 + .../spacetemplates/SpaceTemplatesRest.java | 6 +- .../activity/ActivityRestResourcesTest.java | 5 +- .../test/MockSpacesAdministrationService.java | 29 +- crowdin.yml | 78 +- translations.properties | 59 - webapp/package-lock.json | 2887 ++++++++++++++++- webapp/package.json | 2 +- .../io/meeds/social/SocialApplication.java | 42 + .../SpaceTemplatesManagement_en.properties | 33 + webapp/src/main/resources/social.properties | 18 + .../main/webapp/WEB-INF/gatein-resources.xml | 32 + webapp/src/main/webapp/WEB-INF/portlet.xml | 22 + .../webapp/html/spaceTemplateManagement.html | 9 + .../webapp/vue-apps/common/js/SpaceService.js | 13 + .../components/SpaceTemplatesManagement.vue | 43 + .../components/header/Toolbar.vue | 50 + .../components/list/SpaceTemplateItem.vue | 103 + .../components/list/SpaceTemplateItemMenu.vue | 158 + .../components/list/SpaceTemplates.vue | 169 + .../initComponents.js | 37 + .../js/SpaceTemplateService.js | 87 + .../space-templates-management/main.js | 57 + .../space-templates-management/services.js | 26 + webapp/webpack.common.js | 1 + 68 files changed, 6086 insertions(+), 530 deletions(-) create mode 100644 component/core/src/main/java/io/meeds/social/space/constant/Priority.java create mode 100644 component/core/src/main/java/io/meeds/social/space/constant/PublicSiteVisibility.java create mode 100644 component/core/src/main/java/io/meeds/social/space/constant/Registration.java create mode 100644 component/core/src/main/java/io/meeds/social/space/constant/Visibility.java create mode 100644 component/core/src/main/java/io/meeds/social/space/template/dao/SpaceTemplateDAO.java create mode 100644 component/core/src/main/java/io/meeds/social/space/template/entity/SpaceTemplateEntity.java create mode 100644 component/core/src/main/java/io/meeds/social/space/template/model/SpaceTemplate.java create mode 100644 component/core/src/main/java/io/meeds/social/space/template/model/SpaceTemplateFilter.java create mode 100644 component/core/src/main/java/io/meeds/social/space/template/plugin/attachment/SpaceTemplateBannerAttachmentPlugin.java create mode 100644 component/core/src/main/java/io/meeds/social/space/template/plugin/translation/SpaceTemplateTranslationPlugin.java create mode 100644 component/core/src/main/java/io/meeds/social/space/template/service/SpaceTemplateService.java create mode 100644 component/core/src/main/java/io/meeds/social/space/template/storage/SpaceTemplateStorage.java create mode 100644 component/core/src/main/java/io/meeds/social/space/template/utils/EntityMapper.java delete mode 100644 component/core/src/main/resources/db/changelog/social-rdbms.db.changelog-master.xml create mode 100644 component/core/src/test/java/io/meeds/social/space/template/plugin/attachment/SpaceTemplateBannerAttachmentPluginTest.java create mode 100644 component/core/src/test/java/io/meeds/social/space/template/plugin/translation/SpaceTemplateTranslationPluginTest.java create mode 100644 component/core/src/test/java/io/meeds/social/space/template/service/SpaceTemplateServiceTest.java create mode 100644 component/core/src/test/java/io/meeds/social/space/template/storage/SpaceTemplateStorageTest.java create mode 100644 component/service/src/main/java/io/meeds/social/space/template/rest/SpaceTemplateRest.java delete mode 100644 translations.properties create mode 100644 webapp/src/main/java/io/meeds/social/SocialApplication.java create mode 100644 webapp/src/main/resources/locale/portlet/SpaceTemplatesManagement_en.properties create mode 100644 webapp/src/main/resources/social.properties create mode 100644 webapp/src/main/webapp/html/spaceTemplateManagement.html create mode 100644 webapp/src/main/webapp/vue-apps/space-templates-management/components/SpaceTemplatesManagement.vue create mode 100644 webapp/src/main/webapp/vue-apps/space-templates-management/components/header/Toolbar.vue create mode 100644 webapp/src/main/webapp/vue-apps/space-templates-management/components/list/SpaceTemplateItem.vue create mode 100644 webapp/src/main/webapp/vue-apps/space-templates-management/components/list/SpaceTemplateItemMenu.vue create mode 100644 webapp/src/main/webapp/vue-apps/space-templates-management/components/list/SpaceTemplates.vue create mode 100644 webapp/src/main/webapp/vue-apps/space-templates-management/initComponents.js create mode 100644 webapp/src/main/webapp/vue-apps/space-templates-management/js/SpaceTemplateService.js create mode 100644 webapp/src/main/webapp/vue-apps/space-templates-management/main.js create mode 100644 webapp/src/main/webapp/vue-apps/space-templates-management/services.js diff --git a/component/api/src/main/java/io/meeds/social/translation/service/TranslationService.java b/component/api/src/main/java/io/meeds/social/translation/service/TranslationService.java index 33b3bae9498..66bb2effccd 100644 --- a/component/api/src/main/java/io/meeds/social/translation/service/TranslationService.java +++ b/component/api/src/main/java/io/meeds/social/translation/service/TranslationService.java @@ -87,6 +87,22 @@ String getTranslationLabel(String objectType, String fieldName, Locale locale); + /** + * Retrieves the Translation Label for a given field of an Object (identified + * by its type and id) with a designated {@link Locale}. not. + * + * @param objectType Object type for which the Translation Metadata will be + * attached + * @param objectId Object unique identifier + * @param fieldName Object field + * @param locale {@link Locale} + * @return the translated label for a given {@link Locale} + */ + String getTranslationLabelOrDefault(String objectType, + long objectId, + String fieldName, + Locale locale); + /** * Saves Translation Labels for a given Object's field. This will replace any * existing list of translations @@ -201,4 +217,5 @@ void deleteTranslationLabel(String objectType, * @param objectType Object type */ void removePlugin(String objectType); + } diff --git a/component/api/src/main/java/org/exoplatform/social/core/space/SpaceTemplate.java b/component/api/src/main/java/org/exoplatform/social/core/space/SpaceTemplate.java index 68319e66994..e319343d61e 100644 --- a/component/api/src/main/java/org/exoplatform/social/core/space/SpaceTemplate.java +++ b/component/api/src/main/java/org/exoplatform/social/core/space/SpaceTemplate.java @@ -28,10 +28,13 @@ /** * Definition of space template model. + * @deprecated will be replaced by a managed/stored Space Template */ @NoArgsConstructor @AllArgsConstructor +@Deprecated(forRemoval = true, since = "7.0") public class SpaceTemplate implements Cloneable { + private String name; private String resolvedLabel; diff --git a/component/api/src/main/java/org/exoplatform/social/core/space/SpacesAdministrationService.java b/component/api/src/main/java/org/exoplatform/social/core/space/SpacesAdministrationService.java index c461f951524..e944c9c87a4 100644 --- a/component/api/src/main/java/org/exoplatform/social/core/space/SpacesAdministrationService.java +++ b/component/api/src/main/java/org/exoplatform/social/core/space/SpacesAdministrationService.java @@ -1,10 +1,9 @@ package org.exoplatform.social.core.space; -import org.exoplatform.services.security.MembershipEntry; -import org.exoplatform.social.core.space.model.Space; - import java.util.List; +import org.exoplatform.services.security.MembershipEntry; + /** * Service to manage administration of spaces */ @@ -45,4 +44,15 @@ public interface SpacesAdministrationService { */ boolean canCreateSpace(String username) ; + /** + * Checks if the user is a super manager of all spaces + * + * @param username user name + * @return true if the user is member of super administrators groups, else + * false + */ + default boolean isSuperManager(String username) { + throw new UnsupportedOperationException(); + } + } diff --git a/component/api/src/main/java/org/exoplatform/social/core/space/model/Space.java b/component/api/src/main/java/org/exoplatform/social/core/space/model/Space.java index da1d86b8cf0..96812744cbe 100644 --- a/component/api/src/main/java/org/exoplatform/social/core/space/model/Space.java +++ b/component/api/src/main/java/org/exoplatform/social/core/space/model/Space.java @@ -67,6 +67,10 @@ public class Space implements CacheEntry { /** The template. */ private String template; + @Getter + @Setter + private long templateId; + /** The url. */ private String url; diff --git a/component/api/src/main/java/org/exoplatform/social/core/space/spi/SpaceService.java b/component/api/src/main/java/org/exoplatform/social/core/space/spi/SpaceService.java index a5efe90ede9..0ceedf92b7f 100644 --- a/component/api/src/main/java/org/exoplatform/social/core/space/spi/SpaceService.java +++ b/component/api/src/main/java/org/exoplatform/social/core/space/spi/SpaceService.java @@ -18,6 +18,7 @@ import java.time.Instant; import java.util.List; +import java.util.Map; import org.exoplatform.commons.exception.ObjectNotFoundException; import org.exoplatform.commons.utils.ListAccess; @@ -1255,4 +1256,11 @@ default String getSpacePublicSiteName(Space space) { throw new UnsupportedOperationException(); } + /** + * @return the count of spaces by Space Template identifier + */ + default Map countSpacesByTemplate() { + throw new UnsupportedOperationException(); + } + } diff --git a/component/api/src/main/java/org/exoplatform/social/core/storage/api/SpaceStorage.java b/component/api/src/main/java/org/exoplatform/social/core/storage/api/SpaceStorage.java index b114b125114..1d1b537c30b 100644 --- a/component/api/src/main/java/org/exoplatform/social/core/storage/api/SpaceStorage.java +++ b/component/api/src/main/java/org/exoplatform/social/core/storage/api/SpaceStorage.java @@ -25,6 +25,7 @@ import java.time.Instant; import java.util.List; +import java.util.Map; /** * @author Alain Defrance @@ -852,4 +853,11 @@ default int countCommonSpaces(String userId, String otherUserId){ default Instant getSpaceMembershipDate(long spaceId, String username) { throw new UnsupportedOperationException(); } + + /** + * @return the count of spaces by Space Template identifier + */ + default Map countSpacesByTemplate() { + throw new UnsupportedOperationException(); + } } diff --git a/component/core/src/main/java/io/meeds/social/space/constant/Priority.java b/component/core/src/main/java/io/meeds/social/space/constant/Priority.java new file mode 100644 index 00000000000..0d01f8dd464 --- /dev/null +++ b/component/core/src/main/java/io/meeds/social/space/constant/Priority.java @@ -0,0 +1,25 @@ +/* + * This file is part of the Meeds project (https://meeds.io/). + * + * Copyright (C) 2020 - 2024 Meeds Association contact@meeds.io + * + * This program is free software; you can redistribute it and/or + * modify it under the terms of the GNU Lesser General Public + * License as published by the Free Software Foundation; either + * version 3 of the License, or (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU + * Lesser General Public License for more details. + * + * You should have received a copy of the GNU Lesser General Public License + * along with this program; if not, write to the Free Software Foundation, + * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. + * + */ +package io.meeds.social.space.constant; + +public enum Priority { + HIGH, INTERMEDIATE, LOW +} diff --git a/component/core/src/main/java/io/meeds/social/space/constant/PublicSiteVisibility.java b/component/core/src/main/java/io/meeds/social/space/constant/PublicSiteVisibility.java new file mode 100644 index 00000000000..83710deb8c7 --- /dev/null +++ b/component/core/src/main/java/io/meeds/social/space/constant/PublicSiteVisibility.java @@ -0,0 +1,25 @@ +/* + * This file is part of the Meeds project (https://meeds.io/). + * + * Copyright (C) 2020 - 2024 Meeds Association contact@meeds.io + * + * This program is free software; you can redistribute it and/or + * modify it under the terms of the GNU Lesser General Public + * License as published by the Free Software Foundation; either + * version 3 of the License, or (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU + * Lesser General Public License for more details. + * + * You should have received a copy of the GNU Lesser General Public License + * along with this program; if not, write to the Free Software Foundation, + * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. + * + */ +package io.meeds.social.space.constant; + +public enum PublicSiteVisibility { + MANAGER, MEMBER, INTERNAL, AUTHENTICATED, EVERYONE; +} diff --git a/component/core/src/main/java/io/meeds/social/space/constant/Registration.java b/component/core/src/main/java/io/meeds/social/space/constant/Registration.java new file mode 100644 index 00000000000..609eebb7784 --- /dev/null +++ b/component/core/src/main/java/io/meeds/social/space/constant/Registration.java @@ -0,0 +1,25 @@ +/* + * This file is part of the Meeds project (https://meeds.io/). + * + * Copyright (C) 2020 - 2024 Meeds Association contact@meeds.io + * + * This program is free software; you can redistribute it and/or + * modify it under the terms of the GNU Lesser General Public + * License as published by the Free Software Foundation; either + * version 3 of the License, or (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU + * Lesser General Public License for more details. + * + * You should have received a copy of the GNU Lesser General Public License + * along with this program; if not, write to the Free Software Foundation, + * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. + * + */ +package io.meeds.social.space.constant; + +public enum Registration { + OPEN, VALIDATION, CLOSED +} \ No newline at end of file diff --git a/component/core/src/main/java/io/meeds/social/space/constant/Visibility.java b/component/core/src/main/java/io/meeds/social/space/constant/Visibility.java new file mode 100644 index 00000000000..ca83fef668e --- /dev/null +++ b/component/core/src/main/java/io/meeds/social/space/constant/Visibility.java @@ -0,0 +1,25 @@ +/* + * This file is part of the Meeds project (https://meeds.io/). + * + * Copyright (C) 2020 - 2024 Meeds Association contact@meeds.io + * + * This program is free software; you can redistribute it and/or + * modify it under the terms of the GNU Lesser General Public + * License as published by the Free Software Foundation; either + * version 3 of the License, or (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU + * Lesser General Public License for more details. + * + * You should have received a copy of the GNU Lesser General Public License + * along with this program; if not, write to the Free Software Foundation, + * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. + * + */ +package io.meeds.social.space.constant; + +public enum Visibility { + PUBLIC, PRIVATE, HIDDEN +} diff --git a/component/core/src/main/java/io/meeds/social/space/template/dao/SpaceTemplateDAO.java b/component/core/src/main/java/io/meeds/social/space/template/dao/SpaceTemplateDAO.java new file mode 100644 index 00000000000..314be208462 --- /dev/null +++ b/component/core/src/main/java/io/meeds/social/space/template/dao/SpaceTemplateDAO.java @@ -0,0 +1,36 @@ +/** + * This file is part of the Meeds project (https://meeds.io/). + * + * Copyright (C) 2020 - 2024 Meeds Association contact@meeds.io + * + * This program is free software; you can redistribute it and/or + * modify it under the terms of the GNU Lesser General Public + * License as published by the Free Software Foundation; either + * version 3 of the License, or (at your option) any later version. + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU + * Lesser General Public License for more details. + * + * You should have received a copy of the GNU Lesser General Public License + * along with this program; if not, write to the Free Software Foundation, + * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. + */ +package io.meeds.social.space.template.dao; + +import java.util.List; + +import org.springframework.data.domain.Pageable; +import org.springframework.data.jpa.repository.JpaRepository; +import org.springframework.stereotype.Repository; + +import io.meeds.social.space.template.entity.SpaceTemplateEntity; + +@Repository +public interface SpaceTemplateDAO extends JpaRepository { + + List findByDeletedFalse(Pageable pageable); + + List findByDeletedFalseAndEnabledTrue(Pageable pageable); + +} diff --git a/component/core/src/main/java/io/meeds/social/space/template/entity/SpaceTemplateEntity.java b/component/core/src/main/java/io/meeds/social/space/template/entity/SpaceTemplateEntity.java new file mode 100644 index 00000000000..1e41443c3aa --- /dev/null +++ b/component/core/src/main/java/io/meeds/social/space/template/entity/SpaceTemplateEntity.java @@ -0,0 +1,93 @@ +/** + * This file is part of the Meeds project (https://meeds.io/). + * + * Copyright (C) 2020 - 2024 Meeds Association contact@meeds.io + * + * This program is free software; you can redistribute it and/or + * modify it under the terms of the GNU Lesser General Public + * License as published by the Free Software Foundation; either + * version 3 of the License, or (at your option) any later version. + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU + * Lesser General Public License for more details. + * + * You should have received a copy of the GNU Lesser General Public License + * along with this program; if not, write to the Free Software Foundation, + * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. + */ +package io.meeds.social.space.template.entity; + +import java.util.List; + +import org.exoplatform.commons.utils.StringListConverter; + +import io.meeds.social.space.constant.Registration; +import io.meeds.social.space.constant.Visibility; + +import jakarta.persistence.Column; +import jakarta.persistence.Convert; +import jakarta.persistence.Entity; +import jakarta.persistence.GeneratedValue; +import jakarta.persistence.GenerationType; +import jakarta.persistence.Id; +import jakarta.persistence.SequenceGenerator; +import jakarta.persistence.Table; +import lombok.AllArgsConstructor; +import lombok.Data; +import lombok.NoArgsConstructor; + +@Entity(name = "SpaceTemplate") +@Table(name = "SOC_SPACE_TEMPLATES") +@Data +@NoArgsConstructor +@AllArgsConstructor +public class SpaceTemplateEntity { + + @Id + @SequenceGenerator(name = "SEQ_SOC_SPACE_TEMPLATE_ID", sequenceName = "SEQ_SOC_SPACE_TEMPLATE_ID", allocationSize = 1) + @GeneratedValue(strategy = GenerationType.AUTO, generator = "SEQ_SOC_SPACE_TEMPLATE_ID") + @Column(name = "ID") + private Long id; + + @Column(name = "ICON") + private String icon; + + @Column(name = "ENABLED") + private boolean enabled; + + @Column(name = "DELETED") + private boolean deleted; + + @Column(name = "IS_SYSTEM") + private boolean system; + + @Column(name = "TEMPLATE_ORDER") + private long order; + + @Convert(converter = StringListConverter.class) + @Column(name = "PERMISSIONS") + private List permissions; + + @Convert(converter = StringListConverter.class) + @Column(name = "SPACE_LAYOUT_PERMISSIONS") + private List spaceLayoutPermissions; + + @Convert(converter = StringListConverter.class) + @Column(name = "SPACE_DELETE_PERMISSIONS") + private List spaceDeletePermissions; + + @Convert(converter = StringListConverter.class) + @Column(name = "SPACE_FIELDS") + private List spaceFields; + + @Column(name = "SPACE_DEFAULT_VISIBILITY") + private Visibility spaceDefaultVisibility; + + @Column(name = "SPACE_DEFAULT_ACCESS") + private Registration spaceDefaultRegistration; + + @Column(name = "SPACE_ALLOW_CONTENT_CREATION") + private boolean spaceAllowContentCreation; + +} diff --git a/component/core/src/main/java/io/meeds/social/space/template/model/SpaceTemplate.java b/component/core/src/main/java/io/meeds/social/space/template/model/SpaceTemplate.java new file mode 100644 index 00000000000..d4bfcc00a68 --- /dev/null +++ b/component/core/src/main/java/io/meeds/social/space/template/model/SpaceTemplate.java @@ -0,0 +1,67 @@ +/** + * This file is part of the Meeds project (https://meeds.io/). + * + * Copyright (C) 2020 - 2024 Meeds Association contact@meeds.io + * + * This program is free software; you can redistribute it and/or + * modify it under the terms of the GNU Lesser General Public + * License as published by the Free Software Foundation; either + * version 3 of the License, or (at your option) any later version. + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU + * Lesser General Public License for more details. + * + * You should have received a copy of the GNU Lesser General Public License + * along with this program; if not, write to the Free Software Foundation, + * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. + */ +package io.meeds.social.space.template.model; + +import java.util.List; + +import io.meeds.social.space.constant.Registration; +import io.meeds.social.space.constant.Visibility; + +import lombok.AllArgsConstructor; +import lombok.Data; +import lombok.NoArgsConstructor; + +@Data +@NoArgsConstructor +@AllArgsConstructor +public class SpaceTemplate { + + private long id; + + private String name; + + private String description; + + private long bannerFileId; + + private String icon; + + private boolean enabled; + + private boolean deleted; + + private boolean system; + + private long order; + + private List permissions; + + private List spaceLayoutPermissions; + + private List spaceDeletePermissions; + + private List spaceFields; + + private Visibility spaceDefaultVisibility; + + private Registration spaceDefaultRegistration; + + private boolean spaceAllowContentCreation; + +} diff --git a/component/core/src/main/java/io/meeds/social/space/template/model/SpaceTemplateFilter.java b/component/core/src/main/java/io/meeds/social/space/template/model/SpaceTemplateFilter.java new file mode 100644 index 00000000000..5d7bd2c8ebd --- /dev/null +++ b/component/core/src/main/java/io/meeds/social/space/template/model/SpaceTemplateFilter.java @@ -0,0 +1,38 @@ +/** + * This file is part of the Meeds project (https://meeds.io/). + * + * Copyright (C) 2020 - 2024 Meeds Association contact@meeds.io + * + * This program is free software; you can redistribute it and/or + * modify it under the terms of the GNU Lesser General Public + * License as published by the Free Software Foundation; either + * version 3 of the License, or (at your option) any later version. + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU + * Lesser General Public License for more details. + * + * You should have received a copy of the GNU Lesser General Public License + * along with this program; if not, write to the Free Software Foundation, + * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. + */ +package io.meeds.social.space.template.model; + +import java.util.Locale; + +import lombok.AllArgsConstructor; +import lombok.Data; +import lombok.NoArgsConstructor; + +@Data +@NoArgsConstructor +@AllArgsConstructor +public class SpaceTemplateFilter { + + private String username; + + private Locale locale; + + private boolean includeDisabled; + +} diff --git a/component/core/src/main/java/io/meeds/social/space/template/plugin/attachment/SpaceTemplateBannerAttachmentPlugin.java b/component/core/src/main/java/io/meeds/social/space/template/plugin/attachment/SpaceTemplateBannerAttachmentPlugin.java new file mode 100644 index 00000000000..de551ab63b1 --- /dev/null +++ b/component/core/src/main/java/io/meeds/social/space/template/plugin/attachment/SpaceTemplateBannerAttachmentPlugin.java @@ -0,0 +1,88 @@ +/** + * This file is part of the Meeds project (https://meeds.io/). + * + * Copyright (C) 2020 - 2024 Meeds Association contact@meeds.io + * + * This program is free software; you can redistribute it and/or + * modify it under the terms of the GNU Lesser General Public + * License as published by the Free Software Foundation; either + * version 3 of the License, or (at your option) any later version. + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU + * Lesser General Public License for more details. + * + * You should have received a copy of the GNU Lesser General Public License + * along with this program; if not, write to the Free Software Foundation, + * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. + */ +package io.meeds.social.space.template.plugin.attachment; + +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.core.Ordered; +import org.springframework.core.annotation.Order; +import org.springframework.stereotype.Component; + +import org.exoplatform.commons.exception.ObjectNotFoundException; +import org.exoplatform.services.security.Identity; +import org.exoplatform.social.attachment.AttachmentPlugin; +import org.exoplatform.social.attachment.AttachmentService; + +import io.meeds.social.space.template.service.SpaceTemplateService; + +import jakarta.annotation.PostConstruct; + +@Component +@Order(Ordered.HIGHEST_PRECEDENCE) +public class SpaceTemplateBannerAttachmentPlugin extends AttachmentPlugin { + + public static final String OBJECT_TYPE = "spaceTemplateBanner"; + + @Autowired + protected SpaceTemplateService spaceTemplateService; + + @Autowired + protected AttachmentService attachmentService; + + @PostConstruct + public void init() { + attachmentService.addPlugin(this); + } + + @Override + public String getObjectType() { + return OBJECT_TYPE; + } + + @Override + public boolean hasEditPermission(Identity userIdentity, String entityId) throws ObjectNotFoundException { + if (userIdentity == null) { + return false; + } + return spaceTemplateService.canManageTemplates(userIdentity.getUserId()); + } + + @Override + public boolean hasAccessPermission(Identity userIdentity, String entityId) throws ObjectNotFoundException { + if (entityId == null || userIdentity == null) { + return false; + } + long templateId = Long.parseLong(entityId); + String username = userIdentity.getUserId(); + if (spaceTemplateService.getSpaceTemplate(templateId) == null) { + throw new ObjectNotFoundException(String.format("Space Template with id %s not found", templateId)); + } + return spaceTemplateService.canViewTemplate(templateId, username); + } + + @Override + public long getAudienceId(String objectId) throws ObjectNotFoundException { + return 0; + } + + @Override + public long getSpaceId(String objectId) throws ObjectNotFoundException { + return 0; + } + +} diff --git a/component/core/src/main/java/io/meeds/social/space/template/plugin/translation/SpaceTemplateTranslationPlugin.java b/component/core/src/main/java/io/meeds/social/space/template/plugin/translation/SpaceTemplateTranslationPlugin.java new file mode 100644 index 00000000000..65c2604d839 --- /dev/null +++ b/component/core/src/main/java/io/meeds/social/space/template/plugin/translation/SpaceTemplateTranslationPlugin.java @@ -0,0 +1,83 @@ +/** + * This file is part of the Meeds project (https://meeds.io/). + * + * Copyright (C) 2020 - 2024 Meeds Association contact@meeds.io + * + * This program is free software; you can redistribute it and/or + * modify it under the terms of the GNU Lesser General Public + * License as published by the Free Software Foundation; either + * version 3 of the License, or (at your option) any later version. + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU + * Lesser General Public License for more details. + * + * You should have received a copy of the GNU Lesser General Public License + * along with this program; if not, write to the Free Software Foundation, + * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. + */ +package io.meeds.social.space.template.plugin.translation; + +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.core.Ordered; +import org.springframework.core.annotation.Order; +import org.springframework.stereotype.Component; + +import org.exoplatform.commons.exception.ObjectNotFoundException; + +import io.meeds.social.space.template.service.SpaceTemplateService; +import io.meeds.social.translation.plugin.TranslationPlugin; +import io.meeds.social.translation.service.TranslationService; + +import jakarta.annotation.PostConstruct; + +@Component +@Order(Ordered.HIGHEST_PRECEDENCE) +public class SpaceTemplateTranslationPlugin extends TranslationPlugin { + + public static final String OBJECT_TYPE = "spaceTemplate"; + + public static final String DESCRIPTION_FIELD_NAME = "description"; + + public static final String TITLE_FIELD_NAME = "title"; + + @Autowired + protected SpaceTemplateService spaceTemplateService; + + @Autowired + protected TranslationService translationService; + + @PostConstruct + public void init() { + translationService.addPlugin(this); + } + + @Override + public String getObjectType() { + return OBJECT_TYPE; + } + + @Override + public boolean hasEditPermission(long templateId, String username) throws ObjectNotFoundException { + return spaceTemplateService.canManageTemplates(username); + } + + @Override + public boolean hasAccessPermission(long templateId, String username) throws ObjectNotFoundException { + if (spaceTemplateService.getSpaceTemplate(templateId) == null) { + throw new ObjectNotFoundException(String.format("Space Template with id %s not found", templateId)); + } + return spaceTemplateService.canViewTemplate(templateId, username); + } + + @Override + public long getAudienceId(long templateId) { + return 0; + } + + @Override + public long getSpaceId(long templateId) { + return 0; + } + +} diff --git a/component/core/src/main/java/io/meeds/social/space/template/service/SpaceTemplateService.java b/component/core/src/main/java/io/meeds/social/space/template/service/SpaceTemplateService.java new file mode 100644 index 00000000000..cfe601023f9 --- /dev/null +++ b/component/core/src/main/java/io/meeds/social/space/template/service/SpaceTemplateService.java @@ -0,0 +1,220 @@ +/** + * This file is part of the Meeds project (https://meeds.io/). + * + * Copyright (C) 2020 - 2024 Meeds Association contact@meeds.io + * + * This program is free software; you can redistribute it and/or + * modify it under the terms of the GNU Lesser General Public + * License as published by the Free Software Foundation; either + * version 3 of the License, or (at your option) any later version. + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU + * Lesser General Public License for more details. + * + * You should have received a copy of the GNU Lesser General Public License + * along with this program; if not, write to the Free Software Foundation, + * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. + */ +package io.meeds.social.space.template.service; + +import java.util.Collections; +import java.util.List; +import java.util.Locale; +import java.util.Objects; + +import org.apache.commons.collections4.CollectionUtils; +import org.apache.commons.lang3.StringUtils; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.data.domain.Pageable; +import org.springframework.stereotype.Service; + +import org.exoplatform.commons.exception.ObjectNotFoundException; +import org.exoplatform.portal.config.UserACL; +import org.exoplatform.services.log.ExoLogger; +import org.exoplatform.services.log.Log; +import org.exoplatform.services.security.Identity; +import org.exoplatform.services.security.MembershipEntry; +import org.exoplatform.social.attachment.AttachmentService; +import org.exoplatform.social.core.space.SpacesAdministrationService; + +import io.meeds.social.space.template.model.SpaceTemplate; +import io.meeds.social.space.template.model.SpaceTemplateFilter; +import io.meeds.social.space.template.plugin.attachment.SpaceTemplateBannerAttachmentPlugin; +import io.meeds.social.space.template.plugin.translation.SpaceTemplateTranslationPlugin; +import io.meeds.social.space.template.storage.SpaceTemplateStorage; +import io.meeds.social.translation.service.TranslationService; + +@Service +public class SpaceTemplateService { + + private static final Log LOG = ExoLogger.getLogger(SpaceTemplateService.class); + + @Autowired + protected TranslationService translationService; + + @Autowired + protected AttachmentService attachmentService; + + @Autowired + protected SpacesAdministrationService spacesAdministrationService; + + @Autowired + protected UserACL userAcl; + + @Autowired + protected SpaceTemplateStorage spaceTemplateStorage; + + public List getSpaceTemplates() { + return getSpaceTemplates(null, Pageable.unpaged(), false); + } + + public List getSpaceTemplates(SpaceTemplateFilter spaceTemplateFilter, Pageable pageable, boolean expand) { + if (spaceTemplateFilter != null + && StringUtils.isBlank(spaceTemplateFilter.getUsername())) { + return Collections.emptyList(); + } else { + boolean includeDisabled = spaceTemplateFilter == null || spaceTemplateFilter.isIncludeDisabled(); + List spaceTemplates = includeDisabled ? spaceTemplateStorage.getSpaceTemplates(pageable) : + spaceTemplateStorage.getEnabledSpaceTemplates(pageable); + return spaceTemplates.stream() + .map(spaceTemplate -> { + if (spaceTemplateFilter != null + && !canViewTemplate(spaceTemplate.getId(), spaceTemplateFilter.getUsername())) { + return null; + } else if (expand) { + computeSpaceTemplateAttributes(spaceTemplate, + spaceTemplateFilter == null ? null : + spaceTemplateFilter.getLocale()); + } + return spaceTemplate; + }) + .filter(Objects::nonNull) + .toList(); + } + } + + public SpaceTemplate getSpaceTemplate(long templateId) { + return spaceTemplateStorage.getSpaceTemplate(templateId); + } + + public SpaceTemplate getSpaceTemplate(long templateId, + String username, + Locale locale, + boolean expand) throws IllegalAccessException { + SpaceTemplate spaceTemplate = spaceTemplateStorage.getSpaceTemplate(templateId); + if (spaceTemplate == null) { + return null; + } + if (!canViewTemplate(templateId, username)) { + throw new IllegalAccessException(); + } + if (expand) { + computeSpaceTemplateAttributes(spaceTemplate, locale); + } + return spaceTemplate; + } + + public boolean canManageTemplates(String username) { + return spacesAdministrationService.isSuperManager(username); + } + + public boolean canViewTemplate(long templateId, String username) { + if (canManageTemplates(username)) { + return true; + } else if (userAcl.isAnonymousUser(username)) { + return false; + } + SpaceTemplate spaceTemplate = getSpaceTemplate(templateId); + Identity aclIdentity = userAcl.getUserIdentity(username); + return spaceTemplate != null + && aclIdentity != null + && spaceTemplate.getPermissions() + .stream() + .anyMatch(expression -> aclIdentity.isMemberOf(getMembershipEntry(expression))); + } + + public SpaceTemplate createSpaceTemplate(SpaceTemplate spaceTemplate, String username) throws IllegalAccessException { + if (!canManageTemplates(username)) { + throw new IllegalAccessException("User isn't authorized to create a space template"); + } + return createSpaceTemplate(spaceTemplate); + } + + public SpaceTemplate createSpaceTemplate(SpaceTemplate spaceTemplate) { + if (spaceTemplate.getId() != 0) { + throw new IllegalArgumentException("Space template to create shouldn't have an id"); + } + return spaceTemplateStorage.createSpaceTemplate(spaceTemplate); + } + + public SpaceTemplate updateSpaceTemplate(SpaceTemplate spaceTemplate, String username) throws ObjectNotFoundException, + IllegalAccessException { + if (!canManageTemplates(username)) { + throw new IllegalAccessException("User isn't authorized to update a space template"); + } else if (spaceTemplate.isDeleted()) { + throw new IllegalArgumentException("Can't mark space template as deleted through update method"); + } + return updateSpaceTemplate(spaceTemplate); + } + + public SpaceTemplate updateSpaceTemplate(SpaceTemplate spaceTemplate) throws ObjectNotFoundException { + SpaceTemplate storedSpaceTemplate = spaceTemplateStorage.getSpaceTemplate(spaceTemplate.getId()); + if (storedSpaceTemplate == null || storedSpaceTemplate.isDeleted()) { + throw new ObjectNotFoundException("Space Template doesn't exist"); + } + return spaceTemplateStorage.updateSpaceTemplate(spaceTemplate); + } + + public void deleteSpaceTemplate(long templateId, String username) throws IllegalAccessException, ObjectNotFoundException { + if (!canManageTemplates(username)) { + throw new IllegalAccessException("User isn't authorized to create a space template"); + } + SpaceTemplate spaceTemplate = getSpaceTemplate(templateId); + if (spaceTemplate != null && spaceTemplate.isSystem()) { + throw new IllegalAccessException("Can't delete a system space template"); + } + deleteSpaceTemplate(templateId); + } + + public void deleteSpaceTemplate(long templateId) throws ObjectNotFoundException { + SpaceTemplate spaceTemplate = spaceTemplateStorage.getSpaceTemplate(templateId); + if (spaceTemplate == null || spaceTemplate.isDeleted()) { + throw new ObjectNotFoundException(String.format("Space template with id %s doesn't exist", templateId)); + } + spaceTemplate.setDeleted(true); + spaceTemplateStorage.updateSpaceTemplate(spaceTemplate); + + try { + attachmentService.deleteAttachments(SpaceTemplateBannerAttachmentPlugin.OBJECT_TYPE, String.valueOf(templateId)); + } catch (Exception e) { + LOG.debug("Error while deleting attachments of deleted Page Template", e); + } + try { + translationService.deleteTranslationLabels(SpaceTemplateTranslationPlugin.OBJECT_TYPE, templateId); + } catch (ObjectNotFoundException e) { + LOG.debug("Error while deleting translation labels of deleted Page Template", e); + } + } + + private void computeSpaceTemplateAttributes(SpaceTemplate spaceTemplate, Locale locale) { + spaceTemplate.setName(translationService.getTranslationLabelOrDefault(SpaceTemplateTranslationPlugin.OBJECT_TYPE, + spaceTemplate.getId(), + SpaceTemplateTranslationPlugin.TITLE_FIELD_NAME, + locale)); + spaceTemplate.setDescription(translationService.getTranslationLabelOrDefault(SpaceTemplateTranslationPlugin.OBJECT_TYPE, + spaceTemplate.getId(), + SpaceTemplateTranslationPlugin.DESCRIPTION_FIELD_NAME, + locale)); + List attachmentFileIds = attachmentService.getAttachmentFileIds(SpaceTemplateBannerAttachmentPlugin.OBJECT_TYPE, + String.valueOf(spaceTemplate.getId())); + if (CollectionUtils.isNotEmpty(attachmentFileIds)) { + spaceTemplate.setBannerFileId(Long.parseLong(attachmentFileIds.get(0))); + } + } + + private MembershipEntry getMembershipEntry(String expression) { + return expression.contains(":") ? MembershipEntry.parse(expression) : new MembershipEntry(expression); + } + +} diff --git a/component/core/src/main/java/io/meeds/social/space/template/storage/SpaceTemplateStorage.java b/component/core/src/main/java/io/meeds/social/space/template/storage/SpaceTemplateStorage.java new file mode 100644 index 00000000000..adc8228af00 --- /dev/null +++ b/component/core/src/main/java/io/meeds/social/space/template/storage/SpaceTemplateStorage.java @@ -0,0 +1,87 @@ +/** + * This file is part of the Meeds project (https://meeds.io/). + * + * Copyright (C) 2020 - 2024 Meeds Association contact@meeds.io + * + * This program is free software; you can redistribute it and/or + * modify it under the terms of the GNU Lesser General Public + * License as published by the Free Software Foundation; either + * version 3 of the License, or (at your option) any later version. + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU + * Lesser General Public License for more details. + * + * You should have received a copy of the GNU Lesser General Public License + * along with this program; if not, write to the Free Software Foundation, + * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. + */ +package io.meeds.social.space.template.storage; + +import java.util.List; + +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.cache.annotation.CacheEvict; +import org.springframework.cache.annotation.Cacheable; +import org.springframework.data.domain.Pageable; +import org.springframework.stereotype.Component; + +import org.exoplatform.commons.exception.ObjectNotFoundException; + +import io.meeds.social.space.template.dao.SpaceTemplateDAO; +import io.meeds.social.space.template.entity.SpaceTemplateEntity; +import io.meeds.social.space.template.model.SpaceTemplate; +import io.meeds.social.space.template.utils.EntityMapper; + +@Component +public class SpaceTemplateStorage { + + @Autowired + protected SpaceTemplateDAO spaceTemplateDAO; + + @Cacheable(cacheNames = "social.spaceTemplates") + public List getSpaceTemplates(Pageable pageable) { + return spaceTemplateDAO.findByDeletedFalse(pageable) + .stream() + .map(EntityMapper::fromEntity) + .toList(); + } + + @Cacheable(cacheNames = "social.enabledSpaceTemplates") + public List getEnabledSpaceTemplates(Pageable pageable) { + return spaceTemplateDAO.findByDeletedFalseAndEnabledTrue(pageable) + .stream() + .map(EntityMapper::fromEntity) + .toList(); + } + + @Cacheable(cacheNames = "social.spaceTemplates") + public SpaceTemplate getSpaceTemplate(long id) { + return spaceTemplateDAO.findById(id) + .map(EntityMapper::fromEntity) + .orElse(null); + } + + @CacheEvict(cacheNames = { "social.spaceTemplates", "social.enabledSpaceTemplates" }, allEntries = true) + public SpaceTemplate createSpaceTemplate(SpaceTemplate spaceTemplate) { + SpaceTemplateEntity spaceTemplateEntity = EntityMapper.toEntity(spaceTemplate); + spaceTemplateEntity = spaceTemplateDAO.save(spaceTemplateEntity); + return EntityMapper.fromEntity(spaceTemplateEntity); + } + + @CacheEvict(cacheNames = { "social.spaceTemplates", "social.enabledSpaceTemplates" }, allEntries = true) + public SpaceTemplate updateSpaceTemplate(SpaceTemplate spaceTemplate) throws ObjectNotFoundException { + if (!spaceTemplateDAO.existsById(spaceTemplate.getId())) { + throw new ObjectNotFoundException("Space template doesn't exist"); + } + SpaceTemplateEntity spaceTemplateEntity = EntityMapper.toEntity(spaceTemplate); + spaceTemplateEntity = spaceTemplateDAO.save(spaceTemplateEntity); + return EntityMapper.fromEntity(spaceTemplateEntity); + } + + @CacheEvict(cacheNames = { "social.spaceTemplates", "social.enabledSpaceTemplates" }, allEntries = true) + public void deleteSpaceTemplate(long id) { + spaceTemplateDAO.deleteById(id); + } + +} diff --git a/component/core/src/main/java/io/meeds/social/space/template/utils/EntityMapper.java b/component/core/src/main/java/io/meeds/social/space/template/utils/EntityMapper.java new file mode 100644 index 00000000000..ddac44c2bb5 --- /dev/null +++ b/component/core/src/main/java/io/meeds/social/space/template/utils/EntityMapper.java @@ -0,0 +1,79 @@ +/** + * This file is part of the Meeds project (https://meeds.io/). + * + * Copyright (C) 2020 - 2024 Meeds Association contact@meeds.io + * + * This program is free software; you can redistribute it and/or + * modify it under the terms of the GNU Lesser General Public + * License as published by the Free Software Foundation; either + * version 3 of the License, or (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU + * Lesser General Public License for more details. + * + * You should have received a copy of the GNU Lesser General Public License + * along with this program; if not, write to the Free Software Foundation, + * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. + */ +package io.meeds.social.space.template.utils; + +import java.util.Collections; +import java.util.List; + +import org.apache.commons.lang3.StringUtils; + +import io.meeds.social.space.template.entity.SpaceTemplateEntity; +import io.meeds.social.space.template.model.SpaceTemplate; + +public class EntityMapper { + + private EntityMapper() { + // Utils Class + } + + public static SpaceTemplate fromEntity(SpaceTemplateEntity entity) { + return new SpaceTemplate(entity.getId() == null ? 0 : entity.getId().longValue(), + null, + null, + 0l, + entity.getIcon(), + entity.isEnabled(), + entity.isDeleted(), + entity.isSystem(), + entity.getOrder(), + getNonEmptyValueList(entity.getPermissions()), + getNonEmptyValueList(entity.getSpaceLayoutPermissions()), + getNonEmptyValueList(entity.getSpaceDeletePermissions()), + getNonEmptyValueList(entity.getSpaceFields()), + entity.getSpaceDefaultVisibility(), + entity.getSpaceDefaultRegistration(), + entity.isSpaceAllowContentCreation()); + } + + public static SpaceTemplateEntity toEntity(SpaceTemplate model) { + return new SpaceTemplateEntity(model.getId() == 0 ? null : model.getId(), + model.getIcon(), + model.isEnabled(), + model.isDeleted(), + model.isSystem(), + model.getOrder(), + getNonEmptyValueList(model.getPermissions()), + getNonEmptyValueList(model.getSpaceLayoutPermissions()), + getNonEmptyValueList(model.getSpaceDeletePermissions()), + getNonEmptyValueList(model.getSpaceFields()), + model.getSpaceDefaultVisibility(), + model.getSpaceDefaultRegistration(), + model.isSpaceAllowContentCreation()); + } + + private static List getNonEmptyValueList(List list) { + if (list == null) { + return Collections.emptyList(); + } else { + return list.stream().filter(StringUtils::isNotBlank).toList(); + } + } + +} diff --git a/component/core/src/main/java/io/meeds/social/translation/service/TranslationServiceImpl.java b/component/core/src/main/java/io/meeds/social/translation/service/TranslationServiceImpl.java index aab739a778c..4288dbecec5 100644 --- a/component/core/src/main/java/io/meeds/social/translation/service/TranslationServiceImpl.java +++ b/component/core/src/main/java/io/meeds/social/translation/service/TranslationServiceImpl.java @@ -28,6 +28,7 @@ import org.exoplatform.services.listener.ListenerService; import org.exoplatform.services.log.ExoLogger; import org.exoplatform.services.log.Log; +import org.exoplatform.services.resources.LocaleConfigService; import io.meeds.social.translation.model.TranslationEvent; import io.meeds.social.translation.model.TranslationField; @@ -47,13 +48,18 @@ public class TranslationServiceImpl implements TranslationService { private static final String NO_PERMISSION_TO_DELETE_MESSAGE = "User %s doesn't have enough permissions to delete translations of object of type %s identified by %s"; + private LocaleConfigService localeConfigService; + private ListenerService listenerService; private TranslationStorage translationStorage; private Map translationPlugins = new HashMap<>(); - public TranslationServiceImpl(TranslationStorage translationStorage, ListenerService listenerService) { + public TranslationServiceImpl(TranslationStorage translationStorage, + LocaleConfigService localeConfigService, + ListenerService listenerService) { + this.localeConfigService = localeConfigService; this.translationStorage = translationStorage; this.listenerService = listenerService; } @@ -70,9 +76,9 @@ public void removePlugin(String objectType) { @Override public TranslationField getTranslationField(String objectType, - long objectId, - String fieldName, - String username) throws IllegalAccessException, ObjectNotFoundException { + long objectId, + String fieldName, + String username) throws IllegalAccessException, ObjectNotFoundException { checkParameters(objectType, objectId, fieldName); checkAccessPermission(objectType, objectId, username); @@ -81,8 +87,8 @@ public TranslationField getTranslationField(String objectType, @Override public TranslationField getTranslationField(String objectType, - long objectId, - String fieldName) { + long objectId, + String fieldName) { return translationStorage.getTranslationField(objectType, objectId, fieldName); } @@ -94,6 +100,32 @@ public String getTranslationLabel(String objectType, return translationStorage.getTranslationLabel(objectType, objectId, fieldName, locale); } + @Override + public String getTranslationLabelOrDefault(String objectType, + long objectId, + String fieldName, + Locale locale) { + if (locale == null) { + locale = localeConfigService.getDefaultLocaleConfig().getLocale(); + } + TranslationField translationField = getTranslationField(objectType, + objectId, + fieldName); + if (translationField != null && MapUtils.isNotEmpty(translationField.getLabels())) { + String label = translationField.getLabels().get(locale); + if (label == null) { + Locale defaultLocale = localeConfigService.getDefaultLocaleConfig().getLocale(); + label = translationField.getLabels().get(defaultLocale); + } + if (label == null) { + label = translationField.getLabels().values().iterator().next(); + } + return label; + } else { + return null; + } + } + @Override public void saveTranslationLabels(String objectType, long objectId, @@ -198,7 +230,9 @@ private void checkAccessPermission(String objectType, long objectId, String user } } - private void checkEditPermission(String objectType, long objectId, String username, + private void checkEditPermission(String objectType, + long objectId, + String username, String message) throws ObjectNotFoundException, IllegalAccessException { TranslationPlugin translationPlugin = translationPlugins.get(objectType); if (!translationPlugin.hasEditPermission(objectId, username)) { diff --git a/component/core/src/main/java/org/exoplatform/social/core/jpa/storage/EntityConverterUtils.java b/component/core/src/main/java/org/exoplatform/social/core/jpa/storage/EntityConverterUtils.java index 866bd35497f..9938e006e21 100644 --- a/component/core/src/main/java/org/exoplatform/social/core/jpa/storage/EntityConverterUtils.java +++ b/component/core/src/main/java/org/exoplatform/social/core/jpa/storage/EntityConverterUtils.java @@ -33,16 +33,16 @@ import org.exoplatform.social.core.jpa.storage.entity.IdentityEntity; import org.exoplatform.social.core.jpa.storage.entity.ProfileExperienceEntity; import org.exoplatform.social.core.jpa.storage.entity.SpaceEntity; -import org.exoplatform.social.core.jpa.storage.entity.SpaceEntity.PRIORITY; -import org.exoplatform.social.core.jpa.storage.entity.SpaceEntity.PublicSiteVisibility; -import org.exoplatform.social.core.jpa.storage.entity.SpaceEntity.REGISTRATION; -import org.exoplatform.social.core.jpa.storage.entity.SpaceEntity.VISIBILITY; import org.exoplatform.social.core.jpa.storage.entity.SpaceMemberEntity; import org.exoplatform.social.core.relationship.model.Relationship; import org.exoplatform.social.core.service.LinkProvider; import org.exoplatform.social.core.space.model.Space; +import io.meeds.social.space.constant.Priority; +import io.meeds.social.space.constant.PublicSiteVisibility; +import io.meeds.social.space.constant.Registration; import io.meeds.social.space.constant.SpaceMembershipStatus; +import io.meeds.social.space.constant.Visibility; public class EntityConverterUtils { @@ -282,27 +282,28 @@ public static SpaceEntity buildFrom(Space space, SpaceEntity spaceEntity) { spaceEntity.setCreatedDate(space.getCreatedTime() > 0 ? new Date(space.getCreatedTime()) : new Date()); spaceEntity.setDescription(space.getDescription()); spaceEntity.setTemplate(space.getTemplate()); + spaceEntity.setTemplateId(space.getTemplateId() == 0 ? null : space.getTemplateId()); spaceEntity.setDisplayName(space.getDisplayName()); spaceEntity.setGroupId(space.getGroupId()); spaceEntity.setPrettyName(space.getPrettyName()); spaceEntity.setPublicSiteId(space.getPublicSiteId()); spaceEntity.setPublicSiteVisibility(space.getPublicSiteVisibility() == null ? PublicSiteVisibility.MANAGER : PublicSiteVisibility.valueOf(space.getPublicSiteVisibility().toUpperCase())); - PRIORITY priority = null; + Priority priority = null; if (Space.HIGH_PRIORITY.equals(space.getPriority())) { - priority = PRIORITY.HIGH; + priority = Priority.HIGH; } else if (Space.INTERMEDIATE_PRIORITY.equals(space.getPriority())) { - priority = PRIORITY.INTERMEDIATE; + priority = Priority.INTERMEDIATE; } else if (Space.LOW_PRIORITY.equals(space.getPriority())) { - priority = PRIORITY.LOW; + priority = Priority.LOW; } spaceEntity.setPriority(priority); if (space.getRegistration() != null) { - spaceEntity.setRegistration(REGISTRATION.valueOf(space.getRegistration().toUpperCase())); + spaceEntity.setRegistration(Registration.valueOf(space.getRegistration().toUpperCase())); } spaceEntity.setUrl(space.getUrl()); - VISIBILITY visibility = null; + Visibility visibility = null; if (space.getVisibility() != null) { - visibility = VISIBILITY.valueOf(space.getVisibility().toUpperCase()); + visibility = Visibility.valueOf(space.getVisibility().toUpperCase()); } spaceEntity.setVisibility(visibility); buildMembers(space, spaceEntity); diff --git a/component/core/src/main/java/org/exoplatform/social/core/jpa/storage/RDBMSSpaceStorageImpl.java b/component/core/src/main/java/org/exoplatform/social/core/jpa/storage/RDBMSSpaceStorageImpl.java index 833a15a0d0f..90bb350e4c0 100644 --- a/component/core/src/main/java/org/exoplatform/social/core/jpa/storage/RDBMSSpaceStorageImpl.java +++ b/component/core/src/main/java/org/exoplatform/social/core/jpa/storage/RDBMSSpaceStorageImpl.java @@ -26,6 +26,7 @@ import java.util.HashSet; import java.util.LinkedList; import java.util.List; +import java.util.Map; import java.util.Set; import java.util.stream.Collectors; @@ -653,6 +654,7 @@ public int countPendingSpaceRequestsToManage(String username) { return spaceMemberDAO.countPendingSpaceRequestsToManage(username); } + @Override public List findSpaceExternalInvitationsBySpaceId(String spaceId) { return spaceExternalInvitationDAO.findSpaceExternalInvitationsBySpaceId(spaceId) .stream() @@ -660,6 +662,7 @@ public List findSpaceExternalInvitationsBySpaceId(Strin .toList(); } + @Override public void saveSpaceExternalInvitation(String spaceId, String email, String tokenId) { SpaceExternalInvitationEntity spaceExternalInvitation = new SpaceExternalInvitationEntity(); spaceExternalInvitation.setSpaceId(spaceId); @@ -669,25 +672,52 @@ public void saveSpaceExternalInvitation(String spaceId, String email, String tok spaceExternalInvitationDAO.create(spaceExternalInvitation); } + @Override public SpaceExternalInvitation findSpaceExternalInvitationById(String invitationId) { SpaceExternalInvitationEntity spaceExternalInvitationEntity = spaceExternalInvitationDAO.find(Long.parseLong(invitationId)); return fillSpaceExternalInvitationFromEntity(spaceExternalInvitationEntity); } + @Override public void deleteSpaceExternalInvitation(SpaceExternalInvitation spaceExternalInvitation) { SpaceExternalInvitationEntity spaceExternalInvitationEntity = spaceExternalInvitationDAO.find(spaceExternalInvitation.getInvitationId()); spaceExternalInvitationDAO.delete(spaceExternalInvitationEntity); } + @Override public List findExternalInvitationsSpacesByEmail(String email) { return spaceExternalInvitationDAO.findExternalInvitationsSpacesByEmail(email); } + @Override public void deleteExternalUserInvitations(String email) { spaceExternalInvitationDAO.deleteExternalUserInvitations(email); } + @Override + public int countExternalMembers(Long spaceId) { + return spaceMemberDAO.countExternalMembers(spaceId); + } + + @Override + public List getCommonSpaces(String userId, String otherUserId, int offset, int limit) { + List commonSpaces = spaceDAO.getCommonSpaces(userId, otherUserId, offset, limit); + return commonSpaces.stream() + .map(this::fillSpaceFromEntity) + .toList(); + } + + @Override + public int countCommonSpaces(String userId, String otherUserId) { + return spaceDAO.countCommonSpaces(userId, otherUserId); + } + + @Override + public Map countSpacesByTemplate() { + return spaceDAO.countSpacesByTemplate(); + } + private String[] getSpaceMembers(long spaceId, SpaceMembershipStatus status) { int countSpaceMembers = spaceMemberDAO.countSpaceMembers(spaceId, status); if (countSpaceMembers == 0) { @@ -863,6 +893,7 @@ private Space fillSpaceSimpleFromEntity(SpaceEntity entity, Space space) { } space.setDescription(entity.getDescription()); space.setTemplate(entity.getTemplate()); + space.setTemplateId(entity.getTemplateId() == null ? 0 : entity.getTemplateId().longValue()); if (entity.getVisibility() != null) { space.setVisibility(entity.getVisibility().name().toLowerCase()); } @@ -929,24 +960,6 @@ public void setIdentityStorage(IdentityStorage identityStorage) { this.identityStorage = identityStorage; } - @Override - public int countExternalMembers(Long spaceId) { - return spaceMemberDAO.countExternalMembers(spaceId); - } - - @Override - public List getCommonSpaces(String userId, String otherUserId, int offset, int limit) { - List commonSpaces = spaceDAO.getCommonSpaces(userId, otherUserId, offset, limit); - return commonSpaces.stream() - .map(this::fillSpaceFromEntity) - .toList(); - } - - @Override - public int countCommonSpaces(String userId, String otherUserId) { - return spaceDAO.countCommonSpaces(userId, otherUserId); - } - private Instant computeCreatedDate(Token token) { return token == null ? null : Instant.ofEpochMilli(token.getExpirationTimeMillis()) diff --git a/component/core/src/main/java/org/exoplatform/social/core/jpa/storage/dao/SpaceDAO.java b/component/core/src/main/java/org/exoplatform/social/core/jpa/storage/dao/SpaceDAO.java index a5edb974b17..028b3a1ced8 100644 --- a/component/core/src/main/java/org/exoplatform/social/core/jpa/storage/dao/SpaceDAO.java +++ b/component/core/src/main/java/org/exoplatform/social/core/jpa/storage/dao/SpaceDAO.java @@ -17,6 +17,7 @@ package org.exoplatform.social.core.jpa.storage.dao; import java.util.List; +import java.util.Map; import org.exoplatform.commons.api.persistence.GenericDAO; import org.exoplatform.social.core.jpa.storage.entity.SpaceEntity; @@ -57,4 +58,12 @@ default List getCommonSpaces(String userId,String otherUserId, int default int countCommonSpaces(String userId,String otherUserId) { throw new UnsupportedOperationException(); } + + /** + * @return the count of spaces by Space Template identifier + */ + default Map countSpacesByTemplate() { + throw new UnsupportedOperationException(); + } + } diff --git a/component/core/src/main/java/org/exoplatform/social/core/jpa/storage/dao/jpa/SpaceDAOImpl.java b/component/core/src/main/java/org/exoplatform/social/core/jpa/storage/dao/jpa/SpaceDAOImpl.java index a7b483bc35a..b0e70e48b03 100644 --- a/component/core/src/main/java/org/exoplatform/social/core/jpa/storage/dao/jpa/SpaceDAOImpl.java +++ b/component/core/src/main/java/org/exoplatform/social/core/jpa/storage/dao/jpa/SpaceDAOImpl.java @@ -19,10 +19,13 @@ import java.util.Collections; import java.util.List; +import java.util.Map; import java.util.stream.Collectors; import jakarta.persistence.*; +import org.apache.commons.collections4.CollectionUtils; + import org.exoplatform.commons.persistence.impl.GenericDAOJPAImpl; import org.exoplatform.social.core.jpa.storage.dao.SpaceDAO; import org.exoplatform.social.core.jpa.storage.entity.SpaceEntity; @@ -39,7 +42,7 @@ public List getLastSpaces(int limit) { } else { return resultList.stream() .map(tuple -> tuple.get(0, Long.class)) - .collect(Collectors.toList()); + .toList(); } } @@ -48,7 +51,7 @@ public SpaceEntity getSpaceByGroupId(String groupId) { TypedQuery query = getEntityManager().createNamedQuery("SpaceEntity.getSpaceByGroupId", SpaceEntity.class); query.setParameter("groupId", groupId); try { - return query.getSingleResult(); + return query.getSingleResult(); } catch (NoResultException ex) { return null; } @@ -59,7 +62,7 @@ public SpaceEntity getSpaceByURL(String url) { TypedQuery query = getEntityManager().createNamedQuery("SpaceEntity.getSpaceByURL", SpaceEntity.class); query.setParameter("url", url); try { - return query.getSingleResult(); + return query.getSingleResult(); } catch (NoResultException ex) { return null; } @@ -70,7 +73,7 @@ public SpaceEntity getSpaceByDisplayName(String spaceDisplayName) { TypedQuery query = getEntityManager().createNamedQuery("SpaceEntity.getSpaceByDisplayName", SpaceEntity.class); query.setParameter("displayName", spaceDisplayName); try { - return query.getSingleResult(); + return query.getSingleResult(); } catch (NoResultException ex) { return null; } @@ -82,7 +85,7 @@ public SpaceEntity getSpaceByPrettyName(String spacePrettyName) { query.setParameter("prettyName", spacePrettyName); query.setMaxResults(1); try { - return query.getSingleResult(); + return query.getSingleResult(); } catch (NoResultException ex) { return null; } @@ -103,7 +106,7 @@ public List getCommonSpaces(String userId, String otherUserId, int throw new IllegalArgumentException("limit must be > 0"); } TypedQuery query = getEntityManager().createNamedQuery("SpaceEntity.getCommonSpacesBetweenTwoUsers", - SpaceEntity.class); + SpaceEntity.class); query.setParameter("userId", userId); query.setParameter("otherUserId", otherUserId); query.setFirstResult(offset); @@ -122,10 +125,24 @@ public int countCommonSpaces(String userId, String otherUserId) { } TypedQuery query = getEntityManager().createNamedQuery("SpaceEntity.countCommonSpacesBetweenTwoUsers", - Long.class); + Long.class); query.setParameter("userId", userId); query.setParameter("otherUserId", otherUserId); return query.getSingleResult().intValue(); } + + @Override + public Map countSpacesByTemplate() { + TypedQuery query = getEntityManager().createNamedQuery("SpaceEntity.countSpacesByTemplate", + Tuple.class); + List resultList = query.getResultList(); + if (CollectionUtils.isEmpty(resultList)) { + return Collections.emptyMap(); + } else { + return resultList.stream() + .collect(Collectors.toMap(t -> t.get(0, Long.class), t -> t.get(1, Long.class))); + } + } + } diff --git a/component/core/src/main/java/org/exoplatform/social/core/jpa/storage/dao/jpa/query/SpaceQueryBuilder.java b/component/core/src/main/java/org/exoplatform/social/core/jpa/storage/dao/jpa/query/SpaceQueryBuilder.java index 24e53f75eb1..48756b6ca56 100644 --- a/component/core/src/main/java/org/exoplatform/social/core/jpa/storage/dao/jpa/query/SpaceQueryBuilder.java +++ b/component/core/src/main/java/org/exoplatform/social/core/jpa/storage/dao/jpa/query/SpaceQueryBuilder.java @@ -35,6 +35,7 @@ import org.exoplatform.social.core.space.model.Space; import io.meeds.social.space.constant.SpaceMembershipStatus; +import io.meeds.social.space.constant.Visibility; import jakarta.persistence.EntityManager; import jakarta.persistence.Tuple; @@ -145,7 +146,7 @@ private Predicate buildPredicateFilter(Root root, CriteriaQuery spaceFilter.getRemoteId())); boolean includePrivate = spaceFilter.isIncludePrivate(); if (includePrivate) { - predicates.add(cb.or(cb.equal(root.get(SpaceEntity_.visibility), SpaceEntity.VISIBILITY.PRIVATE), tmp)); + predicates.add(cb.or(cb.equal(root.get(SpaceEntity_.visibility), Visibility.PRIVATE), tmp)); } else { predicates.add(tmp); } @@ -175,7 +176,7 @@ private Predicate buildPredicateFilter(Root root, CriteriaQuery //not hidden boolean notHidden = spaceFilter.isNotHidden(); if (notHidden) { - predicates.add(cb.notEqual(root.get(SpaceEntity_.visibility), SpaceEntity.VISIBILITY.HIDDEN)); + predicates.add(cb.notEqual(root.get(SpaceEntity_.visibility), Visibility.HIDDEN)); } // diff --git a/component/core/src/main/java/org/exoplatform/social/core/jpa/storage/entity/SpaceEntity.java b/component/core/src/main/java/org/exoplatform/social/core/jpa/storage/entity/SpaceEntity.java index 4f4b752db5b..0d8ec407972 100644 --- a/component/core/src/main/java/org/exoplatform/social/core/jpa/storage/entity/SpaceEntity.java +++ b/component/core/src/main/java/org/exoplatform/social/core/jpa/storage/entity/SpaceEntity.java @@ -14,7 +14,6 @@ * You should have received a copy of the GNU Affero General Public License * along with this program. If not, see . */ - package org.exoplatform.social.core.jpa.storage.entity; import java.io.Serializable; @@ -22,41 +21,67 @@ import java.util.HashSet; import java.util.Set; -import jakarta.persistence.*; +import io.meeds.social.space.constant.Priority; +import io.meeds.social.space.constant.PublicSiteVisibility; +import io.meeds.social.space.constant.Registration; +import io.meeds.social.space.constant.Visibility; + +import jakarta.persistence.CascadeType; +import jakarta.persistence.CollectionTable; +import jakarta.persistence.Column; +import jakarta.persistence.ElementCollection; +import jakarta.persistence.Entity; +import jakarta.persistence.FetchType; +import jakarta.persistence.GeneratedValue; +import jakarta.persistence.GenerationType; +import jakarta.persistence.Id; +import jakarta.persistence.JoinColumn; +import jakarta.persistence.NamedQuery; +import jakarta.persistence.OneToMany; +import jakarta.persistence.SequenceGenerator; +import jakarta.persistence.Table; +import jakarta.persistence.Temporal; +import jakarta.persistence.TemporalType; import lombok.Getter; import lombok.Setter; @Entity(name = "SocSpaceEntity") @Table(name = "SOC_SPACES") -@NamedQueries({ - @NamedQuery(name = "SpaceEntity.getLastSpaces", query = "SELECT sp.id, sp.createdDate FROM SocSpaceEntity sp ORDER BY sp.createdDate DESC"), - @NamedQuery(name = "SpaceEntity.getSpaceByGroupId", query = "SELECT sp FROM SocSpaceEntity sp WHERE sp.groupId = :groupId"), - @NamedQuery(name = "SpaceEntity.getSpaceByPrettyName", query = "SELECT sp FROM SocSpaceEntity sp WHERE sp.prettyName = :prettyName"), - @NamedQuery(name = "SpaceEntity.getSpaceByDisplayName", query = "SELECT sp FROM SocSpaceEntity sp WHERE sp.displayName = :displayName"), - @NamedQuery(name = "SpaceEntity.getSpaceByURL", query = "SELECT sp FROM SocSpaceEntity sp WHERE sp.url = :url"), - @NamedQuery( - name = "SpaceEntity.getCommonSpacesBetweenTwoUsers", - query = - "SELECT spaces FROM SocSpaceEntity spaces " - + "WHERE spaces.id IN ( " - + "SELECT distinct (t1.space.id) FROM SocSpaceMember t1, SocSpaceMember t2 " - + " WHERE t1.userId = :userId " - + " AND t2.userId = :otherUserId " - + " AND t1.space.id = t2.space.id" - + " )" - ), - @NamedQuery( - name = "SpaceEntity.countCommonSpacesBetweenTwoUsers", - query = - "SELECT COUNT(*) FROM SocSpaceEntity spaces " - + "WHERE spaces.id IN ( " - + "SELECT distinct (t1.space.id) FROM SocSpaceMember t1, SocSpaceMember t2 " - + " WHERE t1.userId = :userId " - + " AND t2.userId = :otherUserId " - + " AND t1.space.id = t2.space.id" - + " )" - ), -}) +@NamedQuery(name = "SpaceEntity.getLastSpaces", query = "SELECT sp.id, sp.createdDate FROM SocSpaceEntity sp ORDER BY sp.createdDate DESC") +@NamedQuery(name = "SpaceEntity.getSpaceByGroupId", query = "SELECT sp FROM SocSpaceEntity sp WHERE sp.groupId = :groupId") +@NamedQuery(name = "SpaceEntity.getSpaceByPrettyName", query = "SELECT sp FROM SocSpaceEntity sp WHERE sp.prettyName = :prettyName") +@NamedQuery(name = "SpaceEntity.getSpaceByDisplayName", query = "SELECT sp FROM SocSpaceEntity sp WHERE sp.displayName = :displayName") +@NamedQuery(name = "SpaceEntity.getSpaceByURL", query = "SELECT sp FROM SocSpaceEntity sp WHERE sp.url = :url") +@NamedQuery( + name = "SpaceEntity.getCommonSpacesBetweenTwoUsers", + query = + "SELECT spaces FROM SocSpaceEntity spaces " + + "WHERE spaces.id IN ( " + + "SELECT distinct (t1.space.id) FROM SocSpaceMember t1, SocSpaceMember t2 " + + " WHERE t1.userId = :userId " + + " AND t2.userId = :otherUserId " + + " AND t1.space.id = t2.space.id" + + " )" +) +@NamedQuery( + name = "SpaceEntity.countCommonSpacesBetweenTwoUsers", + query = + "SELECT COUNT(*) FROM SocSpaceEntity spaces " + + "WHERE spaces.id IN ( " + + "SELECT distinct (t1.space.id) FROM SocSpaceMember t1, SocSpaceMember t2 " + + " WHERE t1.userId = :userId " + + " AND t2.userId = :otherUserId " + + " AND t1.space.id = t2.space.id" + + " )" +) +@NamedQuery( + name = "SpaceEntity.countSpacesByTemplate", + query = """ + SELECT s.templateId, COUNT(s.id) FROM SocSpaceEntity s + WHERE s.templateId > 0 + GROUP BY s.templateId + """ +) public class SpaceEntity implements Serializable { private static final long serialVersionUID = 3223615477747436986L; @@ -91,7 +116,7 @@ public class SpaceEntity implements Serializable { private String displayName; @Column(name = "REGISTRATION") - private REGISTRATION registration; + private Registration registration; @Column(name = "DESCRIPTION") private String description; @@ -105,10 +130,10 @@ public class SpaceEntity implements Serializable { private Date bannerLastUpdated; @Column(name = "VISIBILITY") - public VISIBILITY visibility; + public Visibility visibility; @Column(name = "PRIORITY") - public PRIORITY priority; + public Priority priority; @Column(name = "GROUP_ID") public String groupId; @@ -119,6 +144,11 @@ public class SpaceEntity implements Serializable { @Column(name = "TEMPLATE") private String template; + @Getter + @Setter + @Column(name = "TEMPLATE_ID") + private Long templateId; + @Temporal(TemporalType.TIMESTAMP) @Column(name = "CREATED_DATE", nullable = false) private Date createdDate = new Date(); @@ -169,11 +199,11 @@ public void setDisplayName(String displayName) { this.displayName = displayName; } - public REGISTRATION getRegistration() { + public Registration getRegistration() { return registration; } - public void setRegistration(REGISTRATION registration) { + public void setRegistration(Registration registration) { this.registration = registration; } @@ -201,19 +231,19 @@ public void setBannerLastUpdated(Date bannerLastUpdated) { this.bannerLastUpdated = bannerLastUpdated; } - public VISIBILITY getVisibility() { + public Visibility getVisibility() { return visibility; } - public void setVisibility(VISIBILITY visibility) { + public void setVisibility(Visibility visibility) { this.visibility = visibility; } - public PRIORITY getPriority() { + public Priority getPriority() { return priority; } - public void setPriority(PRIORITY priority) { + public void setPriority(Priority priority) { this.priority = priority; } @@ -298,19 +328,5 @@ public int hashCode() { return id == null ? 0 : id.intValue(); } - public enum VISIBILITY { - PUBLIC, PRIVATE, HIDDEN - } - public enum PRIORITY { - HIGH, INTERMEDIATE, LOW - } - - public enum REGISTRATION { - OPEN, VALIDATION, CLOSED - } - - public static enum PublicSiteVisibility { - MANAGER, MEMBER, INTERNAL, AUTHENTICATED, EVERYONE; - } } diff --git a/component/core/src/main/java/org/exoplatform/social/core/space/impl/DefaultSpaceApplicationHandler.java b/component/core/src/main/java/org/exoplatform/social/core/space/impl/DefaultSpaceApplicationHandler.java index abb262310f2..e360544630e 100644 --- a/component/core/src/main/java/org/exoplatform/social/core/space/impl/DefaultSpaceApplicationHandler.java +++ b/component/core/src/main/java/org/exoplatform/social/core/space/impl/DefaultSpaceApplicationHandler.java @@ -345,7 +345,7 @@ public void restoreApplicationLayout(Space space, String appId) throws Exception NavigationService navService = ExoContainerContext.getService(NavigationService.class); NavigationContext navContext = SpaceUtils.getGroupNavigationContext(space.getGroupId()); - NodeContext> parentNodeCtx = navService.loadNode(NodeModel.SELF_MODEL, navContext, Scope.CHILDREN, null); + NodeContext> parentNodeCtx = navService.loadNode(NodeModel.SELF_MODEL, navContext, Scope.ALL, null); String spaceTemplateName = space.getTemplate(); SpaceTemplate spaceTemplate = spaceTemplateService.getSpaceTemplateByName(spaceTemplateName); diff --git a/component/core/src/main/java/org/exoplatform/social/core/space/impl/SpaceServiceImpl.java b/component/core/src/main/java/org/exoplatform/social/core/space/impl/SpaceServiceImpl.java index 153f2c9fbdb..de22e058c21 100644 --- a/component/core/src/main/java/org/exoplatform/social/core/space/impl/SpaceServiceImpl.java +++ b/component/core/src/main/java/org/exoplatform/social/core/space/impl/SpaceServiceImpl.java @@ -21,6 +21,7 @@ import java.util.Collection; import java.util.Collections; import java.util.List; +import java.util.Map; import java.util.Objects; import java.util.Set; import java.util.stream.Collectors; @@ -42,9 +43,6 @@ import org.exoplatform.services.organization.Group; import org.exoplatform.services.organization.GroupHandler; import org.exoplatform.services.organization.OrganizationService; -import org.exoplatform.services.security.Authenticator; -import org.exoplatform.services.security.IdentityConstants; -import org.exoplatform.services.security.IdentityRegistry; import org.exoplatform.social.core.binding.model.GroupSpaceBinding; import org.exoplatform.social.core.identity.model.Identity; import org.exoplatform.social.core.identity.model.Profile; @@ -100,11 +98,7 @@ public class SpaceServiceImpl implements SpaceService { private OrganizationService organizationService; - private UserACL userACL; - - private IdentityRegistry identityRegistry; - - private Authenticator authenticator; + private UserACL userAcl; private SpaceLifecycle spaceLifeCycle = new SpaceLifecycle(); @@ -118,25 +112,21 @@ public class SpaceServiceImpl implements SpaceService { private LayoutService layoutService; public SpaceServiceImpl(SpaceStorage spaceStorage, // NOSONAR - SpaceSearchConnector spaceSearchConnector, GroupSpaceBindingStorage groupSpaceBindingStorage, + SpaceSearchConnector spaceSearchConnector, IdentityManager identityManager, - UserACL userACL, - IdentityRegistry identityRegistry, - Authenticator authenticator, SpacesAdministrationService spacesAdministrationService, SpaceTemplateService spaceTemplateService, - LayoutService layoutService) { + LayoutService layoutService, + UserACL userAcl) { this.spaceStorage = spaceStorage; + this.groupSpaceBindingStorage = groupSpaceBindingStorage; this.spaceSearchConnector = spaceSearchConnector; this.identityManager = identityManager; - this.identityRegistry = identityRegistry; - this.userACL = userACL; - this.authenticator = authenticator; this.spacesAdministrationService = spacesAdministrationService; this.spaceTemplateService = spaceTemplateService; - this.groupSpaceBindingStorage = groupSpaceBindingStorage; this.layoutService = layoutService; + this.userAcl = userAcl; } @Override @@ -343,10 +333,9 @@ public void deleteSpace(Space space, boolean deleteGroup) throws SpaceException } if (deleteGroup) { - UserACL acl = getUserACL(); GroupHandler groupHandler = getOrgService().getGroupHandler(); Group deletedGroup = groupHandler.findGroupById(space.getGroupId()); - List mandatories = acl.getMandatoryGroups(); + List mandatories = userAcl.getMandatoryGroups(); if (deletedGroup != null && !isMandatory(groupHandler, deletedGroup, mandatories)) { SpaceUtils.removeGroup(space); } @@ -700,12 +689,12 @@ public boolean canAccessSpacePublicSite(Space space, String username) { return false; } else if (StringUtils.equals(space.getPublicSiteVisibility(), SpaceUtils.EVERYONE)) { return true; - } else if (userACL.isAnonymousUser(username)) { + } else if (userAcl.isAnonymousUser(username)) { return false; } else if (StringUtils.equals(space.getPublicSiteVisibility(), SpaceUtils.AUTHENTICATED)) { return true; } else if (StringUtils.equals(space.getPublicSiteVisibility(), SpaceUtils.INTERNAL)) { - return userACL.getUserIdentity(username).isMemberOf(SpaceUtils.PLATFORM_USERS_GROUP); + return userAcl.getUserIdentity(username).isMemberOf(SpaceUtils.PLATFORM_USERS_GROUP); } else if (StringUtils.equals(space.getPublicSiteVisibility(), SpaceUtils.MEMBER)) { return canViewSpace(space, username); } else if (StringUtils.equals(space.getPublicSiteVisibility(), SpaceUtils.MANAGER)) { @@ -1030,41 +1019,28 @@ public void deleteExternalUserInvitations(String email) { @Override public boolean isSuperManager(String username) { - if (StringUtils.isBlank(username) - || IdentityConstants.ANONIM.equals(username) - || IdentityConstants.SYSTEM.equals(username)) { - return false; - } else if (username.equals(getUserACL().getSuperUser())) { - return true; - } - org.exoplatform.services.security.Identity identity = getIdentity(username); - return identity != null && (identity.isMemberOf(userACL.getAdminGroups()) - || spacesAdministrationService.getSpacesAdministratorsMemberships() - .stream() - .anyMatch(identity::isMemberOf)); + return spacesAdministrationService.isSuperManager(username); } @Override public boolean isContentManager(String username) { - if (StringUtils.isBlank(username) || IdentityConstants.ANONIM.equals(username) || IdentityConstants.SYSTEM.equals(username)) { + if (userAcl.isAnonymousUser(username)) { return false; - } else if (username.equals(getUserACL().getSuperUser())) { + } else if (isSuperManager(username)) { return true; } - org.exoplatform.services.security.Identity identity = getIdentity(username); + org.exoplatform.services.security.Identity identity = userAcl.getUserIdentity(username); return identity != null && identity.isMemberOf(SpaceUtils.PLATFORM_PUBLISHER_GROUP, SpaceUtils.MANAGER); } @Override public boolean isContentPublisher(String username) { - if (StringUtils.isBlank(username) - || IdentityConstants.ANONIM.equals(username) - || IdentityConstants.SYSTEM.equals(username)) { + if (userAcl.isAnonymousUser(username)) { return false; - } else if (username.equals(getUserACL().getSuperUser())) { + } else if (isSuperManager(username)) { return true; } - org.exoplatform.services.security.Identity identity = getIdentity(username); + org.exoplatform.services.security.Identity identity = userAcl.getUserIdentity(username); return identity != null && identity.isMemberOf(SpaceUtils.PLATFORM_PUBLISHER_GROUP, SpaceUtils.PUBLISHER); } @@ -1073,6 +1049,11 @@ public ListAccess getCommonSpaces(String username, String otherUserId) { return new SpaceListAccess(spaceStorage, spaceSearchConnector, SpaceListAccessType.COMMON, username, otherUserId); } + @Override + public Map countSpacesByTemplate() { + return spaceStorage.countSpacesByTemplate(); + } + public void addSpaceListener(SpaceListenerPlugin plugin) { registerSpaceLifeCycleListener(plugin); } @@ -1127,15 +1108,6 @@ private OrganizationService getOrgService() { return organizationService; } - /** - * Gets UserACL - * - * @return userACL - */ - private UserACL getUserACL() { - return userACL; - } - /** * Gets space application handler * @@ -1251,22 +1223,6 @@ private Space removeInvited(Space space, String username) { return space; } - private org.exoplatform.services.security.Identity getIdentity(String username) { - org.exoplatform.services.security.Identity identity = identityRegistry.getIdentity(username); - if (identity == null) { - try { - identity = authenticator.createIdentity(username); - identityRegistry.register(identity); - return identity; - } catch (Exception e) { - LOG.warn("Error while retrieving user {} ACL identity", username, e); - return null; - } - } else { - return identity; - } - } - private SpaceTemplate getSpaceTemplateOrDefault(String templaceName) { SpaceTemplate spaceTemplate = spaceTemplateService.getSpaceTemplateByName(templaceName); if (spaceTemplate == null) { diff --git a/component/core/src/main/java/org/exoplatform/social/core/space/impl/SpacesAdministrationServiceImpl.java b/component/core/src/main/java/org/exoplatform/social/core/space/impl/SpacesAdministrationServiceImpl.java index 7dbff68ce34..9ef9d4878cb 100644 --- a/component/core/src/main/java/org/exoplatform/social/core/space/impl/SpacesAdministrationServiceImpl.java +++ b/component/core/src/main/java/org/exoplatform/social/core/space/impl/SpacesAdministrationServiceImpl.java @@ -144,6 +144,21 @@ public void updateSpacesCreatorsMemberships(List permissionsExp updateSpacesAdministrationPagePermissions(this.spaceCreatorsMemberships); } + @Override + public boolean isSuperManager(String username) { + if (StringUtils.isBlank(username) + || IdentityConstants.ANONIM.equals(username) + || IdentityConstants.SYSTEM.equals(username)) { + return false; + } else if (username.equals(userACL.getSuperUser())) { + return true; + } + org.exoplatform.services.security.Identity identity = userACL.getUserIdentity(username); + return identity != null && (identity.isMemberOf(userACL.getAdminGroups()) + || getSpacesAdministratorsMemberships().stream() + .anyMatch(identity::isMemberOf)); + } + /** * Load Spaces Administration settings * For both Spaces Administrators and Spaces Creators settings, it uses the value stored in the settings if any, diff --git a/component/core/src/main/java/org/exoplatform/social/core/storage/cache/CacheType.java b/component/core/src/main/java/org/exoplatform/social/core/storage/cache/CacheType.java index ba55c2e88b4..81893858fe5 100644 --- a/component/core/src/main/java/org/exoplatform/social/core/storage/cache/CacheType.java +++ b/component/core/src/main/java/org/exoplatform/social/core/storage/cache/CacheType.java @@ -62,12 +62,11 @@ public enum CacheType { SPACE("social.SpaceCache"), SPACE_REF("social.SpaceRefCache"), SPACES_COUNT("social.SpacesCountCache"), + SPACES_COUNT_BY_TEMPLATE("social.SpacesCountByTemplateCache"), SPACES("social.SpacesCache"), // - SPACE_SIMPLE("social.SpaceSimpleCache") - - ; + SPACE_SIMPLE("social.SpaceSimpleCache"); private final String name; diff --git a/component/core/src/main/java/org/exoplatform/social/core/storage/cache/CachedSpaceStorage.java b/component/core/src/main/java/org/exoplatform/social/core/storage/cache/CachedSpaceStorage.java index 8a9cd3bf104..284bfddc672 100644 --- a/component/core/src/main/java/org/exoplatform/social/core/storage/cache/CachedSpaceStorage.java +++ b/component/core/src/main/java/org/exoplatform/social/core/storage/cache/CachedSpaceStorage.java @@ -47,21 +47,25 @@ */ public class CachedSpaceStorage extends RDBMSSpaceStorageImpl { + private static final SpaceKey EMPTY_SPACE_KEY = new SpaceKey("0"); + /** Logger */ private static final Log LOG = ExoLogger.getLogger(CachedSpaceStorage.class); - private final ExoCache exoSpaceCache; - private final ExoCache exoSpaceSimpleCache; - private final ExoCache exoRefSpaceCache; - private final ExoCache exoSpacesCountCache; - private final ExoCache exoSpacesCache; - private final ExoCache exoIdentitiesCache; - - private final FutureExoCache> spaceCache; - private final FutureExoCache> spaceSimpleCache; - private final FutureExoCache> spaceRefCache; - private final FutureExoCache> spacesCountCache; - private final FutureExoCache> spacesCache; + private final ExoCache spaceCache; + private final ExoCache spaceSimpleCache; + private final ExoCache spaceRefCache; + private final ExoCache spacesCountCache; + private final ExoCache> spacesCountByTemplateCache; + private final ExoCache spacesCache; + private final ExoCache identitiesCache; + + private final FutureExoCache> spaceFutureCache; + private final FutureExoCache> spaceSimpleFutureCache; + private final FutureExoCache> spaceRefFutureCache; + private final FutureExoCache> spacesCountFutureCache; + private final FutureExoCache, ServiceContext>> spacesCountByTemplateFutureCache; + private final FutureExoCache> spacesFutureCache; private SocialStorageCacheService cacheService; private CachedActivityStorage cachedActivityStorage; @@ -198,18 +202,20 @@ public CachedSpaceStorage(SpaceDAO spaceDAO, super(spaceDAO, spaceMemberDAO, identityStorage, identityDAO, activityDAO, spaceExternalInvitationDAO, favoriteService, remindPasswordTokenService); this.cacheService = cacheService; - this.exoSpaceCache = cacheService.getSpaceCache(); - this.exoSpaceSimpleCache = cacheService.getSpaceSimpleCache(); - this.exoRefSpaceCache = cacheService.getSpaceRefCache(); - this.exoSpacesCountCache = cacheService.getSpacesCountCache(); - this.exoSpacesCache = cacheService.getSpacesCache(); - this.exoIdentitiesCache = cacheService.getIdentitiesCache(); - - this.spaceCache = CacheType.SPACE.createFutureCache(exoSpaceCache); - this.spaceSimpleCache = CacheType.SPACE_SIMPLE.createFutureCache(exoSpaceSimpleCache); - this.spaceRefCache = CacheType.SPACE_REF.createFutureCache(exoRefSpaceCache); - this.spacesCountCache = CacheType.SPACES_COUNT.createFutureCache(exoSpacesCountCache); - this.spacesCache = CacheType.SPACES.createFutureCache(exoSpacesCache); + this.spaceCache = cacheService.getSpaceCache(); + this.spaceSimpleCache = cacheService.getSpaceSimpleCache(); + this.spaceRefCache = cacheService.getSpaceRefCache(); + this.spacesCountCache = cacheService.getSpacesCountCache(); + this.spacesCache = cacheService.getSpacesCache(); + this.identitiesCache = cacheService.getIdentitiesCache(); + this.spacesCountByTemplateCache = cacheService.getSpacesCountByTemplateCache(); + + this.spaceFutureCache = CacheType.SPACE.createFutureCache(spaceCache); + this.spaceSimpleFutureCache = CacheType.SPACE_SIMPLE.createFutureCache(spaceSimpleCache); + this.spaceRefFutureCache = CacheType.SPACE_REF.createFutureCache(spaceRefCache); + this.spacesCountFutureCache = CacheType.SPACES_COUNT.createFutureCache(spacesCountCache); + this.spacesCountByTemplateFutureCache = CacheType.SPACES_COUNT_BY_TEMPLATE.createFutureCache(spacesCountByTemplateCache); + this.spacesFutureCache = CacheType.SPACES.createFutureCache(spacesCache); } @@ -217,21 +223,21 @@ private void cleanRef(Space removed) { if (removed == null) { return; } - exoRefSpaceCache.remove(new SpaceRefKey(removed.getDisplayName())); - exoRefSpaceCache.remove(new SpaceRefKey(null, removed.getPrettyName())); - exoRefSpaceCache.remove(new SpaceRefKey(null, null, removed.getGroupId())); - exoRefSpaceCache.remove(new SpaceRefKey(null, null, null, removed.getUrl())); - spaceRefCache.remove(new SpaceRefKey(removed.getDisplayName())); spaceRefCache.remove(new SpaceRefKey(null, removed.getPrettyName())); spaceRefCache.remove(new SpaceRefKey(null, null, removed.getGroupId())); spaceRefCache.remove(new SpaceRefKey(null, null, null, removed.getUrl())); + + spaceRefFutureCache.remove(new SpaceRefKey(removed.getDisplayName())); + spaceRefFutureCache.remove(new SpaceRefKey(null, removed.getPrettyName())); + spaceRefFutureCache.remove(new SpaceRefKey(null, null, removed.getGroupId())); + spaceRefFutureCache.remove(new SpaceRefKey(null, null, null, removed.getUrl())); } void clearIdentityCache() { try { - exoIdentitiesCache.select(new IdentityCacheSelector(SpaceIdentityProvider.NAME)); + identitiesCache.select(new IdentityCacheSelector(SpaceIdentityProvider.NAME)); } catch (Exception e) { LOG.error("Error deleting cache entries of provider type 'Space Identities'", e); @@ -242,8 +248,9 @@ void clearIdentityCache() { void clearSpaceCache() { try { - exoSpacesCache.clearCache(); - exoSpacesCountCache.clearCache(); + spacesCache.clearCache(); + spacesCountCache.clearCache(); + spacesCountByTemplateCache.clearCache(); } catch (Exception e) { LOG.error("Error deleting space caches", e); @@ -251,16 +258,14 @@ void clearSpaceCache() { } - /** - * {@inheritDoc} - */ + @Override public Space getSpaceByDisplayName(final String spaceDisplayName) throws SpaceStorageException { // SpaceRefKey refKey = new SpaceRefKey(spaceDisplayName); // - SpaceKey key = spaceRefCache.get( + SpaceKey key = spaceRefFutureCache.get( new ServiceContext() { public SpaceKey execute() { Space space = CachedSpaceStorage.super.getSpaceByDisplayName(spaceDisplayName); @@ -284,9 +289,7 @@ public SpaceKey execute() { } - /** - * {@inheritDoc} - */ + @Override public void saveSpace(final Space space, final boolean isNew) throws SpaceStorageException { // @@ -294,8 +297,8 @@ public void saveSpace(final Space space, final boolean isNew) throws SpaceStorag // - exoSpaceSimpleCache.remove(new SpaceKey(space.getId())); - exoSpaceCache.remove(new SpaceKey(space.getId())); + spaceSimpleCache.remove(new SpaceKey(space.getId())); + spaceCache.remove(new SpaceKey(space.getId())); clearSpaceCache(); clearIdentityCache(); @@ -328,19 +331,17 @@ public void renameSpace(Space space, String newDisplayName) throws SpaceStorageE cachedActivityStorage.clearOwnerStreamCache(oldPrettyName); // remove space cached - exoSpaceCache.remove(new SpaceKey(space.getId())); + spaceCache.remove(new SpaceKey(space.getId())); clearSpaceCache(); clearIdentityCache(); cleanRef(space); - exoRefSpaceCache.remove(new SpaceRefKey(oldDisplayName)); - exoRefSpaceCache.remove(new SpaceRefKey(null, oldPrettyName)); - exoRefSpaceCache.remove(new SpaceRefKey(null, null, space.getGroupId())); - exoRefSpaceCache.remove(new SpaceRefKey(null, null, null, oldUrl)); + spaceRefCache.remove(new SpaceRefKey(oldDisplayName)); + spaceRefCache.remove(new SpaceRefKey(null, oldPrettyName)); + spaceRefCache.remove(new SpaceRefKey(null, null, space.getGroupId())); + spaceRefCache.remove(new SpaceRefKey(null, null, null, oldUrl)); } - /** - * {@inheritDoc} - */ + @Override public void deleteSpace(final String id) throws SpaceStorageException { // @@ -348,7 +349,7 @@ public void deleteSpace(final String id) throws SpaceStorageException { super.deleteSpace(id); // - exoSpaceCache.remove(new SpaceKey(id)); + spaceCache.remove(new SpaceKey(id)); clearSpaceCache(); cleanRef(space); @@ -360,8 +361,8 @@ public void deleteSpace(final String id) throws SpaceStorageException { @Override public void ignoreSpace(String spaceId, String userId) { super.ignoreSpace(spaceId, userId); - exoSpaceSimpleCache.remove(new SpaceKey(spaceId)); - SpaceData spaceData = exoSpaceCache.remove(new SpaceKey(spaceId)); + spaceSimpleCache.remove(new SpaceKey(spaceId)); + SpaceData spaceData = spaceCache.remove(new SpaceKey(spaceId)); if (spaceData != null) { Space space = spaceData.build(); clearSpaceCache(); @@ -373,13 +374,6 @@ public void ignoreSpace(String spaceId, String userId) { } @Override - public boolean isSpaceIgnored(String spaceId, String userId) { - return super.isSpaceIgnored(spaceId, userId); - } - - /** - * {@inheritDoc} - */ public List getManagerSpacesByFilter( final String userId, final SpaceFilter spaceFilter, @@ -387,18 +381,15 @@ public List getManagerSpacesByFilter( final long limit) { SpaceFilterKey key = new SpaceFilterKey(userId, spaceFilter, SpaceType.MANAGER); ListSpacesKey listKey = new ListSpacesKey(key, offset, limit); - ListSpacesData keys = spacesCache.get( - new ServiceContext() { - public ListSpacesData execute() { - if (limit == 0) { - return buildIds(Collections.emptyList()); - } - List got = CachedSpaceStorage.super.getManagerSpacesByFilter(userId, - spaceFilter, - offset, - limit); - return buildIds(got); + ListSpacesData keys = spacesFutureCache.get(() -> { + if (limit == 0) { + return buildIds(Collections.emptyList()); } + List got = CachedSpaceStorage.super.getManagerSpacesByFilter(userId, + spaceFilter, + offset, + limit); + return buildIds(got); }, listKey); return buildSpaces(keys); @@ -408,26 +399,20 @@ public ListSpacesData execute() { @Override public int getManagerSpacesByFilterCount(String userId, SpaceFilter spaceFilter) { SpaceFilterKey key = new SpaceFilterKey(userId, spaceFilter, SpaceType.MANAGER); - return spacesCountCache.get( - new ServiceContext() { - public IntegerData execute() { - return new IntegerData(CachedSpaceStorage.super.getManagerSpacesByFilterCount(userId, spaceFilter)); - } - }, + return spacesCountFutureCache.get(() -> new IntegerData(CachedSpaceStorage.super.getManagerSpacesByFilterCount(userId, + spaceFilter)), key) .build(); } - /** - * {@inheritDoc} - */ + @Override public int getMemberSpacesByFilterCount(final String userId, final SpaceFilter spaceFilter) { // SpaceFilterKey key = new SpaceFilterKey(userId, spaceFilter, SpaceType.MEMBER); // - return spacesCountCache.get( + return spacesCountFutureCache.get( new ServiceContext() { public IntegerData execute() { return new IntegerData(CachedSpaceStorage.super.getMemberSpacesByFilterCount(userId, spaceFilter)); @@ -438,9 +423,7 @@ public IntegerData execute() { } - /** - * {@inheritDoc} - */ + @Override public List getMemberSpacesByFilter( final String userId, final SpaceFilter spaceFilter, final long offset, final long limit) { @@ -449,7 +432,7 @@ public List getMemberSpacesByFilter( ListSpacesKey listKey = new ListSpacesKey(key, offset, limit); // - ListSpacesData keys = spacesCache.get( + ListSpacesData keys = spacesFutureCache.get( new ServiceContext() { public ListSpacesData execute() { if (limit == 0) { @@ -466,16 +449,14 @@ public ListSpacesData execute() { } - /** - * {@inheritDoc} - */ + @Override public int getPendingSpacesByFilterCount(final String userId, final SpaceFilter spaceFilter) { // SpaceFilterKey key = new SpaceFilterKey(userId, spaceFilter, SpaceType.PENDING); // - return spacesCountCache.get( + return spacesCountFutureCache.get( new ServiceContext() { public IntegerData execute() { return new IntegerData(CachedSpaceStorage.super.getPendingSpacesByFilterCount(userId, spaceFilter)); @@ -486,9 +467,7 @@ public IntegerData execute() { } - /** - * {@inheritDoc} - */ + @Override public List getPendingSpacesByFilter( final String userId, final SpaceFilter spaceFilter, final long offset, final long limit) { @@ -497,7 +476,7 @@ public List getPendingSpacesByFilter( ListSpacesKey listKey = new ListSpacesKey(key, offset, limit); // - ListSpacesData keys = spacesCache.get( + ListSpacesData keys = spacesFutureCache.get( new ServiceContext() { public ListSpacesData execute() { if (limit == 0) { @@ -514,16 +493,14 @@ public ListSpacesData execute() { } - /** - * {@inheritDoc} - */ + @Override public int getInvitedSpacesByFilterCount(final String userId, final SpaceFilter spaceFilter) { // SpaceFilterKey key = new SpaceFilterKey(userId, spaceFilter, SpaceType.INVITED); // - return spacesCountCache.get( + return spacesCountFutureCache.get( new ServiceContext() { public IntegerData execute() { return new IntegerData(CachedSpaceStorage.super.getInvitedSpacesByFilterCount(userId, spaceFilter)); @@ -534,9 +511,7 @@ public IntegerData execute() { } - /** - * {@inheritDoc} - */ + @Override public List getInvitedSpacesByFilter(final String userId, final SpaceFilter spaceFilter, final long offset, final long limit) { // @@ -544,7 +519,7 @@ public List getInvitedSpacesByFilter(final String userId, final SpaceFilt ListSpacesKey listKey = new ListSpacesKey(key, offset, limit); // - ListSpacesData keys = spacesCache.get( + ListSpacesData keys = spacesFutureCache.get( new ServiceContext() { public ListSpacesData execute() { if (limit == 0) { @@ -561,16 +536,14 @@ public ListSpacesData execute() { } - /** - * {@inheritDoc} - */ + @Override public int getPublicSpacesByFilterCount(final String userId, final SpaceFilter spaceFilter) { // SpaceFilterKey key = new SpaceFilterKey(userId, spaceFilter, SpaceType.PUBLIC); // - return spacesCountCache.get( + return spacesCountFutureCache.get( new ServiceContext() { public IntegerData execute() { return new IntegerData(CachedSpaceStorage.super.getPublicSpacesByFilterCount(userId, spaceFilter)); @@ -581,15 +554,13 @@ public IntegerData execute() { } - /** - * {@inheritDoc} - */ + @Override public int getAccessibleSpacesByFilterCount(final String userId, final SpaceFilter spaceFilter) { // SpaceFilterKey key = new SpaceFilterKey(userId, spaceFilter, SpaceType.ACCESSIBLE); // - return spacesCountCache.get( + return spacesCountFutureCache.get( new ServiceContext() { public IntegerData execute() { return new IntegerData(CachedSpaceStorage.super.getAccessibleSpacesByFilterCount(userId, spaceFilter)); @@ -600,15 +571,13 @@ public IntegerData execute() { } - /** - * {@inheritDoc} - */ + @Override public int getVisibleSpacesCount(final String userId, final SpaceFilter spaceFilter) throws SpaceStorageException { // SpaceFilterKey key = new SpaceFilterKey(userId, spaceFilter, SpaceType.VISIBLE); // - return spacesCountCache.get( + return spacesCountFutureCache.get( new ServiceContext() { public IntegerData execute() { return new IntegerData(CachedSpaceStorage.super.getVisibleSpacesCount(userId, spaceFilter)); @@ -617,9 +586,7 @@ public IntegerData execute() { key).build(); } - /** - * {@inheritDoc} - */ + @Override public List getAccessibleSpacesByFilter( final String userId, final SpaceFilter spaceFilter, final long offset, final long limit) { @@ -628,7 +595,7 @@ public List getAccessibleSpacesByFilter( ListSpacesKey listKey = new ListSpacesKey(key, offset, limit); // - ListSpacesData keys = spacesCache.get( + ListSpacesData keys = spacesFutureCache.get( new ServiceContext() { public ListSpacesData execute() { if (limit == 0) { @@ -645,16 +612,14 @@ public ListSpacesData execute() { } - /** - * {@inheritDoc} - */ + @Override public int getEditableSpacesByFilterCount(final String userId, final SpaceFilter spaceFilter) { // SpaceFilterKey key = new SpaceFilterKey(userId, spaceFilter, SpaceType.EDITABLE); // - return spacesCountCache.get( + return spacesCountFutureCache.get( new ServiceContext() { public IntegerData execute() { return new IntegerData(CachedSpaceStorage.super.getEditableSpacesByFilterCount(userId, spaceFilter)); @@ -665,9 +630,7 @@ public IntegerData execute() { } - /** - * {@inheritDoc} - */ + @Override public List getEditableSpacesByFilter( final String userId, final SpaceFilter spaceFilter, final long offset, final long limit) { @@ -676,7 +639,7 @@ public List getEditableSpacesByFilter( ListSpacesKey listKey = new ListSpacesKey(key, offset, limit); // - ListSpacesData keys = spacesCache.get( + ListSpacesData keys = spacesFutureCache.get( new ServiceContext() { public ListSpacesData execute() { if (limit == 0) { @@ -693,15 +656,13 @@ public ListSpacesData execute() { } - /** - * {@inheritDoc} - */ + @Override public int getAllSpacesByFilterCount(final SpaceFilter spaceFilter) { // SpaceFilterKey key = new SpaceFilterKey(null, spaceFilter, null); // - return spacesCountCache.get( + return spacesCountFutureCache.get( new ServiceContext() { public IntegerData execute() { return new IntegerData(CachedSpaceStorage.super.getAllSpacesByFilterCount(spaceFilter)); @@ -712,9 +673,7 @@ public IntegerData execute() { } - /** - * {@inheritDoc} - */ + @Override public List getSpacesByFilter(final SpaceFilter spaceFilter, final long offset, final long limit) { // @@ -722,7 +681,7 @@ public List getSpacesByFilter(final SpaceFilter spaceFilter, final long o ListSpacesKey listKey = new ListSpacesKey(key, offset, limit); // - ListSpacesData keys = spacesCache.get( + ListSpacesData keys = spacesFutureCache.get( new ServiceContext() { public ListSpacesData execute() { if (limit == 0) { @@ -739,16 +698,14 @@ public ListSpacesData execute() { } - /** - * {@inheritDoc} - */ + @Override public Space getSpaceById(final String id) throws SpaceStorageException { // SpaceKey key = new SpaceKey(id); // - SpaceData data = spaceCache.get( + SpaceData data = spaceFutureCache.get( new ServiceContext() { public SpaceData execute() { Space space = CachedSpaceStorage.super.getSpaceById(id); @@ -772,18 +729,16 @@ public SpaceData execute() { } - /** - * {@inheritDoc} - */ + @Override public Space getSpaceSimpleById(final String id) throws SpaceStorageException { // SpaceKey key = new SpaceKey(id); - SpaceData data = exoSpaceCache.get(key); + SpaceData data = spaceCache.get(key); if (data != null) { Space s = data.build(); - if (exoSpaceSimpleCache.get(key) == null) { + if (spaceSimpleCache.get(key) == null) { putSpaceInCacheIfNotExists(s); } @@ -791,7 +746,7 @@ public Space getSpaceSimpleById(final String id) throws SpaceStorageException { } // - SpaceSimpleData simpleData = spaceSimpleCache.get( + SpaceSimpleData simpleData = spaceSimpleFutureCache.get( new ServiceContext() { public SpaceSimpleData execute() { Space space = CachedSpaceStorage.super.getSpaceSimpleById(id); @@ -814,16 +769,14 @@ public SpaceSimpleData execute() { } - /** - * {@inheritDoc} - */ + @Override public Space getSpaceByPrettyName(final String spacePrettyName) throws SpaceStorageException { // SpaceRefKey refKey = new SpaceRefKey(null, spacePrettyName); // - SpaceKey key = spaceRefCache.get( + SpaceKey key = spaceRefFutureCache.get( new ServiceContext() { public SpaceKey execute() { Space space = CachedSpaceStorage.super.getSpaceByPrettyName(spacePrettyName); @@ -847,16 +800,14 @@ public SpaceKey execute() { } - /** - * {@inheritDoc} - */ + @Override public Space getSpaceByGroupId(final String groupId) throws SpaceStorageException { // SpaceRefKey refKey = new SpaceRefKey(null, null, groupId); // - SpaceKey key = spaceRefCache.get( + SpaceKey key = spaceRefFutureCache.get( new ServiceContext() { public SpaceKey execute() { Space space = CachedSpaceStorage.super.getSpaceByGroupId(groupId); @@ -880,16 +831,14 @@ public SpaceKey execute() { } - /** - * {@inheritDoc} - */ + @Override public Space getSpaceByUrl(final String url) throws SpaceStorageException { // SpaceRefKey refKey = new SpaceRefKey(null, null, null, url); // - SpaceKey key = spaceRefCache.get( + SpaceKey key = spaceRefFutureCache.get( new ServiceContext() { public SpaceKey execute() { Space space = CachedSpaceStorage.super.getSpaceByUrl(url); @@ -922,7 +871,7 @@ public void updateSpaceAccessed(String remoteId, Space space) throws SpaceStorag if (selector.isUpdateStore()) { super.updateSpaceAccessed(remoteId, space); } - exoSpacesCache.select(selector); + spacesCache.select(selector); } catch (Exception e) { LOG.error("Error while removing cache entries for remoteId=" + remoteId + ", space=" + space.getDisplayName() + " and type=" + SpaceType.LATEST_ACCESSED.name() + " or type=" + SpaceType.VISITED, e); @@ -936,7 +885,7 @@ public List getLastAccessedSpace(final SpaceFilter filter, final int offs ListSpacesKey listKey = new ListSpacesKey(key, offset, limit); // - ListSpacesData keys = spacesCache.get( + ListSpacesData keys = spacesFutureCache.get( new ServiceContext() { public ListSpacesData execute() { if (limit == 0) { @@ -960,7 +909,7 @@ public List getLastSpaces(final int limit) { ListSpacesKey listKey = new ListSpacesKey(key, 0, limit); // - ListSpacesData keys = spacesCache.get( + ListSpacesData keys = spacesFutureCache.get( new ServiceContext() { public ListSpacesData execute() { if (limit == 0) { @@ -982,7 +931,7 @@ public int getNumberOfMemberPublicSpaces(final String userId) { SpaceFilterKey key = new SpaceFilterKey(userId, null, SpaceType.PUBLIC); // - return spacesCountCache.get( + return spacesCountFutureCache.get( new ServiceContext() { public IntegerData execute() { return new IntegerData(CachedSpaceStorage.super.getNumberOfMemberPublicSpaces(userId)); @@ -1000,7 +949,7 @@ public List getVisitedSpaces(final SpaceFilter filter, final int offset, ListSpacesKey listKey = new ListSpacesKey(key, offset, limit); // - ListSpacesData keys = spacesCache.get(new ServiceContext() { + ListSpacesData keys = spacesFutureCache.get(new ServiceContext() { public ListSpacesData execute() { if (limit == 0) { return buildIds(Collections.emptyList()); @@ -1021,17 +970,16 @@ public List getMemberRoleSpaceIdentityIds(String identityId, int offset, ListSpacesKey listKey = new ListSpacesKey(key, offset, limit); // - ListSpacesData keys = spacesCache.get( - new ServiceContext() { - public ListSpacesData execute() { - if (limit == 0) { - return buildIds(Collections.emptyList()); - } - List got = CachedSpaceStorage.super.getMemberRoleSpaceIdentityIds(identityId, offset, limit); - return buildListIdentityIds(got); - } - }, - listKey); + ListSpacesData keys = spacesFutureCache.get(() -> { + if (limit == 0) { + return buildIds(Collections.emptyList()); + } + List got = + CachedSpaceStorage.super.getMemberRoleSpaceIdentityIds(identityId, + offset, + limit); + return buildListIdentityIds(got); + }, listKey); // return buildIds(keys); @@ -1044,56 +992,33 @@ public List getMemberRoleSpaceIds(String identityId, int offset, int lim ListSpacesKey listKey = new ListSpacesKey(key, offset, limit); // - ListSpacesData keys = spacesCache.get( - new ServiceContext() { - public ListSpacesData execute() { - if (limit == 0) { - return buildIds(Collections.emptyList()); - } - List got = CachedSpaceStorage.super.getMemberRoleSpaceIds(identityId, - offset, - limit); - return buildListIdentityIds(got); - } - }, - listKey); + ListSpacesData keys = spacesFutureCache.get(() -> { + if (limit == 0) { + return buildIds(Collections.emptyList()); + } + List got = CachedSpaceStorage.super.getMemberRoleSpaceIds(identityId, + offset, + limit); + return buildListIdentityIds(got); + }, listKey); // return buildIds(keys); } - private SpaceKey putSpaceInCacheIfNotExists(Space space) { - SpaceKey key = new SpaceKey(space.getId()); - if(exoSpaceCache.get(key) == null) { - exoSpaceCache.putLocal(key, new SpaceData(space)); - exoSpaceSimpleCache.putLocal(key, new SpaceSimpleData(space)); - } - SpaceRefKey refKey = new SpaceRefKey(space.getDisplayName(), null, null, null); - if(exoRefSpaceCache.get(refKey) == null) { - exoRefSpaceCache.putLocal(refKey, key); - } - refKey = new SpaceRefKey(null, null, space.getGroupId(), null); - if(exoRefSpaceCache.get(refKey) == null) { - exoRefSpaceCache.putLocal(refKey, key); - } - refKey = new SpaceRefKey(null, space.getPrettyName(), null, null); - if(exoRefSpaceCache.get(refKey) == null) { - exoRefSpaceCache.putLocal(refKey, key); - } - refKey = new SpaceRefKey(null, null, null, space.getUrl()); - if(exoRefSpaceCache.get(refKey) == null) { - exoRefSpaceCache.putLocal(refKey, key); - } - return key; + @Override + public Map countSpacesByTemplate() { + return spacesCountByTemplateFutureCache.get(() -> new HashMap<>(super.countSpacesByTemplate()), + EMPTY_SPACE_KEY); } public void clearSpaceCached(String spaceId) { SpaceKey cacheKey = new SpaceKey(spaceId); - SpaceData cachedSpace = exoSpaceCache.get(cacheKey); + SpaceData cachedSpace = spaceCache.get(cacheKey); if (cachedSpace != null) { Space space = cachedSpace.build(); - exoSpaceSimpleCache.remove(cacheKey); - exoSpaceCache.remove(cacheKey); + spaceSimpleCache.remove(cacheKey); + spaceCache.remove(cacheKey); cleanRef(space); } clearSpaceCache(); @@ -1101,12 +1026,38 @@ public void clearSpaceCached(String spaceId) { } public void clearCaches() { - exoSpaceCache.clearCache(); - exoSpaceSimpleCache.clearCache(); - exoSpacesCountCache.clearCache(); - exoSpacesCache.clearCache(); - exoRefSpaceCache.clearCache(); - exoIdentitiesCache.clearCache(); + spaceCache.clearCache(); + spaceSimpleCache.clearCache(); + spacesCountCache.clearCache(); + spacesCountByTemplateCache.clearCache(); + spacesCache.clearCache(); + spaceRefCache.clearCache(); + identitiesCache.clearCache(); + } + + private SpaceKey putSpaceInCacheIfNotExists(Space space) { + SpaceKey key = new SpaceKey(space.getId()); + if(spaceCache.get(key) == null) { + spaceCache.putLocal(key, new SpaceData(space)); + spaceSimpleCache.putLocal(key, new SpaceSimpleData(space)); + } + SpaceRefKey refKey = new SpaceRefKey(space.getDisplayName(), null, null, null); + if(spaceRefCache.get(refKey) == null) { + spaceRefCache.putLocal(refKey, key); + } + refKey = new SpaceRefKey(null, null, space.getGroupId(), null); + if(spaceRefCache.get(refKey) == null) { + spaceRefCache.putLocal(refKey, key); + } + refKey = new SpaceRefKey(null, space.getPrettyName(), null, null); + if(spaceRefCache.get(refKey) == null) { + spaceRefCache.putLocal(refKey, key); + } + refKey = new SpaceRefKey(null, null, null, space.getUrl()); + if(spaceRefCache.get(refKey) == null) { + spaceRefCache.putLocal(refKey, key); + } + return key; } } diff --git a/component/core/src/main/java/org/exoplatform/social/core/storage/cache/SocialStorageCacheService.java b/component/core/src/main/java/org/exoplatform/social/core/storage/cache/SocialStorageCacheService.java index afc9705c92e..d844be9236a 100644 --- a/component/core/src/main/java/org/exoplatform/social/core/storage/cache/SocialStorageCacheService.java +++ b/component/core/src/main/java/org/exoplatform/social/core/storage/cache/SocialStorageCacheService.java @@ -17,7 +17,8 @@ package org.exoplatform.social.core.storage.cache; -import org.exoplatform.commons.file.model.FileItem; +import java.util.HashMap; + import org.exoplatform.services.cache.CacheService; import org.exoplatform.services.cache.ExoCache; import org.exoplatform.social.core.storage.cache.model.data.ActivityData; @@ -80,6 +81,7 @@ public class SocialStorageCacheService { private final ExoCache spaceCache; private final ExoCache spaceRefCache; private final ExoCache spacesCountCache; + private final ExoCache> spacesCountByTemplateCache; private final ExoCache spacesCache; private final ExoCache spaceSimpleCache; @@ -107,12 +109,17 @@ public SocialStorageCacheService(CacheService cacheService) { this.spaceCache = CacheType.SPACE.getFromService(cacheService); this.spaceRefCache = CacheType.SPACE_REF.getFromService(cacheService); this.spacesCountCache = CacheType.SPACES_COUNT.getFromService(cacheService); + this.spacesCountByTemplateCache = CacheType.SPACES_COUNT_BY_TEMPLATE.getFromService(cacheService); this.spacesCache = CacheType.SPACES.getFromService(cacheService); this.spaceSimpleCache = CacheType.SPACE_SIMPLE.getFromService(cacheService); } + public ExoCache> getSpacesCountByTemplateCache() { + return spacesCountByTemplateCache; + } + public ExoCache getIdentityCache() { return identityCache; } @@ -187,4 +194,5 @@ public ExoCache getSpacesCountCache() { public ExoCache getSpacesCache() { return spacesCache; } + } diff --git a/component/core/src/main/resources/db/changelog/social-rdbms.db.changelog-1.0.0.xml b/component/core/src/main/resources/db/changelog/social-rdbms.db.changelog-1.0.0.xml index b24db35d527..abd24dbad5c 100644 --- a/component/core/src/main/resources/db/changelog/social-rdbms.db.changelog-1.0.0.xml +++ b/component/core/src/main/resources/db/changelog/social-rdbms.db.changelog-1.0.0.xml @@ -1018,4 +1018,39 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/component/core/src/main/resources/db/changelog/social-rdbms.db.changelog-master.xml b/component/core/src/main/resources/db/changelog/social-rdbms.db.changelog-master.xml deleted file mode 100644 index 7abc934bae5..00000000000 --- a/component/core/src/main/resources/db/changelog/social-rdbms.db.changelog-master.xml +++ /dev/null @@ -1,29 +0,0 @@ - - - - - - - \ No newline at end of file diff --git a/component/core/src/main/resources/jpa-entities.idx b/component/core/src/main/resources/jpa-entities.idx index 7306feca9da..15a3f9cb006 100644 --- a/component/core/src/main/resources/jpa-entities.idx +++ b/component/core/src/main/resources/jpa-entities.idx @@ -20,3 +20,4 @@ org.exoplatform.social.core.jpa.storage.entity.ProfileLabelEntity io.meeds.social.link.entity.LinkSettingEntity org.exoplatform.social.core.jpa.storage.entity.SpaceExternalInvitationEntity org.exoplatform.social.core.jpa.storage.entity.MetadataItemEntity +io.meeds.social.space.template.entity.SpaceTemplateEntity diff --git a/component/core/src/test/java/io/meeds/social/space/template/plugin/attachment/SpaceTemplateBannerAttachmentPluginTest.java b/component/core/src/test/java/io/meeds/social/space/template/plugin/attachment/SpaceTemplateBannerAttachmentPluginTest.java new file mode 100644 index 00000000000..d7f91ae6baf --- /dev/null +++ b/component/core/src/test/java/io/meeds/social/space/template/plugin/attachment/SpaceTemplateBannerAttachmentPluginTest.java @@ -0,0 +1,113 @@ +/** + * This file is part of the Meeds project (https://meeds.io/). + * + * Copyright (C) 2020 - 2024 Meeds Association contact@meeds.io + * + * This program is free software; you can redistribute it and/or + * modify it under the terms of the GNU Lesser General Public + * License as published by the Free Software Foundation; either + * version 3 of the License, or (at your option) any later version. + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU + * Lesser General Public License for more details. + * + * You should have received a copy of the GNU Lesser General Public License + * along with this program; if not, write to the Free Software Foundation, + * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. + */ +package io.meeds.social.space.template.plugin.attachment; + +import static org.junit.Assert.assertEquals; +import static org.junit.Assert.assertFalse; +import static org.junit.Assert.assertThrows; +import static org.junit.Assert.assertTrue; +import static org.mockito.Mockito.when; + +import org.junit.Before; +import org.junit.Test; +import org.junit.runner.RunWith; +import org.mockito.Mock; +import org.mockito.junit.MockitoJUnitRunner; + +import org.exoplatform.commons.exception.ObjectNotFoundException; +import org.exoplatform.services.security.Identity; +import org.exoplatform.social.attachment.AttachmentService; + +import io.meeds.social.space.template.model.SpaceTemplate; +import io.meeds.social.space.template.service.SpaceTemplateService; + +import lombok.SneakyThrows; + +@RunWith(MockitoJUnitRunner.class) +public class SpaceTemplateBannerAttachmentPluginTest { + + @Mock + private SpaceTemplateService spaceTemplateService; + + @Mock + private AttachmentService attachmentService; + + private SpaceTemplateBannerAttachmentPlugin attachmentPlugin; + + @Mock + private Identity userIdentity; + + @Mock + private SpaceTemplate spaceTemplate; + + @Before + public void init() { + attachmentPlugin = new SpaceTemplateBannerAttachmentPlugin(); + attachmentPlugin.attachmentService = attachmentService; + attachmentPlugin.spaceTemplateService = spaceTemplateService; + } + + @Test + public void getObjectType() { + assertEquals(SpaceTemplateBannerAttachmentPlugin.OBJECT_TYPE, attachmentPlugin.getObjectType()); + } + + @Test + @SneakyThrows + public void testHasEditPermission() { + assertFalse(attachmentPlugin.hasEditPermission(null, null)); + assertFalse(attachmentPlugin.hasEditPermission(userIdentity, null)); + + when(userIdentity.getUserId()).thenReturn("test"); + assertFalse(attachmentPlugin.hasEditPermission(userIdentity, null)); + + when(spaceTemplateService.canManageTemplates("test")).thenReturn(true); + assertTrue(attachmentPlugin.hasEditPermission(userIdentity, "2")); + } + + @Test + @SneakyThrows + public void testHasAccessPermission() { + assertFalse(attachmentPlugin.hasAccessPermission(null, null)); + assertFalse(attachmentPlugin.hasAccessPermission(userIdentity, null)); + assertFalse(attachmentPlugin.hasAccessPermission(null, "2")); + + when(userIdentity.getUserId()).thenReturn("test"); + assertThrows(ObjectNotFoundException.class, () -> attachmentPlugin.hasAccessPermission(userIdentity, "2")); + + when(spaceTemplateService.getSpaceTemplate(2)).thenReturn(spaceTemplate); + assertFalse(attachmentPlugin.hasAccessPermission(userIdentity, "2")); + + when(spaceTemplateService.canViewTemplate(2, "test")).thenReturn(true); + assertTrue(attachmentPlugin.hasAccessPermission(userIdentity, "2")); + } + + @Test + @SneakyThrows + public void getAudienceId() { + assertEquals(0l, attachmentPlugin.getAudienceId(null)); + } + + @Test + @SneakyThrows + public void getSpaceId() { + assertEquals(0l, attachmentPlugin.getSpaceId("")); + } + +} diff --git a/component/core/src/test/java/io/meeds/social/space/template/plugin/translation/SpaceTemplateTranslationPluginTest.java b/component/core/src/test/java/io/meeds/social/space/template/plugin/translation/SpaceTemplateTranslationPluginTest.java new file mode 100644 index 00000000000..d0ac653e3b5 --- /dev/null +++ b/component/core/src/test/java/io/meeds/social/space/template/plugin/translation/SpaceTemplateTranslationPluginTest.java @@ -0,0 +1,102 @@ +/** + * This file is part of the Meeds project (https://meeds.io/). + * + * Copyright (C) 2020 - 2024 Meeds Association contact@meeds.io + * + * This program is free software; you can redistribute it and/or + * modify it under the terms of the GNU Lesser General Public + * License as published by the Free Software Foundation; either + * version 3 of the License, or (at your option) any later version. + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU + * Lesser General Public License for more details. + * + * You should have received a copy of the GNU Lesser General Public License + * along with this program; if not, write to the Free Software Foundation, + * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. + */ +package io.meeds.social.space.template.plugin.translation; + +import static org.junit.Assert.assertEquals; +import static org.junit.Assert.assertFalse; +import static org.junit.Assert.assertThrows; +import static org.junit.Assert.assertTrue; +import static org.mockito.Mockito.when; + +import org.junit.Before; +import org.junit.Test; +import org.junit.runner.RunWith; +import org.mockito.Mock; +import org.mockito.junit.MockitoJUnitRunner; + +import org.exoplatform.commons.exception.ObjectNotFoundException; + +import io.meeds.social.space.template.model.SpaceTemplate; +import io.meeds.social.space.template.service.SpaceTemplateService; +import io.meeds.social.translation.service.TranslationService; + +import lombok.SneakyThrows; + +@RunWith(MockitoJUnitRunner.class) +public class SpaceTemplateTranslationPluginTest { + + private static final String TEST_USER = "test"; + + @Mock + private SpaceTemplateService spaceTemplateService; + + @Mock + private TranslationService translationService; + + @Mock + private SpaceTemplate spaceTemplate; + + private SpaceTemplateTranslationPlugin translationPlugin; + + @Before + public void init() { + translationPlugin = new SpaceTemplateTranslationPlugin(); + translationPlugin.translationService = translationService; + translationPlugin.spaceTemplateService = spaceTemplateService; + } + + @Test + @SneakyThrows + public void testHasEditPermission() { + assertFalse(translationPlugin.hasEditPermission(0, TEST_USER)); + when(spaceTemplateService.canManageTemplates(TEST_USER)).thenReturn(true); + assertTrue(translationPlugin.hasEditPermission(0, TEST_USER)); + } + + @Test + @SneakyThrows + public void testHasAccessPermission() { + assertThrows(ObjectNotFoundException.class, () -> translationPlugin.hasAccessPermission(2, TEST_USER)); + when(spaceTemplateService.canManageTemplates(TEST_USER)).thenReturn(true); + assertTrue(translationPlugin.hasEditPermission(0, TEST_USER)); + + when(spaceTemplateService.getSpaceTemplate(2)).thenReturn(spaceTemplate); + assertFalse(translationPlugin.hasAccessPermission(2, TEST_USER)); + assertThrows(ObjectNotFoundException.class, () -> translationPlugin.hasAccessPermission(3, TEST_USER)); + + when(spaceTemplateService.canViewTemplate(2, TEST_USER)).thenReturn(true); + assertTrue(translationPlugin.hasAccessPermission(2, TEST_USER)); + } + + @Test + public void getObjectType() { + assertEquals(SpaceTemplateTranslationPlugin.OBJECT_TYPE, translationPlugin.getObjectType()); + } + + @Test + public void getAudienceId() { + assertEquals(0l, translationPlugin.getAudienceId(0)); + } + + @Test + public void getSpaceId() { + assertEquals(0l, translationPlugin.getSpaceId(0)); + } + +} diff --git a/component/core/src/test/java/io/meeds/social/space/template/service/SpaceTemplateServiceTest.java b/component/core/src/test/java/io/meeds/social/space/template/service/SpaceTemplateServiceTest.java new file mode 100644 index 00000000000..d88ab60d8a3 --- /dev/null +++ b/component/core/src/test/java/io/meeds/social/space/template/service/SpaceTemplateServiceTest.java @@ -0,0 +1,246 @@ +/** + * This file is part of the Meeds project (https://meeds.io/). + * + * Copyright (C) 2020 - 2024 Meeds Association contact@meeds.io + * + * This program is free software; you can redistribute it and/or + * modify it under the terms of the GNU Lesser General Public + * License as published by the Free Software Foundation; either + * version 3 of the License, or (at your option) any later version. + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU + * Lesser General Public License for more details. + * + * You should have received a copy of the GNU Lesser General Public License + * along with this program; if not, write to the Free Software Foundation, + * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. + */ +package io.meeds.social.space.template.service; + +import static org.junit.Assert.assertFalse; +import static org.junit.Assert.assertNotNull; +import static org.junit.Assert.assertNull; +import static org.junit.Assert.assertThrows; +import static org.junit.Assert.assertTrue; +import static org.junit.jupiter.api.Assertions.assertEquals; +import static org.mockito.Mockito.verify; +import static org.mockito.Mockito.when; + +import java.util.Arrays; +import java.util.List; +import java.util.Locale; + +import org.junit.Before; +import org.junit.Test; +import org.junit.runner.RunWith; +import org.mockito.Mock; +import org.mockito.junit.MockitoJUnitRunner; +import org.springframework.data.domain.Pageable; + +import org.exoplatform.commons.exception.ObjectNotFoundException; +import org.exoplatform.portal.config.UserACL; +import org.exoplatform.services.security.Identity; +import org.exoplatform.services.security.MembershipEntry; +import org.exoplatform.social.attachment.AttachmentService; +import org.exoplatform.social.core.space.SpacesAdministrationService; + +import io.meeds.social.space.constant.Registration; +import io.meeds.social.space.constant.Visibility; +import io.meeds.social.space.template.model.SpaceTemplate; +import io.meeds.social.space.template.model.SpaceTemplateFilter; +import io.meeds.social.space.template.storage.SpaceTemplateStorage; +import io.meeds.social.translation.service.TranslationService; + +@RunWith(MockitoJUnitRunner.class) +public class SpaceTemplateServiceTest { + + private static final String SPACE_FIELDS = "spaceFields"; + + private static final String SPACE_DELETE_PERMISSIONS = "spaceDeletePermissions"; + + private static final String SPACE_LAYOUT_PERMISSIONS = "spaceLayoutPermissions"; + + private static final String CREATE_AND_ACCESS_PERMISSIONS = "permissions"; + + private static final String TEST_USER = "testuser"; + + @Mock + protected TranslationService translationService; + + @Mock + protected AttachmentService attachmentService; + + @Mock + protected SpacesAdministrationService spacesAdministrationService; + + @Mock + protected UserACL userAcl; + + @Mock + private SpaceTemplateStorage spaceTemplateStorage; + + @Mock + private Identity userIdentity; + + private SpaceTemplateService spaceTemplateService; + + @Before + public void init() { + spaceTemplateService = new SpaceTemplateService(); + spaceTemplateService.translationService = translationService; + spaceTemplateService.attachmentService = attachmentService; + spaceTemplateService.spacesAdministrationService = spacesAdministrationService; + spaceTemplateService.userAcl = userAcl; + spaceTemplateService.spaceTemplateStorage = spaceTemplateStorage; + } + + @Test + public void testGetSpaceTemplates() { + Pageable pageable = Pageable.unpaged(); + List spaceTemplates = spaceTemplateService.getSpaceTemplates(new SpaceTemplateFilter(), pageable, true); + assertNotNull(spaceTemplates); + assertEquals(0, spaceTemplates.size()); + + SpaceTemplate spaceTemplate = newSpaceTemplate(2l); + when(spaceTemplateStorage.getSpaceTemplates(pageable)).then(invocation -> List.of(spaceTemplate)); + spaceTemplates = spaceTemplateService.getSpaceTemplates(null, pageable, true); + assertNotNull(spaceTemplates); + assertEquals(1, spaceTemplates.size()); + assertEquals(spaceTemplate, spaceTemplates.get(0)); + + SpaceTemplateFilter spaceTemplateFilter = new SpaceTemplateFilter(); + spaceTemplateFilter.setUsername(TEST_USER); + spaceTemplateFilter.setIncludeDisabled(true); + spaceTemplates = spaceTemplateService.getSpaceTemplates(spaceTemplateFilter, pageable, true); + assertNotNull(spaceTemplates); + assertEquals(0, spaceTemplates.size()); + + setCanViewTemplate(true); + when(spaceTemplateStorage.getSpaceTemplate(2l)).thenReturn(spaceTemplate); + + spaceTemplates = spaceTemplateService.getSpaceTemplates(spaceTemplateFilter, pageable, false); + assertNotNull(spaceTemplates); + assertEquals(1, spaceTemplates.size()); + assertEquals(spaceTemplate, spaceTemplates.get(0)); + } + + @Test + public void testGetSpaceTemplate() throws IllegalAccessException { + assertNull(spaceTemplateService.getSpaceTemplate(2l)); + assertNull(spaceTemplateService.getSpaceTemplate(2l, TEST_USER, Locale.ENGLISH, false)); + + SpaceTemplate spaceTemplate = newSpaceTemplate(2l); + when(spaceTemplateStorage.getSpaceTemplate(2l)).thenReturn(spaceTemplate); + assertEquals(spaceTemplate, spaceTemplateService.getSpaceTemplate(2l)); + + assertThrows(IllegalAccessException.class, () -> spaceTemplateService.getSpaceTemplate(2l, TEST_USER, Locale.ENGLISH, false)); + setCanViewTemplate(true); + assertEquals(spaceTemplate, spaceTemplateService.getSpaceTemplate(2l, TEST_USER, Locale.ENGLISH, false)); + } + + @Test + public void testCanManageTemplates() { + assertFalse(spaceTemplateService.canManageTemplates(TEST_USER)); + setCanManageTemplate(true); + assertTrue(spaceTemplateService.canManageTemplates(TEST_USER)); + } + + @Test + public void testCanViewTemplateWhenManager() { + assertFalse(spaceTemplateService.canViewTemplate(2l, null)); + assertFalse(spaceTemplateService.canViewTemplate(2l, TEST_USER)); + setCanManageTemplate(true); + assertTrue(spaceTemplateService.canViewTemplate(2l, TEST_USER)); + } + + @Test + public void testCanViewTemplateWhenMemberOfPermissions() { + assertFalse(spaceTemplateService.canViewTemplate(2l, TEST_USER)); + setCanViewTemplate(true); + assertFalse(spaceTemplateService.canViewTemplate(2l, TEST_USER)); + when(spaceTemplateStorage.getSpaceTemplate(2l)).thenReturn(newSpaceTemplate(2l)); + assertTrue(spaceTemplateService.canViewTemplate(2l, TEST_USER)); + } + + @Test + public void testCreateSpaceTemplate() throws IllegalAccessException { + assertThrows(IllegalAccessException.class, () -> spaceTemplateService.createSpaceTemplate(newSpaceTemplate(0l), TEST_USER)); + setCanManageTemplate(true); + assertThrows(IllegalArgumentException.class, () -> spaceTemplateService.createSpaceTemplate(newSpaceTemplate(2l), TEST_USER)); + SpaceTemplate spaceTemplate = newSpaceTemplate(0l); + spaceTemplateService.createSpaceTemplate(spaceTemplate, TEST_USER); + verify(spaceTemplateStorage).createSpaceTemplate(spaceTemplate); + } + + @Test + public void testUpdateSpaceTemplate() throws ObjectNotFoundException, IllegalAccessException { + SpaceTemplate spaceTemplate = newSpaceTemplate(2l); + assertThrows(IllegalAccessException.class, () -> spaceTemplateService.updateSpaceTemplate(spaceTemplate, TEST_USER)); + setCanManageTemplate(true); + spaceTemplate.setDeleted(true); + assertThrows(IllegalArgumentException.class, () -> spaceTemplateService.updateSpaceTemplate(spaceTemplate, TEST_USER)); + + SpaceTemplate savedSpaceTemplate = newSpaceTemplate(2l); + when(spaceTemplateStorage.getSpaceTemplate(2l)).thenReturn(savedSpaceTemplate); + savedSpaceTemplate.setDeleted(true); + spaceTemplate.setDeleted(false); + + assertThrows(ObjectNotFoundException.class, () -> spaceTemplateService.updateSpaceTemplate(spaceTemplate, TEST_USER)); + + savedSpaceTemplate.setDeleted(false); + spaceTemplateService.updateSpaceTemplate(spaceTemplate, TEST_USER); + verify(spaceTemplateStorage).updateSpaceTemplate(spaceTemplate); + } + + @Test + public void testDeleteSpaceTemplate() throws IllegalAccessException, ObjectNotFoundException { + assertThrows(IllegalAccessException.class, () -> spaceTemplateService.deleteSpaceTemplate(2l, TEST_USER)); + setCanManageTemplate(true); + assertThrows(ObjectNotFoundException.class, () -> spaceTemplateService.deleteSpaceTemplate(2l, TEST_USER)); + + SpaceTemplate savedSpaceTemplate = newSpaceTemplate(2l); + when(spaceTemplateStorage.getSpaceTemplate(2l)).thenReturn(savedSpaceTemplate); + savedSpaceTemplate.setSystem(false); + savedSpaceTemplate.setDeleted(true); + assertThrows(ObjectNotFoundException.class, () -> spaceTemplateService.deleteSpaceTemplate(2l, TEST_USER)); + + savedSpaceTemplate.setDeleted(false); + savedSpaceTemplate.setSystem(true); + assertThrows(IllegalAccessException.class, () -> spaceTemplateService.deleteSpaceTemplate(2l, TEST_USER)); + savedSpaceTemplate.setSystem(false); + + spaceTemplateService.deleteSpaceTemplate(2l, TEST_USER); + verify(spaceTemplateStorage).updateSpaceTemplate(savedSpaceTemplate); + assertTrue(savedSpaceTemplate.isDeleted()); + } + + private void setCanViewTemplate(boolean hasAccess) { + when(userAcl.getUserIdentity(TEST_USER)).thenReturn(userIdentity); + when(userIdentity.isMemberOf(new MembershipEntry(CREATE_AND_ACCESS_PERMISSIONS))).thenReturn(hasAccess); + } + + private void setCanManageTemplate(boolean hasAccess) { + when(spacesAdministrationService.isSuperManager(TEST_USER)).thenReturn(hasAccess); + } + + private SpaceTemplate newSpaceTemplate(long id) { + return new SpaceTemplate(id, + "name", + "description", + 6l, + "icon", + true, + false, + true, + 5l, + Arrays.asList(CREATE_AND_ACCESS_PERMISSIONS), + Arrays.asList(SPACE_LAYOUT_PERMISSIONS), + Arrays.asList(SPACE_DELETE_PERMISSIONS), + Arrays.asList(SPACE_FIELDS), + Visibility.PRIVATE, + Registration.VALIDATION, + true); + } + +} diff --git a/component/core/src/test/java/io/meeds/social/space/template/storage/SpaceTemplateStorageTest.java b/component/core/src/test/java/io/meeds/social/space/template/storage/SpaceTemplateStorageTest.java new file mode 100644 index 00000000000..5ca35f46d56 --- /dev/null +++ b/component/core/src/test/java/io/meeds/social/space/template/storage/SpaceTemplateStorageTest.java @@ -0,0 +1,184 @@ +/** + * This file is part of the Meeds project (https://meeds.io/). + * + * Copyright (C) 2020 - 2024 Meeds Association contact@meeds.io + * + * This program is free software; you can redistribute it and/or + * modify it under the terms of the GNU Lesser General Public + * License as published by the Free Software Foundation; either + * version 3 of the License, or (at your option) any later version. + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU + * Lesser General Public License for more details. + * + * You should have received a copy of the GNU Lesser General Public License + * along with this program; if not, write to the Free Software Foundation, + * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. + */ +package io.meeds.social.space.template.storage; + +import static org.junit.Assert.assertEquals; +import static org.junit.Assert.assertNotNull; +import static org.junit.Assert.assertNull; +import static org.junit.Assert.assertThrows; +import static org.mockito.ArgumentMatchers.any; +import static org.mockito.Mockito.verify; +import static org.mockito.Mockito.when; + +import java.util.Arrays; +import java.util.List; +import java.util.Optional; + +import org.junit.Before; +import org.junit.Test; +import org.junit.runner.RunWith; +import org.mockito.Mock; +import org.mockito.junit.MockitoJUnitRunner; +import org.springframework.data.domain.Pageable; + +import org.exoplatform.commons.exception.ObjectNotFoundException; + +import io.meeds.social.space.constant.Registration; +import io.meeds.social.space.constant.Visibility; +import io.meeds.social.space.template.dao.SpaceTemplateDAO; +import io.meeds.social.space.template.entity.SpaceTemplateEntity; +import io.meeds.social.space.template.model.SpaceTemplate; + +@RunWith(MockitoJUnitRunner.class) +public class SpaceTemplateStorageTest { + + @Mock + private SpaceTemplateDAO spaceTemplateDAO; + + private SpaceTemplateStorage spaceTemplateStorage; + + @Before + public void init() { + spaceTemplateStorage = new SpaceTemplateStorage(); + spaceTemplateStorage.spaceTemplateDAO = spaceTemplateDAO; + } + + @Test + public void testGetSpaceTemplates() { + SpaceTemplateEntity spaceTemplateEntity = newSpaceTemplateEntity(); + when(spaceTemplateDAO.findByDeletedFalse(any())).thenAnswer(invocation -> List.of(spaceTemplateEntity)); + List spaceTemplates = spaceTemplateStorage.getSpaceTemplates(Pageable.unpaged()); + assertNotNull(spaceTemplates); + assertEquals(1l, spaceTemplates.size()); + + spaceTemplates = spaceTemplateStorage.getSpaceTemplates(Pageable.unpaged()); + SpaceTemplate spaceTemplate = spaceTemplates.get(0); + checkEntityEqualsModel(spaceTemplateEntity, spaceTemplate); + } + + @Test + public void testGetEnabledSpaceTemplates() { + SpaceTemplateEntity spaceTemplateEntity = newSpaceTemplateEntity(); + when(spaceTemplateDAO.findByDeletedFalseAndEnabledTrue(any())).thenAnswer(invocation -> List.of(spaceTemplateEntity)); + List spaceTemplates = spaceTemplateStorage.getEnabledSpaceTemplates(Pageable.unpaged()); + assertNotNull(spaceTemplates); + assertEquals(1l, spaceTemplates.size()); + + spaceTemplates = spaceTemplateStorage.getEnabledSpaceTemplates(Pageable.unpaged()); + SpaceTemplate spaceTemplate = spaceTemplates.get(0); + checkEntityEqualsModel(spaceTemplateEntity, spaceTemplate); + } + + @Test + public void testGetSpaceTemplate() { + assertNull(spaceTemplateStorage.getSpaceTemplate(3l)); + + SpaceTemplateEntity spaceTemplateEntity = newSpaceTemplateEntity(); + when(spaceTemplateDAO.findById(3l)).thenAnswer(invocation -> Optional.of(spaceTemplateEntity)); + SpaceTemplate spaceTemplate = spaceTemplateStorage.getSpaceTemplate(3l); + assertNotNull(spaceTemplate); + checkEntityEqualsModel(spaceTemplateEntity, spaceTemplate); + } + + @Test + public void testCreateSpaceTemplate() { + SpaceTemplate spaceTemplate = newSpaceTemplate(); + when(spaceTemplateDAO.save(any())).thenReturn(newSpaceTemplateEntity()); + SpaceTemplate createdSpaceTemplate = spaceTemplateStorage.createSpaceTemplate(spaceTemplate); + assertNotNull(createdSpaceTemplate); + spaceTemplate.setName(null); + spaceTemplate.setDescription(null); + spaceTemplate.setBannerFileId(0); + assertEquals(spaceTemplate, createdSpaceTemplate); + } + + @Test + public void testUpdateSpaceTemplate() throws ObjectNotFoundException { + SpaceTemplate spaceTemplate = newSpaceTemplate(); + assertThrows(ObjectNotFoundException.class, () -> spaceTemplateStorage.updateSpaceTemplate(spaceTemplate)); + when(spaceTemplateDAO.existsById(spaceTemplate.getId())).thenReturn(true); + when(spaceTemplateDAO.save(any())).thenReturn(newSpaceTemplateEntity()); + SpaceTemplate updatedSpaceTemplate = spaceTemplateStorage.updateSpaceTemplate(spaceTemplate); + assertNotNull(updatedSpaceTemplate); + spaceTemplate.setName(null); + spaceTemplate.setDescription(null); + spaceTemplate.setBannerFileId(0); + assertEquals(spaceTemplate, updatedSpaceTemplate); + } + + @Test + public void testDeleteSpaceTemplate() { + spaceTemplateStorage.deleteSpaceTemplate(2l); + verify(spaceTemplateDAO).deleteById(2l); + } + + private void checkEntityEqualsModel(SpaceTemplateEntity spaceTemplateEntity, SpaceTemplate spaceTemplate) { + assertNotNull(spaceTemplate); + assertEquals(spaceTemplateEntity.getId().longValue(), spaceTemplate.getId()); + assertEquals(spaceTemplateEntity.getIcon(), spaceTemplate.getIcon()); + assertEquals(spaceTemplateEntity.getOrder(), spaceTemplate.getOrder()); + assertEquals(spaceTemplateEntity.getPermissions(), spaceTemplate.getPermissions()); + assertEquals(spaceTemplateEntity.getIcon(), spaceTemplate.getIcon()); + assertEquals(spaceTemplateEntity.isEnabled(), spaceTemplate.isEnabled()); + assertEquals(spaceTemplateEntity.isDeleted(), spaceTemplate.isDeleted()); + assertEquals(spaceTemplateEntity.isSystem(), spaceTemplate.isSystem()); + assertEquals(spaceTemplateEntity.getSpaceDeletePermissions(), spaceTemplate.getSpaceDeletePermissions()); + assertEquals(spaceTemplateEntity.getSpaceLayoutPermissions(), spaceTemplate.getSpaceLayoutPermissions()); + assertEquals(spaceTemplateEntity.getSpaceFields(), spaceTemplate.getSpaceFields()); + assertEquals(spaceTemplateEntity.getSpaceDefaultVisibility(), spaceTemplate.getSpaceDefaultVisibility()); + assertEquals(spaceTemplateEntity.getSpaceDefaultRegistration(), spaceTemplate.getSpaceDefaultRegistration()); + assertEquals(spaceTemplateEntity.isSpaceAllowContentCreation(), spaceTemplate.isSpaceAllowContentCreation()); + } + + private SpaceTemplateEntity newSpaceTemplateEntity() { + return new SpaceTemplateEntity(2l, + "icon", + true, + false, + true, + 5l, + Arrays.asList("permissions"), + Arrays.asList("spaceLayoutPermissions"), + Arrays.asList("spaceDeletePermissions"), + Arrays.asList("spaceFields"), + Visibility.PRIVATE, + Registration.VALIDATION, + true); + } + + private SpaceTemplate newSpaceTemplate() { + return new SpaceTemplate(2l, + "name", + "description", + 6l, + "icon", + true, + false, + true, + 5l, + Arrays.asList("permissions"), + Arrays.asList("spaceLayoutPermissions"), + Arrays.asList("spaceDeletePermissions"), + Arrays.asList("spaceFields"), + Visibility.PRIVATE, + Registration.VALIDATION, + true); + } + +} diff --git a/component/core/src/test/java/org/exoplatform/social/core/jpa/storage/dao/SpaceDAOTest.java b/component/core/src/test/java/org/exoplatform/social/core/jpa/storage/dao/SpaceDAOTest.java index 2092a723f4e..5bb6dc5026f 100644 --- a/component/core/src/test/java/org/exoplatform/social/core/jpa/storage/dao/SpaceDAOTest.java +++ b/component/core/src/test/java/org/exoplatform/social/core/jpa/storage/dao/SpaceDAOTest.java @@ -32,7 +32,10 @@ import org.exoplatform.social.core.jpa.storage.entity.SpaceMemberEntity; import org.exoplatform.social.core.jpa.test.BaseCoreTest; +import io.meeds.social.space.constant.Priority; +import io.meeds.social.space.constant.Registration; import io.meeds.social.space.constant.SpaceMembershipStatus; +import io.meeds.social.space.constant.Visibility; public class SpaceDAOTest extends BaseCoreTest { private SpaceDAO spaceDAO; @@ -106,10 +109,10 @@ private SpaceEntity createSpace(String spacePrettyName) { spaceEntity.setDisplayName("testDisplayName"); spaceEntity.setGroupId("testGroupId"); spaceEntity.setPrettyName(spacePrettyName); - spaceEntity.setPriority(SpaceEntity.PRIORITY.HIGH); - spaceEntity.setRegistration(SpaceEntity.REGISTRATION.OPEN); + spaceEntity.setPriority(Priority.HIGH); + spaceEntity.setRegistration(Registration.OPEN); spaceEntity.setUrl("testUrl"); - spaceEntity.setVisibility(SpaceEntity.VISIBILITY.PRIVATE); + spaceEntity.setVisibility(Visibility.PRIVATE); spaceEntity.setBannerLastUpdated(new Date()); addMember(spaceEntity, "root", SpaceMembershipStatus.PENDING); diff --git a/component/core/src/test/java/org/exoplatform/social/core/jpa/storage/dao/SpaceMemberDAOTest.java b/component/core/src/test/java/org/exoplatform/social/core/jpa/storage/dao/SpaceMemberDAOTest.java index 8859980b32e..effb98b671a 100644 --- a/component/core/src/test/java/org/exoplatform/social/core/jpa/storage/dao/SpaceMemberDAOTest.java +++ b/component/core/src/test/java/org/exoplatform/social/core/jpa/storage/dao/SpaceMemberDAOTest.java @@ -31,7 +31,10 @@ import org.exoplatform.social.core.jpa.storage.entity.SpaceMemberEntity; import org.exoplatform.social.core.jpa.test.BaseCoreTest; +import io.meeds.social.space.constant.Priority; +import io.meeds.social.space.constant.Registration; import io.meeds.social.space.constant.SpaceMembershipStatus; +import io.meeds.social.space.constant.Visibility; public class SpaceMemberDAOTest extends BaseCoreTest { @@ -122,10 +125,10 @@ private SpaceEntity createSpace(String name) { spaceEntity.setDisplayName(name); spaceEntity.setGroupId(name+"GroupId"); spaceEntity.setPrettyName(name); - spaceEntity.setPriority(SpaceEntity.PRIORITY.HIGH); - spaceEntity.setRegistration(SpaceEntity.REGISTRATION.OPEN); + spaceEntity.setPriority(Priority.HIGH); + spaceEntity.setRegistration(Registration.OPEN); spaceEntity.setUrl("testUrl"); - spaceEntity.setVisibility(SpaceEntity.VISIBILITY.PRIVATE); + spaceEntity.setVisibility(Visibility.PRIVATE); spaceEntity.setBannerLastUpdated(new Date()); return spaceEntity; } diff --git a/component/core/src/test/java/org/exoplatform/social/core/test/NoContainerTestSuite.java b/component/core/src/test/java/org/exoplatform/social/core/test/NoContainerTestSuite.java index b66c52d6271..88d893529fe 100644 --- a/component/core/src/test/java/org/exoplatform/social/core/test/NoContainerTestSuite.java +++ b/component/core/src/test/java/org/exoplatform/social/core/test/NoContainerTestSuite.java @@ -37,6 +37,10 @@ import io.meeds.social.authorization.AuthorizationManagerTest; import io.meeds.social.core.search.SpaceSearchConnectorTest; import io.meeds.social.image.plugin.ImageAttachmentPluginTest; +import io.meeds.social.space.template.plugin.attachment.SpaceTemplateBannerAttachmentPluginTest; +import io.meeds.social.space.template.plugin.translation.SpaceTemplateTranslationPluginTest; +import io.meeds.social.space.template.service.SpaceTemplateServiceTest; +import io.meeds.social.space.template.storage.SpaceTemplateStorageTest; import io.meeds.social.upgrade.SpaceNavigationIconUpgradePluginTest; @RunWith(Suite.class) @@ -58,6 +62,10 @@ SpaceNavigationIconUpgradePluginTest.class, AuthorizationManagerTest.class, SpaceSearchConnectorTest.class, + SpaceTemplateBannerAttachmentPluginTest.class, + SpaceTemplateTranslationPluginTest.class, + SpaceTemplateStorageTest.class, + SpaceTemplateServiceTest.class, }) public class NoContainerTestSuite { diff --git a/component/service/src/main/java/io/meeds/social/space/template/rest/SpaceTemplateRest.java b/component/service/src/main/java/io/meeds/social/space/template/rest/SpaceTemplateRest.java new file mode 100644 index 00000000000..ac4eb20404d --- /dev/null +++ b/component/service/src/main/java/io/meeds/social/space/template/rest/SpaceTemplateRest.java @@ -0,0 +1,164 @@ +/** + * This file is part of the Meeds project (https://meeds.io/). + * + * Copyright (C) 2020 - 2024 Meeds Association contact@meeds.io + * + * This program is free software; you can redistribute it and/or + * modify it under the terms of the GNU Lesser General Public + * License as published by the Free Software Foundation; either + * version 3 of the License, or (at your option) any later version. + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU + * Lesser General Public License for more details. + * + * You should have received a copy of the GNU Lesser General Public License + * along with this program; if not, write to the Free Software Foundation, + * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. + */ +package io.meeds.social.space.template.rest; + +import java.util.List; + +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.data.domain.Pageable; +import org.springframework.http.HttpStatus; +import org.springframework.security.access.annotation.Secured; +import org.springframework.web.bind.annotation.DeleteMapping; +import org.springframework.web.bind.annotation.GetMapping; +import org.springframework.web.bind.annotation.PathVariable; +import org.springframework.web.bind.annotation.PostMapping; +import org.springframework.web.bind.annotation.PutMapping; +import org.springframework.web.bind.annotation.RequestBody; +import org.springframework.web.bind.annotation.RequestMapping; +import org.springframework.web.bind.annotation.RequestParam; +import org.springframework.web.bind.annotation.RestController; +import org.springframework.web.server.ResponseStatusException; + +import org.exoplatform.commons.exception.ObjectNotFoundException; + +import io.meeds.social.space.template.model.SpaceTemplate; +import io.meeds.social.space.template.model.SpaceTemplateFilter; +import io.meeds.social.space.template.service.SpaceTemplateService; + +import io.swagger.v3.oas.annotations.Operation; +import io.swagger.v3.oas.annotations.Parameter; +import io.swagger.v3.oas.annotations.responses.ApiResponse; +import io.swagger.v3.oas.annotations.responses.ApiResponses; +import io.swagger.v3.oas.annotations.tags.Tag; +import jakarta.servlet.http.HttpServletRequest; + +@RestController +@RequestMapping("/space/templates") +@Tag(name = "/social/rest/space/templates", description = "Managing space templates") +public class SpaceTemplateRest { + + @Autowired + private SpaceTemplateService spaceTemplateService; + + @GetMapping + @Secured("users") + @Operation(summary = "Retrieve space templates", method = "GET", description = "This retrieves space templates") + @ApiResponses(value = { @ApiResponse(responseCode = "200", description = "Request fulfilled"), }) + public List getSpaceTemplates(HttpServletRequest request, + Pageable pageable, + @Parameter(description = "Whether include disabled templates or not") + @RequestParam("includeDisabled") + boolean includeDisabled) { + SpaceTemplateFilter spaceTemplateFilter = new SpaceTemplateFilter(request.getRemoteUser(), + request.getLocale(), + includeDisabled); + return spaceTemplateService.getSpaceTemplates(spaceTemplateFilter, pageable, true); + } + + @GetMapping("{id}") + @Secured("users") + @Operation(summary = "Retrieve a Space template designated by its id", method = "GET", + description = "This will retrieve a Space template designated by its id") + @ApiResponses(value = { + @ApiResponse(responseCode = "200", description = "Request fulfilled"), + @ApiResponse(responseCode = "403", description = "Forbidden"), + @ApiResponse(responseCode = "404", description = "Not found"), + }) + public SpaceTemplate getSpaceTemplate(HttpServletRequest request, + @Parameter(description = "Space template identifier") + @PathVariable("id") + long id) { + try { + SpaceTemplate spaceTemplate = spaceTemplateService.getSpaceTemplate(id, request.getRemoteUser(), request.getLocale(), true); + if (spaceTemplate == null) { + throw new ResponseStatusException(HttpStatus.NOT_FOUND); + } + return spaceTemplate; + } catch (IllegalAccessException e) { + throw new ResponseStatusException(HttpStatus.FORBIDDEN, e.getMessage()); + } + } + + @PostMapping + @Secured("users") + @Operation(summary = "Create a Space template", method = "POST", description = "This creates a new Space template") + @ApiResponses(value = { + @ApiResponse(responseCode = "200", description = "Request fulfilled"), + @ApiResponse(responseCode = "403", description = "Forbidden"), + }) + public SpaceTemplate createSpaceTemplate(HttpServletRequest request, + @RequestBody + SpaceTemplate spaceTemplate) { + try { + return spaceTemplateService.createSpaceTemplate(spaceTemplate, request.getRemoteUser()); + } catch (IllegalArgumentException e) { + throw new ResponseStatusException(HttpStatus.BAD_REQUEST, e.getMessage()); + } catch (IllegalAccessException e) { + throw new ResponseStatusException(HttpStatus.FORBIDDEN, e.getMessage()); + } + } + + @PutMapping("{id}") + @Secured("users") + @Operation(summary = "Update a Space template", method = "PUT", description = "This updates an existing Space template") + @ApiResponses(value = { + @ApiResponse(responseCode = "200", description = "Request fulfilled"), + @ApiResponse(responseCode = "403", description = "Forbidden"), + @ApiResponse(responseCode = "404", description = "Not found"), + }) + public void updateSpaceTemplate(HttpServletRequest request, + @Parameter(description = "Space template identifier") + @PathVariable("id") + long id, + @RequestBody + SpaceTemplate spaceTemplate) { + try { + spaceTemplate.setId(id); + spaceTemplateService.updateSpaceTemplate(spaceTemplate, request.getRemoteUser()); + } catch (IllegalArgumentException e) { + throw new ResponseStatusException(HttpStatus.BAD_REQUEST, e.getMessage()); + } catch (ObjectNotFoundException e) { + throw new ResponseStatusException(HttpStatus.NOT_FOUND, e.getMessage()); + } catch (IllegalAccessException e) { + throw new ResponseStatusException(HttpStatus.FORBIDDEN, e.getMessage()); + } + } + + @DeleteMapping("{id}") + @Secured("users") + @Operation(summary = "Deletes a Space template", method = "DELETE", description = "This deletes an existing Space template") + @ApiResponses(value = { + @ApiResponse(responseCode = "200", description = "Request fulfilled"), + @ApiResponse(responseCode = "403", description = "Forbidden"), + @ApiResponse(responseCode = "404", description = "Not found"), + }) + public void deleteSpaceTemplate(HttpServletRequest request, + @Parameter(description = "Space template identifier") + @PathVariable("id") + long id) { + try { + spaceTemplateService.deleteSpaceTemplate(id, request.getRemoteUser()); + } catch (ObjectNotFoundException e) { + throw new ResponseStatusException(HttpStatus.NOT_FOUND, e.getMessage()); + } catch (IllegalAccessException e) { + throw new ResponseStatusException(HttpStatus.FORBIDDEN, e.getMessage()); + } + } + +} diff --git a/component/service/src/main/java/org/exoplatform/social/rest/impl/space/SpaceRest.java b/component/service/src/main/java/org/exoplatform/social/rest/impl/space/SpaceRest.java index 698593feb58..83cb5f3c623 100644 --- a/component/service/src/main/java/org/exoplatform/social/rest/impl/space/SpaceRest.java +++ b/component/service/src/main/java/org/exoplatform/social/rest/impl/space/SpaceRest.java @@ -437,6 +437,19 @@ public Response getSpaceById( return buildSpaceResponse(space, expand, uriInfo, request); } + @GET + @Path("countByTemplate") + @Produces(MediaType.APPLICATION_JSON) + @RolesAllowed("administrators") + @Operation( + summary = "Gets the spaces count by Space Template", + method = "GET", + description = "This returns the spaces count by Space template identifier") + @ApiResponses(value = { @ApiResponse(responseCode = "200", description = "Request fulfilled") }) + public Response countSpacesByTemplate() { + return Response.ok(spaceService.countSpacesByTemplate()).build(); + } + @GET @Path("byPrettyName/{prettyName}") @RolesAllowed("users") diff --git a/component/service/src/main/java/org/exoplatform/social/rest/impl/spacetemplates/SpaceTemplatesRest.java b/component/service/src/main/java/org/exoplatform/social/rest/impl/spacetemplates/SpaceTemplatesRest.java index 68d918bf7f1..7169c40bc56 100644 --- a/component/service/src/main/java/org/exoplatform/social/rest/impl/spacetemplates/SpaceTemplatesRest.java +++ b/component/service/src/main/java/org/exoplatform/social/rest/impl/spacetemplates/SpaceTemplatesRest.java @@ -49,6 +49,8 @@ import org.exoplatform.social.rest.api.RestUtils; import org.exoplatform.social.service.rest.api.VersionResources; +import io.meeds.social.space.template.rest.SpaceTemplateRest; + import io.swagger.v3.oas.annotations.Operation; import io.swagger.v3.oas.annotations.Parameter; import io.swagger.v3.oas.annotations.responses.ApiResponse; @@ -57,12 +59,12 @@ import jakarta.servlet.http.HttpServletRequest; /** - * * Provides REST Services for manipulating space templates. - * + * @deprecated will be replaced by {@link SpaceTemplateRest} */ @Path(VersionResources.VERSION_ONE + "/social/spaceTemplates") @Tag(name = VersionResources.VERSION_ONE + "/social/spaceTemplates", description = "Managing Spaces Templates") +@Deprecated(forRemoval = true) public class SpaceTemplatesRest implements ResourceContainer { private static final Log LOG = ExoLogger.getLogger(SpaceTemplatesRest.class); diff --git a/component/service/src/test/java/org/exoplatform/social/rest/impl/activity/ActivityRestResourcesTest.java b/component/service/src/test/java/org/exoplatform/social/rest/impl/activity/ActivityRestResourcesTest.java index fbafd4e2c36..e7c4b3a268e 100644 --- a/component/service/src/test/java/org/exoplatform/social/rest/impl/activity/ActivityRestResourcesTest.java +++ b/component/service/src/test/java/org/exoplatform/social/rest/impl/activity/ActivityRestResourcesTest.java @@ -28,8 +28,6 @@ public class ActivityRestResourcesTest extends AbstractResourceTest { private IdentityStorage identityStorage; - private IdentityManager identityManager; - private ActivityManager activityManager; private RelationshipManager relationshipManager; @@ -46,13 +44,12 @@ public class ActivityRestResourcesTest extends AbstractResourceTest { private Identity testSpaceIdentity; - public void setUp() throws Exception { + public void setUp() throws Exception { super.setUp(); System.setProperty("gatein.email.domain.url", "localhost:8080"); identityStorage = getContainer().getComponentInstanceOfType(IdentityStorage.class); - identityManager = getContainer().getComponentInstanceOfType(IdentityManager.class); activityManager = getContainer().getComponentInstanceOfType(ActivityManager.class); relationshipManager = getContainer().getComponentInstanceOfType(RelationshipManager.class); spaceService = getContainer().getComponentInstanceOfType(SpaceService.class); diff --git a/component/service/src/test/java/org/exoplatform/social/service/test/MockSpacesAdministrationService.java b/component/service/src/test/java/org/exoplatform/social/service/test/MockSpacesAdministrationService.java index d308ebf0031..273f90c9be1 100644 --- a/component/service/src/test/java/org/exoplatform/social/service/test/MockSpacesAdministrationService.java +++ b/component/service/src/test/java/org/exoplatform/social/service/test/MockSpacesAdministrationService.java @@ -1,6 +1,9 @@ package org.exoplatform.social.service.test; import org.apache.commons.lang3.StringUtils; + +import org.exoplatform.container.ExoContainerContext; +import org.exoplatform.portal.config.UserACL; import org.exoplatform.services.security.IdentityConstants; import org.exoplatform.services.security.MembershipEntry; import org.exoplatform.social.core.space.SpacesAdministrationService; @@ -12,7 +15,7 @@ public class MockSpacesAdministrationService implements SpacesAdministrationServ private List spacesAdministrators = new ArrayList<>(); - private List spacesCreators = new ArrayList<>(); + private List spacesCreators = new ArrayList<>(); @Override public List getSpacesAdministratorsMemberships() { @@ -21,7 +24,7 @@ public List getSpacesAdministratorsMemberships() { @Override public void updateSpacesAdministratorsMemberships(List permissionsExpressions) { - if(permissionsExpressions != null) { + if (permissionsExpressions != null) { spacesAdministrators = new ArrayList<>(permissionsExpressions); } else { spacesAdministrators.clear(); @@ -35,7 +38,7 @@ public List getSpacesCreatorsMemberships() { @Override public void updateSpacesCreatorsMemberships(List permissionsExpressions) { - if(permissionsExpressions != null) { + if (permissionsExpressions != null) { spacesCreators = new ArrayList<>(permissionsExpressions); } else { spacesCreators.clear(); @@ -44,11 +47,25 @@ public void updateSpacesCreatorsMemberships(List permissionsExp @Override public boolean canCreateSpace(String username) { - if (StringUtils.isBlank(username) || IdentityConstants.ANONIM.equals(username) || IdentityConstants.SYSTEM.equals(username)) { + return StringUtils.isNotBlank(username) + && !IdentityConstants.ANONIM.equals(username) + && !IdentityConstants.SYSTEM.equals(username); + } + + @Override + public boolean isSuperManager(String username) { + UserACL userAcl = ExoContainerContext.getService(UserACL.class); + if (StringUtils.isBlank(username) + || IdentityConstants.ANONIM.equals(username) + || IdentityConstants.SYSTEM.equals(username)) { return false; - } - else { + } else if (username.equals(userAcl.getSuperUser())) { return true; } + org.exoplatform.services.security.Identity identity = userAcl.getUserIdentity(username); + return identity != null && (identity.isMemberOf(userAcl.getAdminGroups()) + || getSpacesAdministratorsMemberships().stream() + .anyMatch(identity::isMemberOf)); } + } diff --git a/crowdin.yml b/crowdin.yml index 46576384da6..3c84dc1c3f4 100644 --- a/crowdin.yml +++ b/crowdin.yml @@ -71,9 +71,9 @@ files: [ "escape_special_characters": 0, "escape_quotes" : 0, }, - # extension + # webapp { - "source" : "/extension/war/src/main/resources/locale/notification/template/Notification_en.properties", + "source" : "/webapp/src/main/resources/locale/notification/template/Notification_en.properties", "translation" : "%original_path%/%file_name%!_%locale_with_underscore%.%file_extension%", "translation_replace" : { @@ -91,7 +91,7 @@ files: [ "escape_quotes" : 0, }, { - "source" : "/extension/war/src/main/resources/locale/portal/HamburgerMenu_en.properties", + "source" : "/webapp/src/main/resources/locale/portal/HamburgerMenu_en.properties", "translation" : "%original_path%/%file_name%!_%locale_with_underscore%.%file_extension%", "translation_replace" : { @@ -108,9 +108,8 @@ files: [ "escape_special_characters": 0, "escape_quotes" : 0, }, - # webapp { - "source": "/webapp/portlet/src/main/resources/locale/portal_en.properties", + "source": "/webapp/src/main/resources/locale/portal_en.properties", "translation": "%original_path%/%file_name%!_%locale_with_underscore%.%file_extension%", "translation_replace": { @@ -128,7 +127,7 @@ files: [ "escape_quotes": 0, }, { - "source": "/webapp/portlet/src/main/resources/locale/portlet/Portlets_en.properties", + "source": "/webapp/src/main/resources/locale/portlet/Portlets_en.properties", "translation": "%original_path%/%file_name%!_%locale_with_underscore%.%file_extension%", "translation_replace": { @@ -140,13 +139,13 @@ files: [ "sv_SE": "sv_SE","th_TH": "th","tl_PH": "tl","tr_TR": "tr","uk_UA": "uk","ur_IN": "ur_IN","vi_VN": "vi", "zh_CN": "zh_CN","zh_TW": "zh_TW", }, - "dest": "hub/social/webapp/Portlets.properties", + "dest": "hub/social/webapps.properties", "update_option": "update_as_unapproved", "escape_special_characters": 0, "escape_quotes": 0, }, { - "source": "/webapp/portlet/src/main/resources/locale/portlet/GeneralSettings_en.properties", + "source": "/webapp/src/main/resources/locale/portlet/GeneralSettings_en.properties", "translation": "%original_path%/%file_name%!_%locale_with_underscore%.%file_extension%", "translation_replace": { @@ -164,7 +163,7 @@ files: [ "escape_quotes": 0, }, { - "source": "/webapp/portlet/src/main/resources/locale/portlet/Image_en.properties", + "source": "/webapp/src/main/resources/locale/portlet/Image_en.properties", "translation": "%original_path%/%file_name%!_%locale_with_underscore%.%file_extension%", "translation_replace": { @@ -182,7 +181,7 @@ files: [ "escape_quotes": 0, }, { - "source": "/webapp/portlet/src/main/resources/locale/portlet/social/SpaceInfosPortlet_en.properties", + "source": "/webapp/src/main/resources/locale/portlet/social/SpaceInfosPortlet_en.properties", "translation": "%original_path%/%file_name%!_%locale_with_underscore%.%file_extension%", "translation_replace": { @@ -200,7 +199,7 @@ files: [ "escape_quotes": 0, }, { - "source": "/webapp/portlet/src/main/resources/locale/portlet/whoisonline/whoisonline_en.properties", + "source": "/webapp/src/main/resources/locale/portlet/whoisonline/whoisonline_en.properties", "translation": "%original_path%/%file_name%!_%locale_with_underscore%.%file_extension%", "translation_replace": { @@ -218,7 +217,7 @@ files: [ "escape_quotes": 0, }, { - "source": "/webapp/portlet/src/main/resources/locale/portlet/social/GettingStartedPortlet_en.properties", + "source": "/webapp/src/main/resources/locale/portlet/social/GettingStartedPortlet_en.properties", "translation": "%original_path%/%file_name%!_%locale_with_underscore%.%file_extension%", "translation_replace": { @@ -236,7 +235,7 @@ files: [ "escape_quotes": 0, }, { - "source": "/webapp/portlet/src/main/resources/locale/portlet/social/ExternalSpacesListApplication_en.properties", + "source": "/webapp/src/main/resources/locale/portlet/social/ExternalSpacesListApplication_en.properties", "translation": "%original_path%/%file_name%!_%locale_with_underscore%.%file_extension%", "translation_replace": { @@ -254,7 +253,7 @@ files: [ "escape_quotes": 0, }, { - "source": "/webapp/portlet/src/main/resources/locale/portlet/social/PeopleListApplication_en.properties", + "source": "/webapp/src/main/resources/locale/portlet/social/PeopleListApplication_en.properties", "translation": "%original_path%/%file_name%!_%locale_with_underscore%.%file_extension%", "translation_replace": { @@ -272,7 +271,7 @@ files: [ "escape_quotes": 0, }, { - "source": "/webapp/portlet/src/main/resources/locale/portlet/social/PeopleOverview_en.properties", + "source": "/webapp/src/main/resources/locale/portlet/social/PeopleOverview_en.properties", "translation": "%original_path%/%file_name%!_%locale_with_underscore%.%file_extension%", "translation_replace": { @@ -290,7 +289,7 @@ files: [ "escape_quotes": 0, }, { - "source": "/webapp/portlet/src/main/resources/locale/portlet/social/ProfileAboutMe_en.properties", + "source": "/webapp/src/main/resources/locale/portlet/social/ProfileAboutMe_en.properties", "translation": "%original_path%/%file_name%!_%locale_with_underscore%.%file_extension%", "translation_replace": { @@ -308,7 +307,7 @@ files: [ "escape_quotes": 0, }, { - "source": "/webapp/portlet/src/main/resources/locale/portlet/social/ProfileContactInformation_en.properties", + "source": "/webapp/src/main/resources/locale/portlet/social/ProfileContactInformation_en.properties", "translation": "%original_path%/%file_name%!_%locale_with_underscore%.%file_extension%", "translation_replace": { @@ -326,7 +325,7 @@ files: [ "escape_quotes": 0, }, { - "source": "/webapp/portlet/src/main/resources/locale/portlet/social/ProfileHeader_en.properties", + "source": "/webapp/src/main/resources/locale/portlet/social/ProfileHeader_en.properties", "translation": "%original_path%/%file_name%!_%locale_with_underscore%.%file_extension%", "translation_replace": { @@ -344,7 +343,7 @@ files: [ "escape_quotes": 0, }, { - "source": "/webapp/portlet/src/main/resources/locale/portlet/social/ProfileWorkExperience_en.properties", + "source": "/webapp/src/main/resources/locale/portlet/social/ProfileWorkExperience_en.properties", "translation": "%original_path%/%file_name%!_%locale_with_underscore%.%file_extension%", "translation_replace": { @@ -362,7 +361,7 @@ files: [ "escape_quotes": 0, }, { - "source": "/webapp/portlet/src/main/resources/locale/portlet/social/SpacesAdministrationPortlet_en.properties", + "source": "/webapp/src/main/resources/locale/portlet/social/SpacesAdministrationPortlet_en.properties", "translation": "%original_path%/%file_name%!_%locale_with_underscore%.%file_extension%", "translation_replace": { @@ -380,7 +379,7 @@ files: [ "escape_quotes": 0, }, { - "source": "/webapp/portlet/src/main/resources/locale/portlet/social/SpacesListApplication_en.properties", + "source": "/webapp/src/main/resources/locale/portlet/social/SpacesListApplication_en.properties", "translation": "%original_path%/%file_name%!_%locale_with_underscore%.%file_extension%", "translation_replace": { @@ -398,7 +397,7 @@ files: [ "escape_quotes": 0, }, { - "source": "/webapp/portlet/src/main/resources/locale/portlet/social/UserPopup_en.properties", + "source": "/webapp/src/main/resources/locale/portlet/social/UserPopup_en.properties", "translation": "%original_path%/%file_name%!_%locale_with_underscore%.%file_extension%", "translation_replace": { @@ -416,7 +415,7 @@ files: [ "escape_quotes": 0, }, { - "source": "/webapp/portlet/src/main/resources/locale/portlet/social/SpacesOverview_en.properties", + "source": "/webapp/src/main/resources/locale/portlet/social/SpacesOverview_en.properties", "translation": "%original_path%/%file_name%!_%locale_with_underscore%.%file_extension%", "translation_replace": { @@ -434,7 +433,7 @@ files: [ "escape_quotes": 0, }, { - "source": "/webapp/portlet/src/main/resources/locale/portlet/social/SuggestionsPortlet_en.properties", + "source": "/webapp/src/main/resources/locale/portlet/social/SuggestionsPortlet_en.properties", "translation": "%original_path%/%file_name%!_%locale_with_underscore%.%file_extension%", "translation_replace": { @@ -452,7 +451,7 @@ files: [ "escape_quotes": 0, }, { - "source": "/webapp/portlet/src/main/resources/locale/portlet/social/ComplementaryFilter_en.properties", + "source": "/webapp/src/main/resources/locale/portlet/social/ComplementaryFilter_en.properties", "translation": "%original_path%/%file_name%!_%locale_with_underscore%.%file_extension%", "translation_replace": { @@ -470,7 +469,7 @@ files: [ "escape_quotes": 0, }, { - "source": "/webapp/portlet/src/main/resources/locale/portlet/social/UserSettings_en.properties", + "source": "/webapp/src/main/resources/locale/portlet/social/UserSettings_en.properties", "translation": "%original_path%/%file_name%!_%locale_with_underscore%.%file_extension%", "translation_replace": { @@ -488,7 +487,7 @@ files: [ "escape_quotes": 0, }, { - "source": "/webapp/portlet/src/main/resources/locale/portlet/Login_en.properties", + "source": "/webapp/src/main/resources/locale/portlet/Login_en.properties", "translation": "%original_path%/%file_name%!_%locale_with_underscore%.%file_extension%", "translation_replace": { @@ -506,7 +505,7 @@ files: [ "escape_quotes": 0, }, { - "source": "/webapp/portlet/src/main/resources/locale/portlet/UserNotificationPortlet_en.properties", + "source": "/webapp/src/main/resources/locale/portlet/UserNotificationPortlet_en.properties", "translation": "%original_path%/%file_name%!_%locale_with_underscore%.%file_extension%", "translation_replace": { @@ -524,7 +523,7 @@ files: [ "escape_quotes": 0, }, { - "source": "/webapp/portlet/src/main/resources/locale/portlet/NotificationAdministration_en.properties", + "source": "/webapp/src/main/resources/locale/portlet/NotificationAdministration_en.properties", "translation": "%original_path%/%file_name%!_%locale_with_underscore%.%file_extension%", "translation_replace": { @@ -542,7 +541,7 @@ files: [ "escape_quotes": 0, }, { - "source": "/webapp/portlet/src/main/resources/locale/portlet/Links_en.properties", + "source": "/webapp/src/main/resources/locale/portlet/Links_en.properties", "translation": "%original_path%/%file_name%!_%locale_with_underscore%.%file_extension%", "translation_replace": { @@ -560,7 +559,7 @@ files: [ "escape_quotes": 0, }, { - "source": "/webapp/portlet/src/main/resources/locale/portlet/PlatformSettings_en.properties", + "source": "/webapp/src/main/resources/locale/portlet/PlatformSettings_en.properties", "translation": "%original_path%/%file_name%!_%locale_with_underscore%.%file_extension%", "translation_replace": { @@ -577,4 +576,21 @@ files: [ "escape_special_characters": 0, "escape_quotes": 0, }, + { + "source": "/webapp/src/main/resources/locale/portlet/SpaceTemplatesManagement_en.properties", + "translation": "%original_path%/%file_name%!_%locale_with_underscore%.%file_extension%", + "translation_replace": { + "_en!": "","ar_SA": "ar","ar_OM": "aro","az_AZ": "az","ca_ES": "ca","ceb_PH": "ceb", + "co_FR": "co","cs_CZ": "cs","de_DE": "de","el_GR": "el","en_US": "en","es_ES": "es_ES","eu_ES": "eu","fa_IR": "fa", + "fi_FI": "fi","fil_PH": "fil","fr_FR": "fr","hi_IN": "hi","hu_HU": "hu","id_ID": "id","it_IT": "it","ja_JP": "ja", + "kab_KAB": "kab","ko_KR": "ko","lt_LT": "lt","ms_MY": "ms","nl_NL": "nl","no_NO": "no","pcm_NG": "pcm","pl_PL": "pl", + "pt_BR": "pt_BR","pt_PT": "pt_PT","ro_RO": "ro","ru_RU": "ru","sk_SK": "sk","sl_SI": "sl","sq_AL": "sq", + "sv_SE": "sv_SE","th_TH": "th","tl_PH": "tl","tr_TR": "tr","uk_UA": "uk","ur_IN": "ur_IN","vi_VN": "vi", + "zh_CN": "zh_CN","zh_TW": "zh_TW", + }, + "dest": "hub/social/webapp/SpaceTemplatesManagement.properties", + "update_option": "update_as_unapproved", + "escape_special_characters": 0, + "escape_quotes": 0, + }, ] diff --git a/translations.properties b/translations.properties deleted file mode 100644 index f8cc735f808..00000000000 --- a/translations.properties +++ /dev/null @@ -1,59 +0,0 @@ -# -# Copyright (C) 2003-2013 eXo Platform SAS. -# -# This is free software; you can redistribute it and/or modify it -# under the terms of the GNU Lesser General Public License as -# published by the Free Software Foundation; either version 3 of -# the License, or (at your option) any later version. -# -# This software is distributed in the hope that it will be useful, -# but WITHOUT ANY WARRANTY; without even the implied warranty of -# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU -# Lesser General Public License for more details. -# -# You should have received a copy of the GNU Lesser General Public -# License along with this software; if not, write to the Free -# Software Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA -# 02110-1301 USA, or see the FSF site: http://www.fsf.org. -# - -# Mandatory property that indicate the base directory where each file is located in Crowdin (must end with /). -baseDir=platform/social/ - -# Files -# component -component/Core.properties=component/core/src/main/resources/locale/social/Core_en.properties -component/Webui.properties=component/service/src/main/resources/locale/social/Webui_en.properties - -menu/webui.properties=component/service/src/main/resources/locale/social/menu/webui_en.properties - -# extension -extension/Notification.properties=extension/war/src/main/resources/locale/notification/template/Notification_en.properties -extension/hamburgerMenu.properties=extension/war/src/main/resources/locale/portal/HamburgerMenu_en.properties - -# webapp -webapp/portal.properties=webapp/portlet/src/main/resources/locale/portal_en.properties -webapp/Portlets.properties=webapp/portlet/src/main/resources/locale/portlet/Portlets_en.properties -webapp/GeneralSettings.properties=webapp/portlet/src/main/resources/locale/portlet/GeneralSettings_en.properties -webapp/SpaceInfosPortlet.properties=webapp/portlet/src/main/resources/locale/portlet/social/SpaceInfosPortlet_en.properties -webapp/whoisonline.properties=webapp/portlet/src/main/resources/locale/portlet/whoisonline/whoisonline_en.properties -webapp/GettingStarted.properties=webapp/portlet/src/main/resources/locale/portlet/social/GettingStartedPortlet_en.properties -webapp/ExternalSpacesListApplication.properties=webapp/portlet/src/main/resources/locale/portlet/social/ExternalSpacesListApplication_en.properties -webapp/PeopleListApplication.properties=webapp/portlet/src/main/resources/locale/portlet/social/PeopleListApplication_en.properties -webapp/PeopleOverview.properties=webapp/portlet/src/main/resources/locale/portlet/social/PeopleOverview_en.properties -webapp/ProfileAboutMe.properties=webapp/portlet/src/main/resources/locale/portlet/social/ProfileAboutMe_en.properties -webapp/ProfileContactInformation.properties=webapp/portlet/src/main/resources/locale/portlet/social/ProfileContactInformation_en.properties -webapp/ProfileHeader.properties=webapp/portlet/src/main/resources/locale/portlet/social/ProfileHeader_en.properties -webapp/ProfileWorkExperience.properties=webapp/portlet/src/main/resources/locale/portlet/social/ProfileWorkExperience_en.properties -webapp/SpaceAdministrationPortlet.properties=webapp/portlet/src/main/resources/locale/portlet/social/SpacesAdministrationPortlet_en.properties -webapp/SpacesListApplication.properties=webapp/portlet/src/main/resources/locale/portlet/social/SpacesListApplication_en.properties -webapp/UserPopup.properties=webapp/portlet/src/main/resources/locale/portlet/social/UserPopup_en.properties -webapp/SpacesOverview.properties=webapp/portlet/src/main/resources/locale/portlet/social/SpacesOverview_en.properties -webapp/SuggestionsPortlet.properties=webapp/portlet/src/main/resources/locale/portlet/social/SuggestionsPortlet_en.properties -webapp/UserSettings.properties=webapp/portlet/src/main/resources/locale/portlet/social/UserSettings_en.properties -webapp/Login.properties=webapp/portlet/src/main/resources/locale/portlet/Login_en.properties -webapp/UserNotificationPortlet.properties=webapp/portlet/src/main/resources/locale/portlet/UserNotificationPortlet_en.properties -webapp/NotificationAdministration.properties=webapp/portlet/src/main/resources/locale/portlet/NotificationAdministration_en.properties -webapp/Links.properties=webapp/portlet/src/main/resources/locale/portlet/Links_en.properties -webapp/PlatformSettings.properties=webapp/portlet/src/main/resources/locale/portlet/PlatformSettings_en.properties -webapp/ComplementaryFilter.properties=webapp/portlet/src/main/resources/locale/portlet/social/ComplementaryFilter_en.properties diff --git a/webapp/package-lock.json b/webapp/package-lock.json index 6e7c5641538..1a5e83326c5 100644 --- a/webapp/package-lock.json +++ b/webapp/package-lock.json @@ -1,12 +1,12 @@ { "name": "meeds-social", - "version": "6.2.0", - "lockfileVersion": 3, + "version": "7.0.0", + "lockfileVersion": 2, "requires": true, "packages": { "": { "name": "meeds-social", - "version": "6.2.0", + "version": "7.0.0", "license": "LGPL", "devDependencies": { "babel-loader": "^8.3.0", @@ -874,6 +874,8 @@ "resolved": "https://registry.npmjs.org/ajv/-/ajv-8.17.1.tgz", "integrity": "sha512-B/gBuNg5SiMTrPkC+A2+cW0RszwxYmn6VYxB/inlBStS5nx6xHIt/ehKRhIMhqusl7a8LjQoZnjCs5vhwxOQ1g==", "dev": true, + "optional": true, + "peer": true, "dependencies": { "fast-deep-equal": "^3.1.3", "fast-uri": "^3.0.1", @@ -889,7 +891,9 @@ "version": "1.0.0", "resolved": "https://registry.npmjs.org/json-schema-traverse/-/json-schema-traverse-1.0.0.tgz", "integrity": "sha512-NM8/P9n3XjXhIZn1lLhkFaACTOURQXjWhV4BA/RnOv8xvgqtqpAX9IO4mRQxSx1Rlo4tqzeqb0sOlruaOy3dug==", - "dev": true + "dev": true, + "optional": true, + "peer": true }, "node_modules/ajv-keywords": { "version": "3.5.2", @@ -4013,5 +4017,2880 @@ "url": "https://github.com/sponsors/sindresorhus" } } + }, + "dependencies": { + "@ampproject/remapping": { + "version": "2.3.0", + "resolved": "https://registry.npmjs.org/@ampproject/remapping/-/remapping-2.3.0.tgz", + "integrity": "sha512-30iZtAPgz+LTIYoeivqYo853f02jBYSd5uGnGpkFV0M3xOt9aN73erkgYAmZU43x4VfqcnLxW9Kpg3R5LC4YYw==", + "dev": true, + "peer": true, + "requires": { + "@jridgewell/gen-mapping": "^0.3.5", + "@jridgewell/trace-mapping": "^0.3.24" + } + }, + "@babel/code-frame": { + "version": "7.24.7", + "resolved": "https://registry.npmjs.org/@babel/code-frame/-/code-frame-7.24.7.tgz", + "integrity": "sha512-BcYH1CVJBO9tvyIZ2jVeXgSIMvGZ2FDRvDdOIVQyuklNKSsx+eppDEBq/g47Ayw+RqNFE+URvOShmf+f/qwAlA==", + "dev": true, + "peer": true, + "requires": { + "@babel/highlight": "^7.24.7", + "picocolors": "^1.0.0" + } + }, + "@babel/compat-data": { + "version": "7.25.2", + "resolved": "https://registry.npmjs.org/@babel/compat-data/-/compat-data-7.25.2.tgz", + "integrity": "sha512-bYcppcpKBvX4znYaPEeFau03bp89ShqNMLs+rmdptMw+heSZh9+z84d2YG+K7cYLbWwzdjtDoW/uqZmPjulClQ==", + "dev": true, + "peer": true + }, + "@babel/core": { + "version": "7.25.2", + "resolved": "https://registry.npmjs.org/@babel/core/-/core-7.25.2.tgz", + "integrity": "sha512-BBt3opiCOxUr9euZ5/ro/Xv8/V7yJ5bjYMqG/C1YAo8MIKAnumZalCN+msbci3Pigy4lIQfPUpfMM27HMGaYEA==", + "dev": true, + "peer": true, + "requires": { + "@ampproject/remapping": "^2.2.0", + "@babel/code-frame": "^7.24.7", + "@babel/generator": "^7.25.0", + "@babel/helper-compilation-targets": "^7.25.2", + "@babel/helper-module-transforms": "^7.25.2", + "@babel/helpers": "^7.25.0", + "@babel/parser": "^7.25.0", + "@babel/template": "^7.25.0", + "@babel/traverse": "^7.25.2", + "@babel/types": "^7.25.2", + "convert-source-map": "^2.0.0", + "debug": "^4.1.0", + "gensync": "^1.0.0-beta.2", + "json5": "^2.2.3", + "semver": "^6.3.1" + } + }, + "@babel/generator": { + "version": "7.25.0", + "resolved": "https://registry.npmjs.org/@babel/generator/-/generator-7.25.0.tgz", + "integrity": "sha512-3LEEcj3PVW8pW2R1SR1M89g/qrYk/m/mB/tLqn7dn4sbBUQyTqnlod+II2U4dqiGtUmkcnAmkMDralTFZttRiw==", + "dev": true, + "peer": true, + "requires": { + "@babel/types": "^7.25.0", + "@jridgewell/gen-mapping": "^0.3.5", + "@jridgewell/trace-mapping": "^0.3.25", + "jsesc": "^2.5.1" + } + }, + "@babel/helper-compilation-targets": { + "version": "7.25.2", + "resolved": "https://registry.npmjs.org/@babel/helper-compilation-targets/-/helper-compilation-targets-7.25.2.tgz", + "integrity": "sha512-U2U5LsSaZ7TAt3cfaymQ8WHh0pxvdHoEk6HVpaexxixjyEquMh0L0YNJNM6CTGKMXV1iksi0iZkGw4AcFkPaaw==", + "dev": true, + "peer": true, + "requires": { + "@babel/compat-data": "^7.25.2", + "@babel/helper-validator-option": "^7.24.8", + "browserslist": "^4.23.1", + "lru-cache": "^5.1.1", + "semver": "^6.3.1" + } + }, + "@babel/helper-module-imports": { + "version": "7.24.7", + "resolved": "https://registry.npmjs.org/@babel/helper-module-imports/-/helper-module-imports-7.24.7.tgz", + "integrity": "sha512-8AyH3C+74cgCVVXow/myrynrAGv+nTVg5vKu2nZph9x7RcRwzmh0VFallJuFTZ9mx6u4eSdXZfcOzSqTUm0HCA==", + "dev": true, + "peer": true, + "requires": { + "@babel/traverse": "^7.24.7", + "@babel/types": "^7.24.7" + } + }, + "@babel/helper-module-transforms": { + "version": "7.25.2", + "resolved": "https://registry.npmjs.org/@babel/helper-module-transforms/-/helper-module-transforms-7.25.2.tgz", + "integrity": "sha512-BjyRAbix6j/wv83ftcVJmBt72QtHI56C7JXZoG2xATiLpmoC7dpd8WnkikExHDVPpi/3qCmO6WY1EaXOluiecQ==", + "dev": true, + "peer": true, + "requires": { + "@babel/helper-module-imports": "^7.24.7", + "@babel/helper-simple-access": "^7.24.7", + "@babel/helper-validator-identifier": "^7.24.7", + "@babel/traverse": "^7.25.2" + } + }, + "@babel/helper-simple-access": { + "version": "7.24.7", + "resolved": "https://registry.npmjs.org/@babel/helper-simple-access/-/helper-simple-access-7.24.7.tgz", + "integrity": "sha512-zBAIvbCMh5Ts+b86r/CjU+4XGYIs+R1j951gxI3KmmxBMhCg4oQMsv6ZXQ64XOm/cvzfU1FmoCyt6+owc5QMYg==", + "dev": true, + "peer": true, + "requires": { + "@babel/traverse": "^7.24.7", + "@babel/types": "^7.24.7" + } + }, + "@babel/helper-string-parser": { + "version": "7.24.8", + "resolved": "https://registry.npmjs.org/@babel/helper-string-parser/-/helper-string-parser-7.24.8.tgz", + "integrity": "sha512-pO9KhhRcuUyGnJWwyEgnRJTSIZHiT+vMD0kPeD+so0l7mxkMT19g3pjY9GTnHySck/hDzq+dtW/4VgnMkippsQ==", + "dev": true + }, + "@babel/helper-validator-identifier": { + "version": "7.24.7", + "resolved": "https://registry.npmjs.org/@babel/helper-validator-identifier/-/helper-validator-identifier-7.24.7.tgz", + "integrity": "sha512-rR+PBcQ1SMQDDyF6X0wxtG8QyLCgUB0eRAGguqRLfkCA87l7yAP7ehq8SNj96OOGTO8OBV70KhuFYcIkHXOg0w==", + "dev": true + }, + "@babel/helper-validator-option": { + "version": "7.24.8", + "resolved": "https://registry.npmjs.org/@babel/helper-validator-option/-/helper-validator-option-7.24.8.tgz", + "integrity": "sha512-xb8t9tD1MHLungh/AIoWYN+gVHaB9kwlu8gffXGSt3FFEIT7RjS+xWbc2vUD1UTZdIpKj/ab3rdqJ7ufngyi2Q==", + "dev": true, + "peer": true + }, + "@babel/helpers": { + "version": "7.25.0", + "resolved": "https://registry.npmjs.org/@babel/helpers/-/helpers-7.25.0.tgz", + "integrity": "sha512-MjgLZ42aCm0oGjJj8CtSM3DB8NOOf8h2l7DCTePJs29u+v7yO/RBX9nShlKMgFnRks/Q4tBAe7Hxnov9VkGwLw==", + "dev": true, + "peer": true, + "requires": { + "@babel/template": "^7.25.0", + "@babel/types": "^7.25.0" + } + }, + "@babel/highlight": { + "version": "7.24.7", + "resolved": "https://registry.npmjs.org/@babel/highlight/-/highlight-7.24.7.tgz", + "integrity": "sha512-EStJpq4OuY8xYfhGVXngigBJRWxftKX9ksiGDnmlY3o7B/V7KIAc9X4oiK87uPJSc/vs5L869bem5fhZa8caZw==", + "dev": true, + "peer": true, + "requires": { + "@babel/helper-validator-identifier": "^7.24.7", + "chalk": "^2.4.2", + "js-tokens": "^4.0.0", + "picocolors": "^1.0.0" + } + }, + "@babel/parser": { + "version": "7.25.3", + "resolved": "https://registry.npmjs.org/@babel/parser/-/parser-7.25.3.tgz", + "integrity": "sha512-iLTJKDbJ4hMvFPgQwwsVoxtHyWpKKPBrxkANrSYewDPaPpT5py5yeVkgPIJ7XYXhndxJpaA3PyALSXQ7u8e/Dw==", + "dev": true, + "requires": { + "@babel/types": "^7.25.2" + } + }, + "@babel/template": { + "version": "7.25.0", + "resolved": "https://registry.npmjs.org/@babel/template/-/template-7.25.0.tgz", + "integrity": "sha512-aOOgh1/5XzKvg1jvVz7AVrx2piJ2XBi227DHmbY6y+bM9H2FlN+IfecYu4Xl0cNiiVejlsCri89LUsbj8vJD9Q==", + "dev": true, + "peer": true, + "requires": { + "@babel/code-frame": "^7.24.7", + "@babel/parser": "^7.25.0", + "@babel/types": "^7.25.0" + } + }, + "@babel/traverse": { + "version": "7.25.3", + "resolved": "https://registry.npmjs.org/@babel/traverse/-/traverse-7.25.3.tgz", + "integrity": "sha512-HefgyP1x754oGCsKmV5reSmtV7IXj/kpaE1XYY+D9G5PvKKoFfSbiS4M77MdjuwlZKDIKFCffq9rPU+H/s3ZdQ==", + "dev": true, + "peer": true, + "requires": { + "@babel/code-frame": "^7.24.7", + "@babel/generator": "^7.25.0", + "@babel/parser": "^7.25.3", + "@babel/template": "^7.25.0", + "@babel/types": "^7.25.2", + "debug": "^4.3.1", + "globals": "^11.1.0" + } + }, + "@babel/types": { + "version": "7.25.2", + "resolved": "https://registry.npmjs.org/@babel/types/-/types-7.25.2.tgz", + "integrity": "sha512-YTnYtra7W9e6/oAZEHj0bJehPRUlLH9/fbpT5LfB0NhQXyALCRkRs3zH9v07IYhkgpqX6Z78FnuccZr/l4Fs4Q==", + "dev": true, + "requires": { + "@babel/helper-string-parser": "^7.24.8", + "@babel/helper-validator-identifier": "^7.24.7", + "to-fast-properties": "^2.0.0" + } + }, + "@discoveryjs/json-ext": { + "version": "0.5.7", + "resolved": "https://registry.npmjs.org/@discoveryjs/json-ext/-/json-ext-0.5.7.tgz", + "integrity": "sha512-dBVuXR082gk3jsFp7Rd/JI4kytwGHecnCoTtXFb7DB6CNHp4rg5k1bhg0nWdLGLnOV71lmDzGQaLMy8iPLY0pw==", + "dev": true + }, + "@eslint-community/eslint-utils": { + "version": "4.4.0", + "resolved": "https://registry.npmjs.org/@eslint-community/eslint-utils/-/eslint-utils-4.4.0.tgz", + "integrity": "sha512-1/sA4dwrzBAyeUoQ6oxahHKmrZvsnLCg4RfxW3ZFGGmQkSNQPFNLV9CUEFQP1x9EYXHTo5p6xdhZM1Ne9p/AfA==", + "dev": true, + "requires": { + "eslint-visitor-keys": "^3.3.0" + } + }, + "@eslint-community/regexpp": { + "version": "4.11.0", + "resolved": "https://registry.npmjs.org/@eslint-community/regexpp/-/regexpp-4.11.0.tgz", + "integrity": "sha512-G/M/tIiMrTAxEWRfLfQJMmGNX28IxBg4PBz8XqQhqUHLFI6TL2htpIB1iQCj144V5ee/JaKyT9/WZ0MGZWfA7A==", + "dev": true + }, + "@eslint/eslintrc": { + "version": "2.1.4", + "resolved": "https://registry.npmjs.org/@eslint/eslintrc/-/eslintrc-2.1.4.tgz", + "integrity": "sha512-269Z39MS6wVJtsoUl10L60WdkhJVdPG24Q4eZTH3nnF6lpvSShEK3wQjDX9JRWAUPvPh7COouPpU9IrqaZFvtQ==", + "dev": true, + "requires": { + "ajv": "^6.12.4", + "debug": "^4.3.2", + "espree": "^9.6.0", + "globals": "^13.19.0", + "ignore": "^5.2.0", + "import-fresh": "^3.2.1", + "js-yaml": "^4.1.0", + "minimatch": "^3.1.2", + "strip-json-comments": "^3.1.1" + }, + "dependencies": { + "globals": { + "version": "13.24.0", + "resolved": "https://registry.npmjs.org/globals/-/globals-13.24.0.tgz", + "integrity": "sha512-AhO5QUcj8llrbG09iWhPU2B204J1xnPeL8kQmVorSsy+Sjj1sk8gIyh6cUocGmH4L0UuhAJy+hJMRA4mgA4mFQ==", + "dev": true, + "requires": { + "type-fest": "^0.20.2" + } + } + } + }, + "@eslint/js": { + "version": "8.57.0", + "resolved": "https://registry.npmjs.org/@eslint/js/-/js-8.57.0.tgz", + "integrity": "sha512-Ys+3g2TaW7gADOJzPt83SJtCDhMjndcDMFVQ/Tj9iA1BfJzFKD9mAUXT3OenpuPHbI6P/myECxRJrofUsDx/5g==", + "dev": true + }, + "@humanwhocodes/config-array": { + "version": "0.11.14", + "resolved": "https://registry.npmjs.org/@humanwhocodes/config-array/-/config-array-0.11.14.tgz", + "integrity": "sha512-3T8LkOmg45BV5FICb15QQMsyUSWrQ8AygVfC7ZG32zOalnqrilm018ZVCw0eapXux8FtA33q8PSRSstjee3jSg==", + "dev": true, + "requires": { + "@humanwhocodes/object-schema": "^2.0.2", + "debug": "^4.3.1", + "minimatch": "^3.0.5" + } + }, + "@humanwhocodes/module-importer": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/@humanwhocodes/module-importer/-/module-importer-1.0.1.tgz", + "integrity": "sha512-bxveV4V8v5Yb4ncFTT3rPSgZBOpCkjfK0y4oVVVJwIuDVBRMDXrPyXRL988i5ap9m9bnyEEjWfm5WkBmtffLfA==", + "dev": true + }, + "@humanwhocodes/object-schema": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/@humanwhocodes/object-schema/-/object-schema-2.0.3.tgz", + "integrity": "sha512-93zYdMES/c1D69yZiKDBj0V24vqNzB/koF26KPaagAfd3P/4gUlh3Dys5ogAK+Exi9QyzlD8x/08Zt7wIKcDcA==", + "dev": true + }, + "@jridgewell/gen-mapping": { + "version": "0.3.5", + "resolved": "https://registry.npmjs.org/@jridgewell/gen-mapping/-/gen-mapping-0.3.5.tgz", + "integrity": "sha512-IzL8ZoEDIBRWEzlCcRhOaCupYyN5gdIK+Q6fbFdPDg6HqX6jpkItn7DFIpW9LQzXG6Df9sA7+OKnq0qlz/GaQg==", + "dev": true, + "requires": { + "@jridgewell/set-array": "^1.2.1", + "@jridgewell/sourcemap-codec": "^1.4.10", + "@jridgewell/trace-mapping": "^0.3.24" + } + }, + "@jridgewell/resolve-uri": { + "version": "3.1.2", + "resolved": "https://registry.npmjs.org/@jridgewell/resolve-uri/-/resolve-uri-3.1.2.tgz", + "integrity": "sha512-bRISgCIjP20/tbWSPWMEi54QVPRZExkuD9lJL+UIxUKtwVJA8wW1Trb1jMs1RFXo1CBTNZ/5hpC9QvmKWdopKw==", + "dev": true + }, + "@jridgewell/set-array": { + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/@jridgewell/set-array/-/set-array-1.2.1.tgz", + "integrity": "sha512-R8gLRTZeyp03ymzP/6Lil/28tGeGEzhx1q2k703KGWRAI1VdvPIXdG70VJc2pAMw3NA6JKL5hhFu1sJX0Mnn/A==", + "dev": true + }, + "@jridgewell/source-map": { + "version": "0.3.6", + "resolved": "https://registry.npmjs.org/@jridgewell/source-map/-/source-map-0.3.6.tgz", + "integrity": "sha512-1ZJTZebgqllO79ue2bm3rIGud/bOe0pP5BjSRCRxxYkEZS8STV7zN84UBbiYu7jy+eCKSnVIUgoWWE/tt+shMQ==", + "dev": true, + "requires": { + "@jridgewell/gen-mapping": "^0.3.5", + "@jridgewell/trace-mapping": "^0.3.25" + } + }, + "@jridgewell/sourcemap-codec": { + "version": "1.5.0", + "resolved": "https://registry.npmjs.org/@jridgewell/sourcemap-codec/-/sourcemap-codec-1.5.0.tgz", + "integrity": "sha512-gv3ZRaISU3fjPAgNsriBRqGWQL6quFx04YMPW/zD8XMLsU32mhCCbfbO6KZFLjvYpCZ8zyDEgqsgf+PwPaM7GQ==", + "dev": true + }, + "@jridgewell/trace-mapping": { + "version": "0.3.25", + "resolved": "https://registry.npmjs.org/@jridgewell/trace-mapping/-/trace-mapping-0.3.25.tgz", + "integrity": "sha512-vNk6aEwybGtawWmy/PzwnGDOjCkLWSD2wqvjGGAgOAwCGWySYXfYoxt00IJkTF+8Lb57DwOb3Aa0o9CApepiYQ==", + "dev": true, + "requires": { + "@jridgewell/resolve-uri": "^3.1.0", + "@jridgewell/sourcemap-codec": "^1.4.14" + } + }, + "@nodelib/fs.scandir": { + "version": "2.1.5", + "resolved": "https://registry.npmjs.org/@nodelib/fs.scandir/-/fs.scandir-2.1.5.tgz", + "integrity": "sha512-vq24Bq3ym5HEQm2NKCr3yXDwjc7vTsEThRDnkp2DK9p1uqLR+DHurm/NOTo0KG7HYHU7eppKZj3MyqYuMBf62g==", + "dev": true, + "requires": { + "@nodelib/fs.stat": "2.0.5", + "run-parallel": "^1.1.9" + } + }, + "@nodelib/fs.stat": { + "version": "2.0.5", + "resolved": "https://registry.npmjs.org/@nodelib/fs.stat/-/fs.stat-2.0.5.tgz", + "integrity": "sha512-RkhPPp2zrqDAQA/2jNhnztcPAlv64XdhIp7a7454A5ovI7Bukxgt7MX7udwAu3zg1DcpPU0rz3VV1SeaqvY4+A==", + "dev": true + }, + "@nodelib/fs.walk": { + "version": "1.2.8", + "resolved": "https://registry.npmjs.org/@nodelib/fs.walk/-/fs.walk-1.2.8.tgz", + "integrity": "sha512-oGB+UxlgWcgQkgwo8GcEGwemoTFt3FIO9ababBmaGwXIoBKZ+GTy0pP185beGg7Llih/NSHSV2XAs1lnznocSg==", + "dev": true, + "requires": { + "@nodelib/fs.scandir": "2.1.5", + "fastq": "^1.6.0" + } + }, + "@types/eslint": { + "version": "8.56.11", + "resolved": "https://registry.npmjs.org/@types/eslint/-/eslint-8.56.11.tgz", + "integrity": "sha512-sVBpJMf7UPo/wGecYOpk2aQya2VUGeHhe38WG7/mN5FufNSubf5VT9Uh9Uyp8/eLJpu1/tuhJ/qTo4mhSB4V4Q==", + "dev": true, + "requires": { + "@types/estree": "*", + "@types/json-schema": "*" + } + }, + "@types/estree": { + "version": "1.0.5", + "resolved": "https://registry.npmjs.org/@types/estree/-/estree-1.0.5.tgz", + "integrity": "sha512-/kYRxGDLWzHOB7q+wtSUQlFrtcdUccpfy+X+9iMBpHK8QLLhx2wIPYuS5DYtR9Wa/YlZAbIovy7qVdB1Aq6Lyw==", + "dev": true + }, + "@types/json-schema": { + "version": "7.0.15", + "resolved": "https://registry.npmjs.org/@types/json-schema/-/json-schema-7.0.15.tgz", + "integrity": "sha512-5+fP8P8MFNC+AyZCDxrB2pkZFPGzqQWUzpSeuuVLvm8VMcorNYavBqoFcxK8bQz4Qsbn4oUEEem4wDLfcysGHA==", + "dev": true + }, + "@types/node": { + "version": "22.3.0", + "resolved": "https://registry.npmjs.org/@types/node/-/node-22.3.0.tgz", + "integrity": "sha512-nrWpWVaDZuaVc5X84xJ0vNrLvomM205oQyLsRt7OHNZbSHslcWsvgFR7O7hire2ZonjLrWBbedmotmIlJDVd6g==", + "dev": true, + "requires": { + "undici-types": "~6.18.2" + } + }, + "@ungap/structured-clone": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/@ungap/structured-clone/-/structured-clone-1.2.0.tgz", + "integrity": "sha512-zuVdFrMJiuCDQUMCzQaD6KL28MjnqqN8XnAqiEq9PNm/hCPTSGfrXCOfwj1ow4LFb/tNymJPwsNbVePc1xFqrQ==", + "dev": true + }, + "@vue/compiler-sfc": { + "version": "2.7.16", + "resolved": "https://registry.npmjs.org/@vue/compiler-sfc/-/compiler-sfc-2.7.16.tgz", + "integrity": "sha512-KWhJ9k5nXuNtygPU7+t1rX6baZeqOYLEforUPjgNDBnLicfHCoi48H87Q8XyLZOrNNsmhuwKqtpDQWjEFe6Ekg==", + "dev": true, + "requires": { + "@babel/parser": "^7.23.5", + "postcss": "^8.4.14", + "prettier": "^1.18.2 || ^2.0.0", + "source-map": "^0.6.1" + } + }, + "@vue/component-compiler-utils": { + "version": "3.3.0", + "resolved": "https://registry.npmjs.org/@vue/component-compiler-utils/-/component-compiler-utils-3.3.0.tgz", + "integrity": "sha512-97sfH2mYNU+2PzGrmK2haqffDpVASuib9/w2/noxiFi31Z54hW+q3izKQXXQZSNhtiUpAI36uSuYepeBe4wpHQ==", + "dev": true, + "requires": { + "consolidate": "^0.15.1", + "hash-sum": "^1.0.2", + "lru-cache": "^4.1.2", + "merge-source-map": "^1.1.0", + "postcss": "^7.0.36", + "postcss-selector-parser": "^6.0.2", + "prettier": "^1.18.2 || ^2.0.0", + "source-map": "~0.6.1", + "vue-template-es2015-compiler": "^1.9.0" + }, + "dependencies": { + "lru-cache": { + "version": "4.1.5", + "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-4.1.5.tgz", + "integrity": "sha512-sWZlbEP2OsHNkXrMl5GYk/jKk70MBng6UU4YI/qGDYbgf6YbP4EvmqISbXCoJiRKs+1bSpFHVgQxvJ17F2li5g==", + "dev": true, + "requires": { + "pseudomap": "^1.0.2", + "yallist": "^2.1.2" + } + }, + "picocolors": { + "version": "0.2.1", + "resolved": "https://registry.npmjs.org/picocolors/-/picocolors-0.2.1.tgz", + "integrity": "sha512-cMlDqaLEqfSaW8Z7N5Jw+lyIW869EzT73/F5lhtY9cLGoVxSXznfgfXMO0Z5K0o0Q2TkTXq+0KFsdnSe3jDViA==", + "dev": true + }, + "postcss": { + "version": "7.0.39", + "resolved": "https://registry.npmjs.org/postcss/-/postcss-7.0.39.tgz", + "integrity": "sha512-yioayjNbHn6z1/Bywyb2Y4s3yvDAeXGOyxqD+LnVOinq6Mdmd++SW2wUNVzavyyHxd6+DxzWGIuosg6P1Rj8uA==", + "dev": true, + "requires": { + "picocolors": "^0.2.1", + "source-map": "^0.6.1" + } + }, + "yallist": { + "version": "2.1.2", + "resolved": "https://registry.npmjs.org/yallist/-/yallist-2.1.2.tgz", + "integrity": "sha512-ncTzHV7NvsQZkYe1DW7cbDLm0YpzHmZF5r/iyP3ZnQtMiJ+pjzisCiMNI+Sj+xQF5pXhSHxSB3uDbsBTzY/c2A==", + "dev": true + } + } + }, + "@webassemblyjs/ast": { + "version": "1.12.1", + "resolved": "https://registry.npmjs.org/@webassemblyjs/ast/-/ast-1.12.1.tgz", + "integrity": "sha512-EKfMUOPRRUTy5UII4qJDGPpqfwjOmZ5jeGFwid9mnoqIFK+e0vqoi1qH56JpmZSzEL53jKnNzScdmftJyG5xWg==", + "dev": true, + "requires": { + "@webassemblyjs/helper-numbers": "1.11.6", + "@webassemblyjs/helper-wasm-bytecode": "1.11.6" + } + }, + "@webassemblyjs/floating-point-hex-parser": { + "version": "1.11.6", + "resolved": "https://registry.npmjs.org/@webassemblyjs/floating-point-hex-parser/-/floating-point-hex-parser-1.11.6.tgz", + "integrity": "sha512-ejAj9hfRJ2XMsNHk/v6Fu2dGS+i4UaXBXGemOfQ/JfQ6mdQg/WXtwleQRLLS4OvfDhv8rYnVwH27YJLMyYsxhw==", + "dev": true + }, + "@webassemblyjs/helper-api-error": { + "version": "1.11.6", + "resolved": "https://registry.npmjs.org/@webassemblyjs/helper-api-error/-/helper-api-error-1.11.6.tgz", + "integrity": "sha512-o0YkoP4pVu4rN8aTJgAyj9hC2Sv5UlkzCHhxqWj8butaLvnpdc2jOwh4ewE6CX0txSfLn/UYaV/pheS2Txg//Q==", + "dev": true + }, + "@webassemblyjs/helper-buffer": { + "version": "1.12.1", + "resolved": "https://registry.npmjs.org/@webassemblyjs/helper-buffer/-/helper-buffer-1.12.1.tgz", + "integrity": "sha512-nzJwQw99DNDKr9BVCOZcLuJJUlqkJh+kVzVl6Fmq/tI5ZtEyWT1KZMyOXltXLZJmDtvLCDgwsyrkohEtopTXCw==", + "dev": true + }, + "@webassemblyjs/helper-numbers": { + "version": "1.11.6", + "resolved": "https://registry.npmjs.org/@webassemblyjs/helper-numbers/-/helper-numbers-1.11.6.tgz", + "integrity": "sha512-vUIhZ8LZoIWHBohiEObxVm6hwP034jwmc9kuq5GdHZH0wiLVLIPcMCdpJzG4C11cHoQ25TFIQj9kaVADVX7N3g==", + "dev": true, + "requires": { + "@webassemblyjs/floating-point-hex-parser": "1.11.6", + "@webassemblyjs/helper-api-error": "1.11.6", + "@xtuc/long": "4.2.2" + } + }, + "@webassemblyjs/helper-wasm-bytecode": { + "version": "1.11.6", + "resolved": "https://registry.npmjs.org/@webassemblyjs/helper-wasm-bytecode/-/helper-wasm-bytecode-1.11.6.tgz", + "integrity": "sha512-sFFHKwcmBprO9e7Icf0+gddyWYDViL8bpPjJJl0WHxCdETktXdmtWLGVzoHbqUcY4Be1LkNfwTmXOJUFZYSJdA==", + "dev": true + }, + "@webassemblyjs/helper-wasm-section": { + "version": "1.12.1", + "resolved": "https://registry.npmjs.org/@webassemblyjs/helper-wasm-section/-/helper-wasm-section-1.12.1.tgz", + "integrity": "sha512-Jif4vfB6FJlUlSbgEMHUyk1j234GTNG9dBJ4XJdOySoj518Xj0oGsNi59cUQF4RRMS9ouBUxDDdyBVfPTypa5g==", + "dev": true, + "requires": { + "@webassemblyjs/ast": "1.12.1", + "@webassemblyjs/helper-buffer": "1.12.1", + "@webassemblyjs/helper-wasm-bytecode": "1.11.6", + "@webassemblyjs/wasm-gen": "1.12.1" + } + }, + "@webassemblyjs/ieee754": { + "version": "1.11.6", + "resolved": "https://registry.npmjs.org/@webassemblyjs/ieee754/-/ieee754-1.11.6.tgz", + "integrity": "sha512-LM4p2csPNvbij6U1f19v6WR56QZ8JcHg3QIJTlSwzFcmx6WSORicYj6I63f9yU1kEUtrpG+kjkiIAkevHpDXrg==", + "dev": true, + "requires": { + "@xtuc/ieee754": "^1.2.0" + } + }, + "@webassemblyjs/leb128": { + "version": "1.11.6", + "resolved": "https://registry.npmjs.org/@webassemblyjs/leb128/-/leb128-1.11.6.tgz", + "integrity": "sha512-m7a0FhE67DQXgouf1tbN5XQcdWoNgaAuoULHIfGFIEVKA6tu/edls6XnIlkmS6FrXAquJRPni3ZZKjw6FSPjPQ==", + "dev": true, + "requires": { + "@xtuc/long": "4.2.2" + } + }, + "@webassemblyjs/utf8": { + "version": "1.11.6", + "resolved": "https://registry.npmjs.org/@webassemblyjs/utf8/-/utf8-1.11.6.tgz", + "integrity": "sha512-vtXf2wTQ3+up9Zsg8sa2yWiQpzSsMyXj0qViVP6xKGCUT8p8YJ6HqI7l5eCnWx1T/FYdsv07HQs2wTFbbof/RA==", + "dev": true + }, + "@webassemblyjs/wasm-edit": { + "version": "1.12.1", + "resolved": "https://registry.npmjs.org/@webassemblyjs/wasm-edit/-/wasm-edit-1.12.1.tgz", + "integrity": "sha512-1DuwbVvADvS5mGnXbE+c9NfA8QRcZ6iKquqjjmR10k6o+zzsRVesil54DKexiowcFCPdr/Q0qaMgB01+SQ1u6g==", + "dev": true, + "requires": { + "@webassemblyjs/ast": "1.12.1", + "@webassemblyjs/helper-buffer": "1.12.1", + "@webassemblyjs/helper-wasm-bytecode": "1.11.6", + "@webassemblyjs/helper-wasm-section": "1.12.1", + "@webassemblyjs/wasm-gen": "1.12.1", + "@webassemblyjs/wasm-opt": "1.12.1", + "@webassemblyjs/wasm-parser": "1.12.1", + "@webassemblyjs/wast-printer": "1.12.1" + } + }, + "@webassemblyjs/wasm-gen": { + "version": "1.12.1", + "resolved": "https://registry.npmjs.org/@webassemblyjs/wasm-gen/-/wasm-gen-1.12.1.tgz", + "integrity": "sha512-TDq4Ojh9fcohAw6OIMXqiIcTq5KUXTGRkVxbSo1hQnSy6lAM5GSdfwWeSxpAo0YzgsgF182E/U0mDNhuA0tW7w==", + "dev": true, + "requires": { + "@webassemblyjs/ast": "1.12.1", + "@webassemblyjs/helper-wasm-bytecode": "1.11.6", + "@webassemblyjs/ieee754": "1.11.6", + "@webassemblyjs/leb128": "1.11.6", + "@webassemblyjs/utf8": "1.11.6" + } + }, + "@webassemblyjs/wasm-opt": { + "version": "1.12.1", + "resolved": "https://registry.npmjs.org/@webassemblyjs/wasm-opt/-/wasm-opt-1.12.1.tgz", + "integrity": "sha512-Jg99j/2gG2iaz3hijw857AVYekZe2SAskcqlWIZXjji5WStnOpVoat3gQfT/Q5tb2djnCjBtMocY/Su1GfxPBg==", + "dev": true, + "requires": { + "@webassemblyjs/ast": "1.12.1", + "@webassemblyjs/helper-buffer": "1.12.1", + "@webassemblyjs/wasm-gen": "1.12.1", + "@webassemblyjs/wasm-parser": "1.12.1" + } + }, + "@webassemblyjs/wasm-parser": { + "version": "1.12.1", + "resolved": "https://registry.npmjs.org/@webassemblyjs/wasm-parser/-/wasm-parser-1.12.1.tgz", + "integrity": "sha512-xikIi7c2FHXysxXe3COrVUPSheuBtpcfhbpFj4gmu7KRLYOzANztwUU0IbsqvMqzuNK2+glRGWCEqZo1WCLyAQ==", + "dev": true, + "requires": { + "@webassemblyjs/ast": "1.12.1", + "@webassemblyjs/helper-api-error": "1.11.6", + "@webassemblyjs/helper-wasm-bytecode": "1.11.6", + "@webassemblyjs/ieee754": "1.11.6", + "@webassemblyjs/leb128": "1.11.6", + "@webassemblyjs/utf8": "1.11.6" + } + }, + "@webassemblyjs/wast-printer": { + "version": "1.12.1", + "resolved": "https://registry.npmjs.org/@webassemblyjs/wast-printer/-/wast-printer-1.12.1.tgz", + "integrity": "sha512-+X4WAlOisVWQMikjbcvY2e0rwPsKQ9F688lksZhBcPycBBuii3O7m8FACbDMWDojpAqvjIncrG8J0XHKyQfVeA==", + "dev": true, + "requires": { + "@webassemblyjs/ast": "1.12.1", + "@xtuc/long": "4.2.2" + } + }, + "@webpack-cli/configtest": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/@webpack-cli/configtest/-/configtest-1.2.0.tgz", + "integrity": "sha512-4FB8Tj6xyVkyqjj1OaTqCjXYULB9FMkqQ8yGrZjRDrYh0nOE+7Lhs45WioWQQMV+ceFlE368Ukhe6xdvJM9Egg==", + "dev": true, + "requires": {} + }, + "@webpack-cli/info": { + "version": "1.5.0", + "resolved": "https://registry.npmjs.org/@webpack-cli/info/-/info-1.5.0.tgz", + "integrity": "sha512-e8tSXZpw2hPl2uMJY6fsMswaok5FdlGNRTktvFk2sD8RjH0hE2+XistawJx1vmKteh4NmGmNUrp+Tb2w+udPcQ==", + "dev": true, + "requires": { + "envinfo": "^7.7.3" + } + }, + "@webpack-cli/serve": { + "version": "1.7.0", + "resolved": "https://registry.npmjs.org/@webpack-cli/serve/-/serve-1.7.0.tgz", + "integrity": "sha512-oxnCNGj88fL+xzV+dacXs44HcDwf1ovs3AuEzvP7mqXw7fQntqIhQ1BRmynh4qEKQSSSRSWVyXRjmTbZIX9V2Q==", + "dev": true, + "requires": {} + }, + "@xtuc/ieee754": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/@xtuc/ieee754/-/ieee754-1.2.0.tgz", + "integrity": "sha512-DX8nKgqcGwsc0eJSqYt5lwP4DH5FlHnmuWWBRy7X0NcaGR0ZtuyeESgMwTYVEtxmsNGY+qit4QYT/MIYTOTPeA==", + "dev": true + }, + "@xtuc/long": { + "version": "4.2.2", + "resolved": "https://registry.npmjs.org/@xtuc/long/-/long-4.2.2.tgz", + "integrity": "sha512-NuHqBY1PB/D8xU6s/thBgOAiAP7HOYDQ32+BFZILJ8ivkUkAHQnWfn6WhL79Owj1qmUnoN/YPhktdIoucipkAQ==", + "dev": true + }, + "acorn": { + "version": "8.12.1", + "resolved": "https://registry.npmjs.org/acorn/-/acorn-8.12.1.tgz", + "integrity": "sha512-tcpGyI9zbizT9JbV6oYE477V6mTlXvvi0T0G3SNIYE2apm/G5huBa1+K89VGeovbg+jycCrfhl3ADxErOuO6Jg==", + "dev": true + }, + "acorn-import-attributes": { + "version": "1.9.5", + "resolved": "https://registry.npmjs.org/acorn-import-attributes/-/acorn-import-attributes-1.9.5.tgz", + "integrity": "sha512-n02Vykv5uA3eHGM/Z2dQrcD56kL8TyDb2p1+0P83PClMnC/nc+anbQRhIOWnSq4Ke/KvDPrY3C9hDtC/A3eHnQ==", + "dev": true, + "requires": {} + }, + "acorn-jsx": { + "version": "5.3.2", + "resolved": "https://registry.npmjs.org/acorn-jsx/-/acorn-jsx-5.3.2.tgz", + "integrity": "sha512-rq9s+JNhf0IChjtDXxllJ7g41oZk5SlXtp0LHwyA5cejwn7vKmKp4pPri6YEePv2PU65sAsegbXtIinmDFDXgQ==", + "dev": true, + "requires": {} + }, + "ajv": { + "version": "6.12.6", + "resolved": "https://registry.npmjs.org/ajv/-/ajv-6.12.6.tgz", + "integrity": "sha512-j3fVLgvTo527anyYyJOGTYJbG+vnnQYvE0m5mmkc1TK+nxAppkCLMIL0aZ4dblVCNoGShhm+kzE4ZUykBoMg4g==", + "dev": true, + "requires": { + "fast-deep-equal": "^3.1.1", + "fast-json-stable-stringify": "^2.0.0", + "json-schema-traverse": "^0.4.1", + "uri-js": "^4.2.2" + } + }, + "ajv-formats": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/ajv-formats/-/ajv-formats-2.1.1.tgz", + "integrity": "sha512-Wx0Kx52hxE7C18hkMEggYlEifqWZtYaRgouJor+WMdPnQyEK13vgEWyVNup7SoeeoLMsr4kf5h6dOW11I15MUA==", + "dev": true, + "requires": {}, + "dependencies": { + "ajv": { + "version": "https://registry.npmjs.org/ajv/-/ajv-8.17.1.tgz", + "integrity": "sha512-B/gBuNg5SiMTrPkC+A2+cW0RszwxYmn6VYxB/inlBStS5nx6xHIt/ehKRhIMhqusl7a8LjQoZnjCs5vhwxOQ1g==", + "dev": true, + "optional": true, + "peer": true, + "requires": { + "fast-deep-equal": "^3.1.3", + "fast-uri": "^3.0.1", + "json-schema-traverse": "^1.0.0", + "require-from-string": "^2.0.2" + } + }, + "json-schema-traverse": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/json-schema-traverse/-/json-schema-traverse-1.0.0.tgz", + "integrity": "sha512-NM8/P9n3XjXhIZn1lLhkFaACTOURQXjWhV4BA/RnOv8xvgqtqpAX9IO4mRQxSx1Rlo4tqzeqb0sOlruaOy3dug==", + "dev": true, + "optional": true, + "peer": true + } + } + }, + "ajv-keywords": { + "version": "3.5.2", + "resolved": "https://registry.npmjs.org/ajv-keywords/-/ajv-keywords-3.5.2.tgz", + "integrity": "sha512-5p6WTN0DdTGVQk6VjcEju19IgaHudalcfabD7yhDGeA6bcQnmL+CpveLJq/3hvfwd1aof6L386Ougkx6RfyMIQ==", + "dev": true, + "requires": {} + }, + "ansi-regex": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.1.tgz", + "integrity": "sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ==", + "dev": true + }, + "ansi-styles": { + "version": "3.2.1", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-3.2.1.tgz", + "integrity": "sha512-VT0ZI6kZRdTh8YyJw3SMbYm/u+NqfsAxEpWO0Pf9sq8/e94WxxOpPKx9FR1FlyCtOVDNOQ+8ntlqFxiRc+r5qA==", + "dev": true, + "peer": true, + "requires": { + "color-convert": "^1.9.0" + } + }, + "anymatch": { + "version": "3.1.3", + "resolved": "https://registry.npmjs.org/anymatch/-/anymatch-3.1.3.tgz", + "integrity": "sha512-KMReFUr0B4t+D+OBkjR3KYqvocp2XaSzO55UcB6mgQMd3KbcE+mWTyvVV7D/zsdEbNnV6acZUutkiHQXvTr1Rw==", + "dev": true, + "requires": { + "normalize-path": "^3.0.0", + "picomatch": "^2.0.4" + } + }, + "argparse": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/argparse/-/argparse-2.0.1.tgz", + "integrity": "sha512-8+9WqebbFzpX9OR+Wa6O29asIogeRMzcGtAINdpMHHyAg10f05aSFVBbcEqGf/PXw1EjAZ+q2/bEBg3DvurK3Q==", + "dev": true + }, + "aria-query": { + "version": "5.3.0", + "resolved": "https://registry.npmjs.org/aria-query/-/aria-query-5.3.0.tgz", + "integrity": "sha512-b0P0sZPKtyu8HkeRAfCq0IfURZK+SuwMjY1UXGBU27wpAiTwQAIlq56IbIO+ytk/JjS1fMR14ee5WBBfKi5J6A==", + "dev": true, + "requires": { + "dequal": "^2.0.3" + } + }, + "babel-loader": { + "version": "8.3.0", + "resolved": "https://registry.npmjs.org/babel-loader/-/babel-loader-8.3.0.tgz", + "integrity": "sha512-H8SvsMF+m9t15HNLMipppzkC+Y2Yq+v3SonZyU70RBL/h1gxPkH08Ot8pEE9Z4Kd+czyWJClmFS8qzIP9OZ04Q==", + "dev": true, + "requires": { + "find-cache-dir": "^3.3.1", + "loader-utils": "^2.0.0", + "make-dir": "^3.1.0", + "schema-utils": "^2.6.5" + } + }, + "balanced-match": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.2.tgz", + "integrity": "sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw==", + "dev": true + }, + "big.js": { + "version": "5.2.2", + "resolved": "https://registry.npmjs.org/big.js/-/big.js-5.2.2.tgz", + "integrity": "sha512-vyL2OymJxmarO8gxMr0mhChsO9QGwhynfuu4+MHTAW6czfq9humCB7rKpUjDd9YUiDPU4mzpyupFSvOClAwbmQ==", + "dev": true + }, + "binary-extensions": { + "version": "2.3.0", + "resolved": "https://registry.npmjs.org/binary-extensions/-/binary-extensions-2.3.0.tgz", + "integrity": "sha512-Ceh+7ox5qe7LJuLHoY0feh3pHuUDHAcRUeyL2VYghZwfpkNIy/+8Ocg0a3UuSoYzavmylwuLWQOf3hl0jjMMIw==", + "dev": true + }, + "bluebird": { + "version": "3.7.2", + "resolved": "https://registry.npmjs.org/bluebird/-/bluebird-3.7.2.tgz", + "integrity": "sha512-XpNj6GDQzdfW+r2Wnn7xiSAd7TM3jzkxGXBGTtWKuSXv1xUV+azxAm8jdWZN06QTQk+2N2XB9jRDkvbmQmcRtg==", + "dev": true + }, + "boolbase": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/boolbase/-/boolbase-1.0.0.tgz", + "integrity": "sha512-JZOSA7Mo9sNGB8+UjSgzdLtokWAky1zbztM3WRLCbZ70/3cTANmQmOdR7y2g+J0e2WXywy1yS468tY+IruqEww==", + "dev": true + }, + "brace-expansion": { + "version": "1.1.11", + "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.11.tgz", + "integrity": "sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA==", + "dev": true, + "requires": { + "balanced-match": "^1.0.0", + "concat-map": "0.0.1" + } + }, + "braces": { + "version": "3.0.3", + "resolved": "https://registry.npmjs.org/braces/-/braces-3.0.3.tgz", + "integrity": "sha512-yQbXgO/OSZVD2IsiLlro+7Hf6Q18EJrKSEsdoMzKePKXct3gvD8oLcOQdIzGupr5Fj+EDe8gO/lxc1BzfMpxvA==", + "dev": true, + "requires": { + "fill-range": "^7.1.1" + } + }, + "browserslist": { + "version": "4.23.3", + "resolved": "https://registry.npmjs.org/browserslist/-/browserslist-4.23.3.tgz", + "integrity": "sha512-btwCFJVjI4YWDNfau8RhZ+B1Q/VLoUITrm3RlP6y1tYGWIOa+InuYiRGXUBXo8nA1qKmHMyLB/iVQg5TT4eFoA==", + "dev": true, + "requires": { + "caniuse-lite": "^1.0.30001646", + "electron-to-chromium": "^1.5.4", + "node-releases": "^2.0.18", + "update-browserslist-db": "^1.1.0" + } + }, + "buffer-from": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/buffer-from/-/buffer-from-1.1.2.tgz", + "integrity": "sha512-E+XQCRwSbaaiChtv6k6Dwgc+bx+Bs6vuKJHHl5kox/BaKbhiXzqQOwK4cO22yElGp2OCmjwVhT3HmxgyPGnJfQ==", + "dev": true + }, + "callsites": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/callsites/-/callsites-3.1.0.tgz", + "integrity": "sha512-P8BjAsXvZS+VIDUI11hHCQEv74YT67YUi5JJFNWIqL235sBmjX4+qx9Muvls5ivyNENctx46xQLQ3aTuE7ssaQ==", + "dev": true + }, + "caniuse-lite": { + "version": "1.0.30001651", + "resolved": "https://registry.npmjs.org/caniuse-lite/-/caniuse-lite-1.0.30001651.tgz", + "integrity": "sha512-9Cf+Xv1jJNe1xPZLGuUXLNkE1BoDkqRqYyFJ9TDYSqhduqA4hu4oR9HluGoWYQC/aj8WHjsGVV+bwkh0+tegRg==", + "dev": true + }, + "chalk": { + "version": "2.4.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-2.4.2.tgz", + "integrity": "sha512-Mti+f9lpJNcwF4tWV8/OrTTtF1gZi+f8FqlyAdouralcFWFQWF2+NgCHShjkCb+IFBLq9buZwE1xckQU4peSuQ==", + "dev": true, + "peer": true, + "requires": { + "ansi-styles": "^3.2.1", + "escape-string-regexp": "^1.0.5", + "supports-color": "^5.3.0" + } + }, + "chokidar": { + "version": "3.6.0", + "resolved": "https://registry.npmjs.org/chokidar/-/chokidar-3.6.0.tgz", + "integrity": "sha512-7VT13fmjotKpGipCW9JEQAusEPE+Ei8nl6/g4FBAmIm0GOOLMua9NDDo/DWp0ZAxCr3cPq5ZpBqmPAQgDda2Pw==", + "dev": true, + "requires": { + "anymatch": "~3.1.2", + "braces": "~3.0.2", + "fsevents": "~2.3.2", + "glob-parent": "~5.1.2", + "is-binary-path": "~2.1.0", + "is-glob": "~4.0.1", + "normalize-path": "~3.0.0", + "readdirp": "~3.6.0" + }, + "dependencies": { + "glob-parent": { + "version": "5.1.2", + "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-5.1.2.tgz", + "integrity": "sha512-AOIgSQCepiJYwP3ARnGx+5VnTu2HBYdzbGP45eLw1vr3zB3vZLeyed1sC9hnbcOc9/SrMyM5RPQrkGz4aS9Zow==", + "dev": true, + "requires": { + "is-glob": "^4.0.1" + } + } + } + }, + "chrome-trace-event": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/chrome-trace-event/-/chrome-trace-event-1.0.4.tgz", + "integrity": "sha512-rNjApaLzuwaOTjCiT8lSDdGN1APCiqkChLMJxJPWLunPAt5fy8xgU9/jNOchV84wfIxrA0lRQB7oCT8jrn/wrQ==", + "dev": true + }, + "clone-deep": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/clone-deep/-/clone-deep-4.0.1.tgz", + "integrity": "sha512-neHB9xuzh/wk0dIHweyAXv2aPGZIVk3pLMe+/RNzINf17fe0OG96QroktYAUm7SM1PBnzTabaLboqqxDyMU+SQ==", + "dev": true, + "requires": { + "is-plain-object": "^2.0.4", + "kind-of": "^6.0.2", + "shallow-clone": "^3.0.0" + } + }, + "color-convert": { + "version": "1.9.3", + "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-1.9.3.tgz", + "integrity": "sha512-QfAUtd+vFdAtFQcC8CCyYt1fYWxSqAiK2cSD6zDB8N3cpsEBAvRxp9zOGg6G/SHHJYAT88/az/IuDGALsNVbGg==", + "dev": true, + "peer": true, + "requires": { + "color-name": "1.1.3" + } + }, + "color-name": { + "version": "1.1.3", + "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.3.tgz", + "integrity": "sha512-72fSenhMw2HZMTVHeCA9KCmpEIbzWiQsjN+BHcBbS9vr1mtt+vJjPdksIBNUmKAW8TFUDPJK5SUU3QhE9NEXDw==", + "dev": true, + "peer": true + }, + "colorette": { + "version": "2.0.20", + "resolved": "https://registry.npmjs.org/colorette/-/colorette-2.0.20.tgz", + "integrity": "sha512-IfEDxwoWIjkeXL1eXcDiow4UbKjhLdq6/EuSVR9GMN7KVH3r9gQ83e73hsz1Nd1T3ijd5xv1wcWRYO+D6kCI2w==", + "dev": true + }, + "commander": { + "version": "2.20.3", + "resolved": "https://registry.npmjs.org/commander/-/commander-2.20.3.tgz", + "integrity": "sha512-GpVkmM8vF2vQUkj2LvZmD35JxeJOLCwJ9cUkugyk2nuhbv3+mJvpLYYt+0+USMxE+oj+ey/lJEnhZw75x/OMcQ==", + "dev": true + }, + "commondir": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/commondir/-/commondir-1.0.1.tgz", + "integrity": "sha512-W9pAhw0ja1Edb5GVdIF1mjZw/ASI0AlShXM83UUGe2DVr5TdAPEA1OA8m/g8zWp9x6On7gqufY+FatDbC3MDQg==", + "dev": true + }, + "concat-map": { + "version": "0.0.1", + "resolved": "https://registry.npmjs.org/concat-map/-/concat-map-0.0.1.tgz", + "integrity": "sha512-/Srv4dswyQNBfohGpz9o6Yb3Gz3SrUDqBH5rTuhGR7ahtlbYKnVxw2bCFMRljaA7EXHaXZ8wsHdodFvbkhKmqg==", + "dev": true + }, + "consolidate": { + "version": "0.15.1", + "resolved": "https://registry.npmjs.org/consolidate/-/consolidate-0.15.1.tgz", + "integrity": "sha512-DW46nrsMJgy9kqAbPt5rKaCr7uFtpo4mSUvLHIUbJEjm0vo+aY5QLwBUq3FK4tRnJr/X0Psc0C4jf/h+HtXSMw==", + "dev": true, + "requires": { + "bluebird": "^3.1.1" + } + }, + "convert-source-map": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/convert-source-map/-/convert-source-map-2.0.0.tgz", + "integrity": "sha512-Kvp459HrV2FEJ1CAsi1Ku+MY3kasH19TFykTz2xWmMeq6bk2NU3XXvfJ+Q61m0xktWwt+1HSYf3JZsTms3aRJg==", + "dev": true, + "peer": true + }, + "cross-spawn": { + "version": "7.0.3", + "resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-7.0.3.tgz", + "integrity": "sha512-iRDPJKUPVEND7dHPO8rkbOnPpyDygcDFtWjpeWNCgy8WP2rXcxXL8TskReQl6OrB2G7+UJrags1q15Fudc7G6w==", + "dev": true, + "requires": { + "path-key": "^3.1.0", + "shebang-command": "^2.0.0", + "which": "^2.0.1" + } + }, + "css-loader": { + "version": "7.1.2", + "resolved": "https://registry.npmjs.org/css-loader/-/css-loader-7.1.2.tgz", + "integrity": "sha512-6WvYYn7l/XEGN8Xu2vWFt9nVzrCn39vKyTEFf/ExEyoksJjjSZV/0/35XPlMbpnr6VGhZIUg5yJrL8tGfes/FA==", + "dev": true, + "peer": true, + "requires": { + "icss-utils": "^5.1.0", + "postcss": "^8.4.33", + "postcss-modules-extract-imports": "^3.1.0", + "postcss-modules-local-by-default": "^4.0.5", + "postcss-modules-scope": "^3.2.0", + "postcss-modules-values": "^4.0.0", + "postcss-value-parser": "^4.2.0", + "semver": "^7.5.4" + }, + "dependencies": { + "semver": { + "version": "7.6.3", + "resolved": "https://registry.npmjs.org/semver/-/semver-7.6.3.tgz", + "integrity": "sha512-oVekP1cKtI+CTDvHWYFUcMtsK/00wmAEfyqKfNdARm8u1wNVhSgaX7A8d4UuIlUI5e84iEwOhs7ZPYRmzU9U6A==", + "dev": true, + "peer": true + } + } + }, + "cssesc": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/cssesc/-/cssesc-3.0.0.tgz", + "integrity": "sha512-/Tb/JcjK111nNScGob5MNtsntNM1aCNUDipB/TkwZFhyDrrE47SOx/18wF2bbjgc3ZzCSKW1T5nt5EbFoAz/Vg==", + "dev": true + }, + "csstype": { + "version": "3.1.3", + "resolved": "https://registry.npmjs.org/csstype/-/csstype-3.1.3.tgz", + "integrity": "sha512-M1uQkMl8rQK/szD0LNhtqxIPLpimGm8sOBwU7lLnCpSbTyY3yeU1Vc7l4KT5zT4s/yOxHH5O7tIuuLOCnLADRw==", + "dev": true + }, + "de-indent": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/de-indent/-/de-indent-1.0.2.tgz", + "integrity": "sha512-e/1zu3xH5MQryN2zdVaF0OrdNLUbvWxzMbi+iNA6Bky7l1RoP8a2fIbRocyHclXt/arDrrR6lL3TqFD9pMQTsg==", + "dev": true + }, + "debug": { + "version": "4.3.6", + "resolved": "https://registry.npmjs.org/debug/-/debug-4.3.6.tgz", + "integrity": "sha512-O/09Bd4Z1fBrU4VzkhFqVgpPzaGbw6Sm9FEkBT1A/YBXQFGuuSxa1dN2nxgxS34JmKXqYx8CZAwEVoJFImUXIg==", + "dev": true, + "requires": { + "ms": "2.1.2" + } + }, + "deep-is": { + "version": "0.1.4", + "resolved": "https://registry.npmjs.org/deep-is/-/deep-is-0.1.4.tgz", + "integrity": "sha512-oIPzksmTg4/MriiaYGO+okXDT7ztn/w3Eptv/+gSIdMdKsJo0u4CfYNFJPy+4SKMuCqGw2wxnA+URMg3t8a/bQ==", + "dev": true + }, + "dequal": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/dequal/-/dequal-2.0.3.tgz", + "integrity": "sha512-0je+qPKHEMohvfRTCEo3CrPG6cAzAYgmzKyxRiYSSDkS6eGJdyVJm7WaYA5ECaAD9wLB2T4EEeymA5aFVcYXCA==", + "dev": true + }, + "doctrine": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/doctrine/-/doctrine-3.0.0.tgz", + "integrity": "sha512-yS+Q5i3hBf7GBkd4KG8a7eBNNWNGLTaEwwYWUijIYM7zrlYDM0BFXHjjPWlWZ1Rg7UaddZeIDmi9jF3HmqiQ2w==", + "dev": true, + "requires": { + "esutils": "^2.0.2" + } + }, + "electron-to-chromium": { + "version": "1.5.9", + "resolved": "https://registry.npmjs.org/electron-to-chromium/-/electron-to-chromium-1.5.9.tgz", + "integrity": "sha512-HfkT8ndXR0SEkU8gBQQM3rz035bpE/hxkZ1YIt4KJPEFES68HfIU6LzKukH0H794Lm83WJtkSAMfEToxCs15VA==", + "dev": true + }, + "emoji-regex": { + "version": "10.3.0", + "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-10.3.0.tgz", + "integrity": "sha512-QpLs9D9v9kArv4lfDEgg1X/gN5XLnf/A6l9cs8SPZLRZR3ZkY9+kwIQTxm+fsSej5UMYGE8fdoaZVIBlqG0XTw==", + "dev": true + }, + "emojis-list": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/emojis-list/-/emojis-list-3.0.0.tgz", + "integrity": "sha512-/kyM18EfinwXZbno9FyUGeFh87KC8HRQBQGildHZbEuRyWFOmv1U10o9BBp8XVZDVNNuQKyIGIu5ZYAAXJ0V2Q==", + "dev": true + }, + "enhanced-resolve": { + "version": "5.17.1", + "resolved": "https://registry.npmjs.org/enhanced-resolve/-/enhanced-resolve-5.17.1.tgz", + "integrity": "sha512-LMHl3dXhTcfv8gM4kEzIUeTQ+7fpdA0l2tUf34BddXPkz2A5xJ5L/Pchd5BL6rdccM9QGvu0sWZzK1Z1t4wwyg==", + "dev": true, + "requires": { + "graceful-fs": "^4.2.4", + "tapable": "^2.2.0" + } + }, + "envinfo": { + "version": "7.13.0", + "resolved": "https://registry.npmjs.org/envinfo/-/envinfo-7.13.0.tgz", + "integrity": "sha512-cvcaMr7KqXVh4nyzGTVqTum+gAiL265x5jUWQIDLq//zOGbW+gSW/C+OWLleY/rs9Qole6AZLMXPbtIFQbqu+Q==", + "dev": true + }, + "es-module-lexer": { + "version": "1.5.4", + "resolved": "https://registry.npmjs.org/es-module-lexer/-/es-module-lexer-1.5.4.tgz", + "integrity": "sha512-MVNK56NiMrOwitFB7cqDwq0CQutbw+0BvLshJSse0MUNU+y1FC3bUS/AQg7oUng+/wKrrki7JfmwtVHkVfPLlw==", + "dev": true + }, + "escalade": { + "version": "3.1.2", + "resolved": "https://registry.npmjs.org/escalade/-/escalade-3.1.2.tgz", + "integrity": "sha512-ErCHMCae19vR8vQGe50xIsVomy19rg6gFu3+r3jkEO46suLMWBksvVyoGgQV+jOfl84ZSOSlmv6Gxa89PmTGmA==", + "dev": true + }, + "escape-string-regexp": { + "version": "1.0.5", + "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-1.0.5.tgz", + "integrity": "sha512-vbRorB5FUQWvla16U8R/qgaFIya2qGzwDrNmCZuYKrbdSUMG6I1ZCGQRefkRVhuOkIGVne7BQ35DSfo1qvJqFg==", + "dev": true, + "peer": true + }, + "eslint": { + "version": "8.57.0", + "resolved": "https://registry.npmjs.org/eslint/-/eslint-8.57.0.tgz", + "integrity": "sha512-dZ6+mexnaTIbSBZWgou51U6OmzIhYM2VcNdtiTtI7qPNZm35Akpr0f6vtw3w1Kmn5PYo+tZVfh13WrhpS6oLqQ==", + "dev": true, + "requires": { + "@eslint-community/eslint-utils": "^4.2.0", + "@eslint-community/regexpp": "^4.6.1", + "@eslint/eslintrc": "^2.1.4", + "@eslint/js": "8.57.0", + "@humanwhocodes/config-array": "^0.11.14", + "@humanwhocodes/module-importer": "^1.0.1", + "@nodelib/fs.walk": "^1.2.8", + "@ungap/structured-clone": "^1.2.0", + "ajv": "^6.12.4", + "chalk": "^4.0.0", + "cross-spawn": "^7.0.2", + "debug": "^4.3.2", + "doctrine": "^3.0.0", + "escape-string-regexp": "^4.0.0", + "eslint-scope": "^7.2.2", + "eslint-visitor-keys": "^3.4.3", + "espree": "^9.6.1", + "esquery": "^1.4.2", + "esutils": "^2.0.2", + "fast-deep-equal": "^3.1.3", + "file-entry-cache": "^6.0.1", + "find-up": "^5.0.0", + "glob-parent": "^6.0.2", + "globals": "^13.19.0", + "graphemer": "^1.4.0", + "ignore": "^5.2.0", + "imurmurhash": "^0.1.4", + "is-glob": "^4.0.0", + "is-path-inside": "^3.0.3", + "js-yaml": "^4.1.0", + "json-stable-stringify-without-jsonify": "^1.0.1", + "levn": "^0.4.1", + "lodash.merge": "^4.6.2", + "minimatch": "^3.1.2", + "natural-compare": "^1.4.0", + "optionator": "^0.9.3", + "strip-ansi": "^6.0.1", + "text-table": "^0.2.0" + }, + "dependencies": { + "ansi-styles": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", + "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", + "dev": true, + "requires": { + "color-convert": "^2.0.1" + } + }, + "chalk": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "dev": true, + "requires": { + "ansi-styles": "^4.1.0", + "supports-color": "^7.1.0" + } + }, + "color-convert": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", + "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", + "dev": true, + "requires": { + "color-name": "~1.1.4" + } + }, + "color-name": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", + "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", + "dev": true + }, + "escape-string-regexp": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-4.0.0.tgz", + "integrity": "sha512-TtpcNJ3XAzx3Gq8sWRzJaVajRs0uVxA2YAkdb1jm2YkPz4G6egUFAyA3n5vtEIZefPk5Wa4UXbKuS5fKkJWdgA==", + "dev": true + }, + "globals": { + "version": "13.24.0", + "resolved": "https://registry.npmjs.org/globals/-/globals-13.24.0.tgz", + "integrity": "sha512-AhO5QUcj8llrbG09iWhPU2B204J1xnPeL8kQmVorSsy+Sjj1sk8gIyh6cUocGmH4L0UuhAJy+hJMRA4mgA4mFQ==", + "dev": true, + "requires": { + "type-fest": "^0.20.2" + } + }, + "has-flag": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", + "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", + "dev": true + }, + "supports-color": { + "version": "7.2.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", + "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", + "dev": true, + "requires": { + "has-flag": "^4.0.0" + } + } + } + }, + "eslint-config-meedsio": { + "version": "1.0.5", + "resolved": "https://registry.npmjs.org/eslint-config-meedsio/-/eslint-config-meedsio-1.0.5.tgz", + "integrity": "sha512-eSae9Ald7T7g4gIdomPGJ7/H+aVd9L+wLvDUICvc8vKfLeKoJSdOH1uoJ1beUytObQbzbJJJC+cHvf+8BxYDdg==", + "dev": true + }, + "eslint-plugin-vue": { + "version": "8.7.1", + "resolved": "https://registry.npmjs.org/eslint-plugin-vue/-/eslint-plugin-vue-8.7.1.tgz", + "integrity": "sha512-28sbtm4l4cOzoO1LtzQPxfxhQABararUb1JtqusQqObJpWX2e/gmVyeYVfepizPFne0Q5cILkYGiBoV36L12Wg==", + "dev": true, + "requires": { + "eslint-utils": "^3.0.0", + "natural-compare": "^1.4.0", + "nth-check": "^2.0.1", + "postcss-selector-parser": "^6.0.9", + "semver": "^7.3.5", + "vue-eslint-parser": "^8.0.1" + }, + "dependencies": { + "semver": { + "version": "7.6.3", + "resolved": "https://registry.npmjs.org/semver/-/semver-7.6.3.tgz", + "integrity": "sha512-oVekP1cKtI+CTDvHWYFUcMtsK/00wmAEfyqKfNdARm8u1wNVhSgaX7A8d4UuIlUI5e84iEwOhs7ZPYRmzU9U6A==", + "dev": true + } + } + }, + "eslint-plugin-vuejs-accessibility": { + "version": "2.4.1", + "resolved": "https://registry.npmjs.org/eslint-plugin-vuejs-accessibility/-/eslint-plugin-vuejs-accessibility-2.4.1.tgz", + "integrity": "sha512-ZRZhPdslplZXSF71MtSG+zXYRAT5KiHR4JVuo/DERQf9noAkDvi5W418VOE1qllmJd7wTenndxi1q8XeDMxdHw==", + "dev": true, + "requires": { + "aria-query": "^5.3.0", + "emoji-regex": "^10.0.0", + "vue-eslint-parser": "^9.0.1" + }, + "dependencies": { + "semver": { + "version": "7.6.3", + "resolved": "https://registry.npmjs.org/semver/-/semver-7.6.3.tgz", + "integrity": "sha512-oVekP1cKtI+CTDvHWYFUcMtsK/00wmAEfyqKfNdARm8u1wNVhSgaX7A8d4UuIlUI5e84iEwOhs7ZPYRmzU9U6A==", + "dev": true + }, + "vue-eslint-parser": { + "version": "9.4.3", + "resolved": "https://registry.npmjs.org/vue-eslint-parser/-/vue-eslint-parser-9.4.3.tgz", + "integrity": "sha512-2rYRLWlIpaiN8xbPiDyXZXRgLGOtWxERV7ND5fFAv5qo1D2N9Fu9MNajBNc6o13lZ+24DAWCkQCvj4klgmcITg==", + "dev": true, + "requires": { + "debug": "^4.3.4", + "eslint-scope": "^7.1.1", + "eslint-visitor-keys": "^3.3.0", + "espree": "^9.3.1", + "esquery": "^1.4.0", + "lodash": "^4.17.21", + "semver": "^7.3.6" + } + } + } + }, + "eslint-scope": { + "version": "7.2.2", + "resolved": "https://registry.npmjs.org/eslint-scope/-/eslint-scope-7.2.2.tgz", + "integrity": "sha512-dOt21O7lTMhDM+X9mB4GX+DZrZtCUJPL/wlcTqxyrx5IvO0IYtILdtrQGQp+8n5S0gwSVmOf9NQrjMOgfQZlIg==", + "dev": true, + "requires": { + "esrecurse": "^4.3.0", + "estraverse": "^5.2.0" + } + }, + "eslint-utils": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/eslint-utils/-/eslint-utils-3.0.0.tgz", + "integrity": "sha512-uuQC43IGctw68pJA1RgbQS8/NP7rch6Cwd4j3ZBtgo4/8Flj4eGE7ZYSZRN3iq5pVUv6GPdW5Z1RFleo84uLDA==", + "dev": true, + "requires": { + "eslint-visitor-keys": "^2.0.0" + }, + "dependencies": { + "eslint-visitor-keys": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-2.1.0.tgz", + "integrity": "sha512-0rSmRBzXgDzIsD6mGdJgevzgezI534Cer5L/vyMX0kHzT/jiB43jRhd9YUlMGYLQy2zprNmoT8qasCGtY+QaKw==", + "dev": true + } + } + }, + "eslint-visitor-keys": { + "version": "3.4.3", + "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-3.4.3.tgz", + "integrity": "sha512-wpc+LXeiyiisxPlEkUzU6svyS1frIO3Mgxj1fdy7Pm8Ygzguax2N3Fa/D/ag1WqbOprdI+uY6wMUl8/a2G+iag==", + "dev": true + }, + "eslint-webpack-plugin": { + "version": "3.2.0", + "resolved": "https://registry.npmjs.org/eslint-webpack-plugin/-/eslint-webpack-plugin-3.2.0.tgz", + "integrity": "sha512-avrKcGncpPbPSUHX6B3stNGzkKFto3eL+DKM4+VyMrVnhPc3vRczVlCq3uhuFOdRvDHTVXuzwk1ZKUrqDQHQ9w==", + "dev": true, + "requires": { + "@types/eslint": "^7.29.0 || ^8.4.1", + "jest-worker": "^28.0.2", + "micromatch": "^4.0.5", + "normalize-path": "^3.0.0", + "schema-utils": "^4.0.0" + }, + "dependencies": { + "ajv": { + "version": "8.17.1", + "resolved": "https://registry.npmjs.org/ajv/-/ajv-8.17.1.tgz", + "integrity": "sha512-B/gBuNg5SiMTrPkC+A2+cW0RszwxYmn6VYxB/inlBStS5nx6xHIt/ehKRhIMhqusl7a8LjQoZnjCs5vhwxOQ1g==", + "dev": true, + "requires": { + "fast-deep-equal": "^3.1.3", + "fast-uri": "^3.0.1", + "json-schema-traverse": "^1.0.0", + "require-from-string": "^2.0.2" + } + }, + "ajv-keywords": { + "version": "5.1.0", + "resolved": "https://registry.npmjs.org/ajv-keywords/-/ajv-keywords-5.1.0.tgz", + "integrity": "sha512-YCS/JNFAUyr5vAuhk1DWm1CBxRHW9LbJ2ozWeemrIqpbsqKjHVxYPyi5GC0rjZIT5JxJ3virVTS8wk4i/Z+krw==", + "dev": true, + "requires": { + "fast-deep-equal": "^3.1.3" + } + }, + "json-schema-traverse": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/json-schema-traverse/-/json-schema-traverse-1.0.0.tgz", + "integrity": "sha512-NM8/P9n3XjXhIZn1lLhkFaACTOURQXjWhV4BA/RnOv8xvgqtqpAX9IO4mRQxSx1Rlo4tqzeqb0sOlruaOy3dug==", + "dev": true + }, + "schema-utils": { + "version": "4.2.0", + "resolved": "https://registry.npmjs.org/schema-utils/-/schema-utils-4.2.0.tgz", + "integrity": "sha512-L0jRsrPpjdckP3oPug3/VxNKt2trR8TcabrM6FOAAlvC/9Phcmm+cuAgTlxBqdBR1WJx7Naj9WHw+aOmheSVbw==", + "dev": true, + "requires": { + "@types/json-schema": "^7.0.9", + "ajv": "^8.9.0", + "ajv-formats": "^2.1.1", + "ajv-keywords": "^5.1.0" + } + } + } + }, + "espree": { + "version": "9.6.1", + "resolved": "https://registry.npmjs.org/espree/-/espree-9.6.1.tgz", + "integrity": "sha512-oruZaFkjorTpF32kDSI5/75ViwGeZginGGy2NoOSg3Q9bnwlnmDm4HLnkl0RE3n+njDXR037aY1+x58Z/zFdwQ==", + "dev": true, + "requires": { + "acorn": "^8.9.0", + "acorn-jsx": "^5.3.2", + "eslint-visitor-keys": "^3.4.1" + } + }, + "esquery": { + "version": "1.6.0", + "resolved": "https://registry.npmjs.org/esquery/-/esquery-1.6.0.tgz", + "integrity": "sha512-ca9pw9fomFcKPvFLXhBKUK90ZvGibiGOvRJNbjljY7s7uq/5YO4BOzcYtJqExdx99rF6aAcnRxHmcUHcz6sQsg==", + "dev": true, + "requires": { + "estraverse": "^5.1.0" + } + }, + "esrecurse": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/esrecurse/-/esrecurse-4.3.0.tgz", + "integrity": "sha512-KmfKL3b6G+RXvP8N1vr3Tq1kL/oCFgn2NYXEtqP8/L3pKapUA4G8cFVaoF3SU323CD4XypR/ffioHmkti6/Tag==", + "dev": true, + "requires": { + "estraverse": "^5.2.0" + } + }, + "estraverse": { + "version": "5.3.0", + "resolved": "https://registry.npmjs.org/estraverse/-/estraverse-5.3.0.tgz", + "integrity": "sha512-MMdARuVEQziNTeJD8DgMqmhwR11BRQ/cBP+pLtYdSTnf3MIO8fFeiINEbX36ZdNlfU/7A9f3gUw49B3oQsvwBA==", + "dev": true + }, + "esutils": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/esutils/-/esutils-2.0.3.tgz", + "integrity": "sha512-kVscqXk4OCp68SZ0dkgEKVi6/8ij300KBWTJq32P/dYeWTSwK41WyTxalN1eRmA5Z9UU/LX9D7FWSmV9SAYx6g==", + "dev": true + }, + "events": { + "version": "3.3.0", + "resolved": "https://registry.npmjs.org/events/-/events-3.3.0.tgz", + "integrity": "sha512-mQw+2fkQbALzQ7V0MY0IqdnXNOeTtP4r0lN9z7AAawCXgqea7bDii20AYrIBrFd/Hx0M2Ocz6S111CaFkUcb0Q==", + "dev": true + }, + "fast-deep-equal": { + "version": "3.1.3", + "resolved": "https://registry.npmjs.org/fast-deep-equal/-/fast-deep-equal-3.1.3.tgz", + "integrity": "sha512-f3qQ9oQy9j2AhBe/H9VC91wLmKBCCU/gDOnKNAYG5hswO7BLKj09Hc5HYNz9cGI++xlpDCIgDaitVs03ATR84Q==", + "dev": true + }, + "fast-json-stable-stringify": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/fast-json-stable-stringify/-/fast-json-stable-stringify-2.1.0.tgz", + "integrity": "sha512-lhd/wF+Lk98HZoTCtlVraHtfh5XYijIjalXck7saUtuanSDyLMxnHhSXEDJqHxD7msR8D0uCmqlkwjCV8xvwHw==", + "dev": true + }, + "fast-levenshtein": { + "version": "2.0.6", + "resolved": "https://registry.npmjs.org/fast-levenshtein/-/fast-levenshtein-2.0.6.tgz", + "integrity": "sha512-DCXu6Ifhqcks7TZKY3Hxp3y6qphY5SJZmrWMDrKcERSOXWQdMhU9Ig/PYrzyw/ul9jOIyh0N4M0tbC5hodg8dw==", + "dev": true + }, + "fast-uri": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/fast-uri/-/fast-uri-3.0.1.tgz", + "integrity": "sha512-MWipKbbYiYI0UC7cl8m/i/IWTqfC8YXsqjzybjddLsFjStroQzsHXkc73JutMvBiXmOvapk+axIl79ig5t55Bw==", + "dev": true + }, + "fastest-levenshtein": { + "version": "1.0.16", + "resolved": "https://registry.npmjs.org/fastest-levenshtein/-/fastest-levenshtein-1.0.16.tgz", + "integrity": "sha512-eRnCtTTtGZFpQCwhJiUOuxPQWRXVKYDn0b2PeHfXL6/Zi53SLAzAHfVhVWK2AryC/WH05kGfxhFIPvTF0SXQzg==", + "dev": true + }, + "fastq": { + "version": "1.17.1", + "resolved": "https://registry.npmjs.org/fastq/-/fastq-1.17.1.tgz", + "integrity": "sha512-sRVD3lWVIXWg6By68ZN7vho9a1pQcN/WBFaAAsDDFzlJjvoGx0P8z7V1t72grFJfJhu3YPZBuu25f7Kaw2jN1w==", + "dev": true, + "requires": { + "reusify": "^1.0.4" + } + }, + "file-entry-cache": { + "version": "6.0.1", + "resolved": "https://registry.npmjs.org/file-entry-cache/-/file-entry-cache-6.0.1.tgz", + "integrity": "sha512-7Gps/XWymbLk2QLYK4NzpMOrYjMhdIxXuIvy2QBsLE6ljuodKvdkWs/cpyJJ3CVIVpH0Oi1Hvg1ovbMzLdFBBg==", + "dev": true, + "requires": { + "flat-cache": "^3.0.4" + } + }, + "fill-range": { + "version": "7.1.1", + "resolved": "https://registry.npmjs.org/fill-range/-/fill-range-7.1.1.tgz", + "integrity": "sha512-YsGpe3WHLK8ZYi4tWDg2Jy3ebRz2rXowDxnld4bkQB00cc/1Zw9AWnC0i9ztDJitivtQvaI9KaLyKrc+hBW0yg==", + "dev": true, + "requires": { + "to-regex-range": "^5.0.1" + } + }, + "find-cache-dir": { + "version": "3.3.2", + "resolved": "https://registry.npmjs.org/find-cache-dir/-/find-cache-dir-3.3.2.tgz", + "integrity": "sha512-wXZV5emFEjrridIgED11OoUKLxiYjAcqot/NJdAkOhlJ+vGzwhOAfcG5OX1jP+S0PcjEn8bdMJv+g2jwQ3Onig==", + "dev": true, + "requires": { + "commondir": "^1.0.1", + "make-dir": "^3.0.2", + "pkg-dir": "^4.1.0" + } + }, + "find-up": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/find-up/-/find-up-5.0.0.tgz", + "integrity": "sha512-78/PXT1wlLLDgTzDs7sjq9hzz0vXD+zn+7wypEe4fXQxCmdmqfGsEPQxmiCSQI3ajFV91bVSsvNtrJRiW6nGng==", + "dev": true, + "requires": { + "locate-path": "^6.0.0", + "path-exists": "^4.0.0" + } + }, + "flat": { + "version": "5.0.2", + "resolved": "https://registry.npmjs.org/flat/-/flat-5.0.2.tgz", + "integrity": "sha512-b6suED+5/3rTpUBdG1gupIl8MPFCAMA0QXwmljLhvCUKcUvdE4gWky9zpuGCcXHOsz4J9wPGNWq6OKpmIzz3hQ==", + "dev": true + }, + "flat-cache": { + "version": "3.2.0", + "resolved": "https://registry.npmjs.org/flat-cache/-/flat-cache-3.2.0.tgz", + "integrity": "sha512-CYcENa+FtcUKLmhhqyctpclsq7QF38pKjZHsGNiSQF5r4FtoKDWabFDl3hzaEQMvT1LHEysw5twgLvpYYb4vbw==", + "dev": true, + "requires": { + "flatted": "^3.2.9", + "keyv": "^4.5.3", + "rimraf": "^3.0.2" + } + }, + "flatted": { + "version": "3.3.1", + "resolved": "https://registry.npmjs.org/flatted/-/flatted-3.3.1.tgz", + "integrity": "sha512-X8cqMLLie7KsNUDSdzeN8FYK9rEt4Dt67OsG/DNGnYTSDBG4uFAJFBnUeiV+zCVAvwFy56IjM9sH51jVaEhNxw==", + "dev": true + }, + "fs.realpath": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/fs.realpath/-/fs.realpath-1.0.0.tgz", + "integrity": "sha512-OO0pH2lK6a0hZnAdau5ItzHPI6pUlvI7jMVnxUQRtw4owF2wk8lOSabtGDCTP4Ggrg2MbGnWO9X8K1t4+fGMDw==", + "dev": true + }, + "fsevents": { + "version": "2.3.3", + "resolved": "https://registry.npmjs.org/fsevents/-/fsevents-2.3.3.tgz", + "integrity": "sha512-5xoDfX+fL7faATnagmWPpbFtwh/R77WmMMqqHGS65C3vvB0YHrgF+B1YmZ3441tMj5n63k0212XNoJwzlhffQw==", + "dev": true, + "optional": true + }, + "function-bind": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/function-bind/-/function-bind-1.1.2.tgz", + "integrity": "sha512-7XHNxH7qX9xG5mIwxkhumTox/MIRNcOgDrxWsMt2pAr23WHp6MrRlN7FBSFpCpr+oVO0F744iUgR82nJMfG2SA==", + "dev": true + }, + "gensync": { + "version": "1.0.0-beta.2", + "resolved": "https://registry.npmjs.org/gensync/-/gensync-1.0.0-beta.2.tgz", + "integrity": "sha512-3hN7NaskYvMDLQY55gnW3NQ+mesEAepTqlg+VEbj7zzqEMBVNhzcGYYeqFo/TlYz6eQiFcp1HcsCZO+nGgS8zg==", + "dev": true, + "peer": true + }, + "glob": { + "version": "7.2.3", + "resolved": "https://registry.npmjs.org/glob/-/glob-7.2.3.tgz", + "integrity": "sha512-nFR0zLpU2YCaRxwoCJvL6UvCH2JFyFVIvwTLsIf21AuHlMskA1hhTdk+LlYJtOlYt9v6dvszD2BGRqBL+iQK9Q==", + "dev": true, + "requires": { + "fs.realpath": "^1.0.0", + "inflight": "^1.0.4", + "inherits": "2", + "minimatch": "^3.1.1", + "once": "^1.3.0", + "path-is-absolute": "^1.0.0" + } + }, + "glob-parent": { + "version": "6.0.2", + "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-6.0.2.tgz", + "integrity": "sha512-XxwI8EOhVQgWp6iDL+3b0r86f4d6AX6zSU55HfB4ydCEuXLXc5FcYeOu+nnGftS4TEju/11rt4KJPTMgbfmv4A==", + "dev": true, + "requires": { + "is-glob": "^4.0.3" + } + }, + "glob-to-regexp": { + "version": "0.4.1", + "resolved": "https://registry.npmjs.org/glob-to-regexp/-/glob-to-regexp-0.4.1.tgz", + "integrity": "sha512-lkX1HJXwyMcprw/5YUZc2s7DrpAiHB21/V+E1rHUrVNokkvB6bqMzT0VfV6/86ZNabt1k14YOIaT7nDvOX3Iiw==", + "dev": true + }, + "globals": { + "version": "11.12.0", + "resolved": "https://registry.npmjs.org/globals/-/globals-11.12.0.tgz", + "integrity": "sha512-WOBp/EEGUiIsJSp7wcv/y6MO+lV9UoncWqxuFfm8eBwzWNgyfBd6Gz+IeKQ9jCmyhoH99g15M3T+QaVHFjizVA==", + "dev": true, + "peer": true + }, + "graceful-fs": { + "version": "4.2.11", + "resolved": "https://registry.npmjs.org/graceful-fs/-/graceful-fs-4.2.11.tgz", + "integrity": "sha512-RbJ5/jmFcNNCcDV5o9eTnBLJ/HszWV0P73bc+Ff4nS/rJj+YaS6IGyiOL0VoBYX+l1Wrl3k63h/KrH+nhJ0XvQ==", + "dev": true + }, + "graphemer": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/graphemer/-/graphemer-1.4.0.tgz", + "integrity": "sha512-EtKwoO6kxCL9WO5xipiHTZlSzBm7WLT627TqC/uVRd0HKmq8NXyebnNYxDoBi7wt8eTWrUrKXCOVaFq9x1kgag==", + "dev": true + }, + "has-flag": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-3.0.0.tgz", + "integrity": "sha512-sKJf1+ceQBr4SMkvQnBDNDtf4TXpVhVGateu0t918bl30FnbE2m4vNLX+VWe/dpjlb+HugGYzW7uQXH98HPEYw==", + "dev": true, + "peer": true + }, + "hash-sum": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/hash-sum/-/hash-sum-1.0.2.tgz", + "integrity": "sha512-fUs4B4L+mlt8/XAtSOGMUO1TXmAelItBPtJG7CyHJfYTdDjwisntGO2JQz7oUsatOY9o68+57eziUVNw/mRHmA==", + "dev": true + }, + "hasown": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/hasown/-/hasown-2.0.2.tgz", + "integrity": "sha512-0hJU9SCPvmMzIBdZFqNPXWa6dqh7WdH0cII9y+CyS8rG3nL48Bclra9HmKhVVUHyPWNH5Y7xDwAB7bfgSjkUMQ==", + "dev": true, + "requires": { + "function-bind": "^1.1.2" + } + }, + "he": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/he/-/he-1.2.0.tgz", + "integrity": "sha512-F/1DnUGPopORZi0ni+CvrCgHQ5FyEAHRLSApuYWMmrbSwoN2Mn/7k+Gl38gJnR7yyDZk6WLXwiGod1JOWNDKGw==", + "dev": true + }, + "icss-utils": { + "version": "5.1.0", + "resolved": "https://registry.npmjs.org/icss-utils/-/icss-utils-5.1.0.tgz", + "integrity": "sha512-soFhflCVWLfRNOPU3iv5Z9VUdT44xFRbzjLsEzSr5AQmgqPMTHdU3PMT1Cf1ssx8fLNJDA1juftYl+PUcv3MqA==", + "dev": true, + "peer": true, + "requires": {} + }, + "ignore": { + "version": "5.3.2", + "resolved": "https://registry.npmjs.org/ignore/-/ignore-5.3.2.tgz", + "integrity": "sha512-hsBTNUqQTDwkWtcdYI2i06Y/nUBEsNEDJKjWdigLvegy8kDuJAS8uRlpkkcQpyEXL0Z/pjDy5HBmMjRCJ2gq+g==", + "dev": true + }, + "immutable": { + "version": "4.3.7", + "resolved": "https://registry.npmjs.org/immutable/-/immutable-4.3.7.tgz", + "integrity": "sha512-1hqclzwYwjRDFLjcFxOM5AYkkG0rpFPpr1RLPMEuGczoS7YA8gLhy8SWXYRAA/XwfEHpfo3cw5JGioS32fnMRw==", + "dev": true + }, + "import-fresh": { + "version": "3.3.0", + "resolved": "https://registry.npmjs.org/import-fresh/-/import-fresh-3.3.0.tgz", + "integrity": "sha512-veYYhQa+D1QBKznvhUHxb8faxlrwUnxseDAbAp457E0wLNio2bOSKnjYDhMj+YiAq61xrMGhQk9iXVk5FzgQMw==", + "dev": true, + "requires": { + "parent-module": "^1.0.0", + "resolve-from": "^4.0.0" + } + }, + "import-local": { + "version": "3.2.0", + "resolved": "https://registry.npmjs.org/import-local/-/import-local-3.2.0.tgz", + "integrity": "sha512-2SPlun1JUPWoM6t3F0dw0FkCF/jWY8kttcY4f599GLTSjh2OCuuhdTkJQsEcZzBqbXZGKMK2OqW1oZsjtf/gQA==", + "dev": true, + "requires": { + "pkg-dir": "^4.2.0", + "resolve-cwd": "^3.0.0" + } + }, + "imurmurhash": { + "version": "0.1.4", + "resolved": "https://registry.npmjs.org/imurmurhash/-/imurmurhash-0.1.4.tgz", + "integrity": "sha512-JmXMZ6wuvDmLiHEml9ykzqO6lwFbof0GG4IkcGaENdCRDDmMVnny7s5HsIgHCbaq0w2MyPhDqkhTUgS2LU2PHA==", + "dev": true + }, + "inflight": { + "version": "1.0.6", + "resolved": "https://registry.npmjs.org/inflight/-/inflight-1.0.6.tgz", + "integrity": "sha512-k92I/b08q4wvFscXCLvqfsHCrjrF7yiXsQuIVvVE7N82W3+aqpzuUdBbfhWcy/FZR3/4IgflMgKLOsvPDrGCJA==", + "dev": true, + "requires": { + "once": "^1.3.0", + "wrappy": "1" + } + }, + "inherits": { + "version": "2.0.4", + "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.4.tgz", + "integrity": "sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==", + "dev": true + }, + "interpret": { + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/interpret/-/interpret-2.2.0.tgz", + "integrity": "sha512-Ju0Bz/cEia55xDwUWEa8+olFpCiQoypjnQySseKtmjNrnps3P+xfpUmGr90T7yjlVJmOtybRvPXhKMbHr+fWnw==", + "dev": true + }, + "is-binary-path": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/is-binary-path/-/is-binary-path-2.1.0.tgz", + "integrity": "sha512-ZMERYes6pDydyuGidse7OsHxtbI7WVeUEozgR/g7rd0xUimYNlvZRE/K2MgZTjWy725IfelLeVcEM97mmtRGXw==", + "dev": true, + "requires": { + "binary-extensions": "^2.0.0" + } + }, + "is-core-module": { + "version": "2.15.0", + "resolved": "https://registry.npmjs.org/is-core-module/-/is-core-module-2.15.0.tgz", + "integrity": "sha512-Dd+Lb2/zvk9SKy1TGCt1wFJFo/MWBPMX5x7KcvLajWTGuomczdQX61PvY5yK6SVACwpoexWo81IfFyoKY2QnTA==", + "dev": true, + "requires": { + "hasown": "^2.0.2" + } + }, + "is-extglob": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/is-extglob/-/is-extglob-2.1.1.tgz", + "integrity": "sha512-SbKbANkN603Vi4jEZv49LeVJMn4yGwsbzZworEoyEiutsN3nJYdbO36zfhGJ6QEDpOZIFkDtnq5JRxmvl3jsoQ==", + "dev": true + }, + "is-glob": { + "version": "4.0.3", + "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-4.0.3.tgz", + "integrity": "sha512-xelSayHH36ZgE7ZWhli7pW34hNbNl8Ojv5KVmkJD4hBdD3th8Tfk9vYasLM+mXWOZhFkgZfxhLSnrwRr4elSSg==", + "dev": true, + "requires": { + "is-extglob": "^2.1.1" + } + }, + "is-number": { + "version": "7.0.0", + "resolved": "https://registry.npmjs.org/is-number/-/is-number-7.0.0.tgz", + "integrity": "sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng==", + "dev": true + }, + "is-path-inside": { + "version": "3.0.3", + "resolved": "https://registry.npmjs.org/is-path-inside/-/is-path-inside-3.0.3.tgz", + "integrity": "sha512-Fd4gABb+ycGAmKou8eMftCupSir5lRxqf4aD/vd0cD2qc4HL07OjCeuHMr8Ro4CoMaeCKDB0/ECBOVWjTwUvPQ==", + "dev": true + }, + "is-plain-object": { + "version": "2.0.4", + "resolved": "https://registry.npmjs.org/is-plain-object/-/is-plain-object-2.0.4.tgz", + "integrity": "sha512-h5PpgXkWitc38BBMYawTYMWJHFZJVnBquFE57xFpjB8pJFiF6gZ+bU+WyI/yqXiFR5mdLsgYNaPe8uao6Uv9Og==", + "dev": true, + "requires": { + "isobject": "^3.0.1" + } + }, + "isexe": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/isexe/-/isexe-2.0.0.tgz", + "integrity": "sha512-RHxMLp9lnKHGHRng9QFhRCMbYAcVpn69smSGcq3f36xjgVVWThj4qqLbTLlq7Ssj8B+fIQ1EuCEGI2lKsyQeIw==", + "dev": true + }, + "isobject": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/isobject/-/isobject-3.0.1.tgz", + "integrity": "sha512-WhB9zCku7EGTj/HQQRz5aUQEUeoQZH2bWcltRErOpymJ4boYE6wL9Tbr23krRPSZ+C5zqNSrSw+Cc7sZZ4b7vg==", + "dev": true + }, + "jest-worker": { + "version": "28.1.3", + "resolved": "https://registry.npmjs.org/jest-worker/-/jest-worker-28.1.3.tgz", + "integrity": "sha512-CqRA220YV/6jCo8VWvAt1KKx6eek1VIHMPeLEbpcfSfkEeWyBNppynM/o6q+Wmw+sOhos2ml34wZbSX3G13//g==", + "dev": true, + "requires": { + "@types/node": "*", + "merge-stream": "^2.0.0", + "supports-color": "^8.0.0" + }, + "dependencies": { + "has-flag": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", + "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", + "dev": true + }, + "supports-color": { + "version": "8.1.1", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-8.1.1.tgz", + "integrity": "sha512-MpUEN2OodtUzxvKQl72cUF7RQ5EiHsGvSsVG0ia9c5RbWGL2CI4C7EpPS8UTBIplnlzZiNuV56w+FuNxy3ty2Q==", + "dev": true, + "requires": { + "has-flag": "^4.0.0" + } + } + } + }, + "js-tokens": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/js-tokens/-/js-tokens-4.0.0.tgz", + "integrity": "sha512-RdJUflcE3cUzKiMqQgsCu06FPu9UdIJO0beYbPhHN4k6apgJtifcoCtT9bcxOpYBtpD2kCM6Sbzg4CausW/PKQ==", + "dev": true, + "peer": true + }, + "js-yaml": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-4.1.0.tgz", + "integrity": "sha512-wpxZs9NoxZaJESJGIZTyDEaYpl0FKSA+FB9aJiyemKhMwkxQg63h4T1KJgUGHpTqPDNRcmmYLugrRjJlBtWvRA==", + "dev": true, + "requires": { + "argparse": "^2.0.1" + } + }, + "jsesc": { + "version": "2.5.2", + "resolved": "https://registry.npmjs.org/jsesc/-/jsesc-2.5.2.tgz", + "integrity": "sha512-OYu7XEzjkCQ3C5Ps3QIZsQfNpqoJyZZA99wd9aWd05NCtC5pWOkShK2mkL6HXQR6/Cy2lbNdPlZBpuQHXE63gA==", + "dev": true, + "peer": true + }, + "json-buffer": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/json-buffer/-/json-buffer-3.0.1.tgz", + "integrity": "sha512-4bV5BfR2mqfQTJm+V5tPPdf+ZpuhiIvTuAB5g8kcrXOZpTT/QwwVRWBywX1ozr6lEuPdbHxwaJlm9G6mI2sfSQ==", + "dev": true + }, + "json-parse-even-better-errors": { + "version": "2.3.1", + "resolved": "https://registry.npmjs.org/json-parse-even-better-errors/-/json-parse-even-better-errors-2.3.1.tgz", + "integrity": "sha512-xyFwyhro/JEof6Ghe2iz2NcXoj2sloNsWr/XsERDK/oiPCfaNhl5ONfp+jQdAZRQQ0IJWNzH9zIZF7li91kh2w==", + "dev": true + }, + "json-schema-traverse": { + "version": "0.4.1", + "resolved": "https://registry.npmjs.org/json-schema-traverse/-/json-schema-traverse-0.4.1.tgz", + "integrity": "sha512-xbbCH5dCYU5T8LcEhhuh7HJ88HXuW3qsI3Y0zOZFKfZEHcpWiHU/Jxzk629Brsab/mMiHQti9wMP+845RPe3Vg==", + "dev": true + }, + "json-stable-stringify-without-jsonify": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/json-stable-stringify-without-jsonify/-/json-stable-stringify-without-jsonify-1.0.1.tgz", + "integrity": "sha512-Bdboy+l7tA3OGW6FjyFHWkP5LuByj1Tk33Ljyq0axyzdk9//JSi2u3fP1QSmd1KNwq6VOKYGlAu87CisVir6Pw==", + "dev": true + }, + "json5": { + "version": "2.2.3", + "resolved": "https://registry.npmjs.org/json5/-/json5-2.2.3.tgz", + "integrity": "sha512-XmOWe7eyHYH14cLdVPoyg+GOH3rYX++KpzrylJwSW98t3Nk+U8XOl8FWKOgwtzdb8lXGf6zYwDUzeHMWfxasyg==", + "dev": true + }, + "keyv": { + "version": "4.5.4", + "resolved": "https://registry.npmjs.org/keyv/-/keyv-4.5.4.tgz", + "integrity": "sha512-oxVHkHR/EJf2CNXnWxRLW6mg7JyCCUcG0DtEGmL2ctUo1PNTin1PUil+r/+4r5MpVgC/fn1kjsx7mjSujKqIpw==", + "dev": true, + "requires": { + "json-buffer": "3.0.1" + } + }, + "kind-of": { + "version": "6.0.3", + "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-6.0.3.tgz", + "integrity": "sha512-dcS1ul+9tmeD95T+x28/ehLgd9mENa3LsvDTtzm3vyBEO7RPptvAD+t44WVXaUjTBRcrpFeFlC8WCruUR456hw==", + "dev": true + }, + "levn": { + "version": "0.4.1", + "resolved": "https://registry.npmjs.org/levn/-/levn-0.4.1.tgz", + "integrity": "sha512-+bT2uH4E5LGE7h/n3evcS/sQlJXCpIp6ym8OWJ5eV6+67Dsql/LaaT7qJBAt2rzfoa/5QBGBhxDix1dMt2kQKQ==", + "dev": true, + "requires": { + "prelude-ls": "^1.2.1", + "type-check": "~0.4.0" + } + }, + "loader-runner": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/loader-runner/-/loader-runner-4.3.0.tgz", + "integrity": "sha512-3R/1M+yS3j5ou80Me59j7F9IMs4PXs3VqRrm0TU3AbKPxlmpoY1TNscJV/oGJXo8qCatFGTfDbY6W6ipGOYXfg==", + "dev": true + }, + "loader-utils": { + "version": "2.0.4", + "resolved": "https://registry.npmjs.org/loader-utils/-/loader-utils-2.0.4.tgz", + "integrity": "sha512-xXqpXoINfFhgua9xiqD8fPFHgkoq1mmmpE92WlDbm9rNRd/EbRb+Gqf908T2DMfuHjjJlksiK2RbHVOdD/MqSw==", + "dev": true, + "requires": { + "big.js": "^5.2.2", + "emojis-list": "^3.0.0", + "json5": "^2.1.2" + } + }, + "locate-path": { + "version": "6.0.0", + "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-6.0.0.tgz", + "integrity": "sha512-iPZK6eYjbxRu3uB4/WZ3EsEIMJFMqAoopl3R+zuq0UjcAm/MO6KCweDgPfP3elTztoKP3KtnVHxTn2NHBSDVUw==", + "dev": true, + "requires": { + "p-locate": "^5.0.0" + } + }, + "lodash": { + "version": "4.17.21", + "resolved": "https://registry.npmjs.org/lodash/-/lodash-4.17.21.tgz", + "integrity": "sha512-v2kDEe57lecTulaDIuNTPy3Ry4gLGJ6Z1O3vE1krgXZNrsQ+LFTGHVxVjcXPs17LhbZVGedAJv8XZ1tvj5FvSg==", + "dev": true + }, + "lodash.merge": { + "version": "4.6.2", + "resolved": "https://registry.npmjs.org/lodash.merge/-/lodash.merge-4.6.2.tgz", + "integrity": "sha512-0KpjqXRVvrYyCsX1swR/XTK0va6VQkQM6MNo7PqW77ByjAhoARA8EfrP1N4+KlKj8YS0ZUCtRT/YUuhyYDujIQ==", + "dev": true + }, + "lru-cache": { + "version": "5.1.1", + "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-5.1.1.tgz", + "integrity": "sha512-KpNARQA3Iwv+jTA0utUVVbrh+Jlrr1Fv0e56GGzAFOXN7dk/FviaDW8LHmK52DlcH4WP2n6gI8vN1aesBFgo9w==", + "dev": true, + "peer": true, + "requires": { + "yallist": "^3.0.2" + } + }, + "make-dir": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/make-dir/-/make-dir-3.1.0.tgz", + "integrity": "sha512-g3FeP20LNwhALb/6Cz6Dd4F2ngze0jz7tbzrD2wAV+o9FeNHe4rL+yK2md0J/fiSf1sa1ADhXqi5+oVwOM/eGw==", + "dev": true, + "requires": { + "semver": "^6.0.0" + } + }, + "merge-source-map": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/merge-source-map/-/merge-source-map-1.1.0.tgz", + "integrity": "sha512-Qkcp7P2ygktpMPh2mCQZaf3jhN6D3Z/qVZHSdWvQ+2Ef5HgRAPBO57A77+ENm0CPx2+1Ce/MYKi3ymqdfuqibw==", + "dev": true, + "requires": { + "source-map": "^0.6.1" + } + }, + "merge-stream": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/merge-stream/-/merge-stream-2.0.0.tgz", + "integrity": "sha512-abv/qOcuPfk3URPfDzmZU1LKmuw8kT+0nIHvKrKgFrwifol/doWcdA4ZqsWQ8ENrFKkd67Mfpo/LovbIUsbt3w==", + "dev": true + }, + "micromatch": { + "version": "4.0.7", + "resolved": "https://registry.npmjs.org/micromatch/-/micromatch-4.0.7.tgz", + "integrity": "sha512-LPP/3KorzCwBxfeUuZmaR6bG2kdeHSbe0P2tY3FLRU4vYrjYz5hI4QZwV0njUx3jeuKe67YukQ1LSPZBKDqO/Q==", + "dev": true, + "requires": { + "braces": "^3.0.3", + "picomatch": "^2.3.1" + } + }, + "mime-db": { + "version": "1.52.0", + "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.52.0.tgz", + "integrity": "sha512-sPU4uV7dYlvtWJxwwxHD0PuihVNiE7TyAbQ5SWxDCB9mUYvOgroQOwYQQOKPJ8CIbE+1ETVlOoK1UC2nU3gYvg==", + "dev": true + }, + "mime-types": { + "version": "2.1.35", + "resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.1.35.tgz", + "integrity": "sha512-ZDY+bPm5zTTF+YpCrAU9nK0UgICYPT0QtT1NZWFv4s++TNkcgVaT0g6+4R2uI4MjQjzysHB1zxuWL50hzaeXiw==", + "dev": true, + "requires": { + "mime-db": "1.52.0" + } + }, + "minimatch": { + "version": "3.1.2", + "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.1.2.tgz", + "integrity": "sha512-J7p63hRiAjw1NDEww1W7i37+ByIrOWO5XQQAzZ3VOcL0PNybwpfmV/N05zFAzwQ9USyEcX6t3UO+K5aqBQOIHw==", + "dev": true, + "requires": { + "brace-expansion": "^1.1.7" + } + }, + "minimist": { + "version": "1.2.8", + "resolved": "https://registry.npmjs.org/minimist/-/minimist-1.2.8.tgz", + "integrity": "sha512-2yyAR8qBkN3YuheJanUpWC5U3bb5osDywNB8RzDVlDwDHbocAJveqqj1u8+SVD7jkWT4yvsHCpWqqWqAxb0zCA==", + "dev": true + }, + "ms": { + "version": "2.1.2", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz", + "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==", + "dev": true + }, + "nanoid": { + "version": "3.3.7", + "resolved": "https://registry.npmjs.org/nanoid/-/nanoid-3.3.7.tgz", + "integrity": "sha512-eSRppjcPIatRIMC1U6UngP8XFcz8MQWGQdt1MTBQ7NaAmvXDfvNxbvWV3x2y6CdEUciCSsDHDQZbhYaB8QEo2g==", + "dev": true + }, + "natural-compare": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/natural-compare/-/natural-compare-1.4.0.tgz", + "integrity": "sha512-OWND8ei3VtNC9h7V60qff3SVobHr996CTwgxubgyQYEpg290h9J0buyECNNJexkFm5sOajh5G116RYA1c8ZMSw==", + "dev": true + }, + "neo-async": { + "version": "2.6.2", + "resolved": "https://registry.npmjs.org/neo-async/-/neo-async-2.6.2.tgz", + "integrity": "sha512-Yd3UES5mWCSqR+qNT93S3UoYUkqAZ9lLg8a7g9rimsWmYGK8cVToA4/sF3RrshdyV3sAGMXVUmpMYOw+dLpOuw==", + "dev": true + }, + "node-releases": { + "version": "2.0.18", + "resolved": "https://registry.npmjs.org/node-releases/-/node-releases-2.0.18.tgz", + "integrity": "sha512-d9VeXT4SJ7ZeOqGX6R5EM022wpL+eWPooLI+5UpWn2jCT1aosUQEhQP214x33Wkwx3JQMvIm+tIoVOdodFS40g==", + "dev": true + }, + "normalize-path": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/normalize-path/-/normalize-path-3.0.0.tgz", + "integrity": "sha512-6eZs5Ls3WtCisHWp9S2GUy8dqkpGi4BVSz3GaqiE6ezub0512ESztXUwUB6C6IKbQkY2Pnb/mD4WYojCRwcwLA==", + "dev": true + }, + "nth-check": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/nth-check/-/nth-check-2.1.1.tgz", + "integrity": "sha512-lqjrjmaOoAnWfMmBPL+XNnynZh2+swxiX3WUE0s4yEHI6m+AwrK2UZOimIRl3X/4QctVqS8AiZjFqyOGrMXb/w==", + "dev": true, + "requires": { + "boolbase": "^1.0.0" + } + }, + "once": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/once/-/once-1.4.0.tgz", + "integrity": "sha512-lNaJgI+2Q5URQBkccEKHTQOPaXdUxnZZElQTZY0MFUAuaEqe1E+Nyvgdz/aIyNi6Z9MzO5dv1H8n58/GELp3+w==", + "dev": true, + "requires": { + "wrappy": "1" + } + }, + "optionator": { + "version": "0.9.4", + "resolved": "https://registry.npmjs.org/optionator/-/optionator-0.9.4.tgz", + "integrity": "sha512-6IpQ7mKUxRcZNLIObR0hz7lxsapSSIYNZJwXPGeF0mTVqGKFIXj1DQcMoT22S3ROcLyY/rz0PWaWZ9ayWmad9g==", + "dev": true, + "requires": { + "deep-is": "^0.1.3", + "fast-levenshtein": "^2.0.6", + "levn": "^0.4.1", + "prelude-ls": "^1.2.1", + "type-check": "^0.4.0", + "word-wrap": "^1.2.5" + } + }, + "p-limit": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/p-limit/-/p-limit-3.1.0.tgz", + "integrity": "sha512-TYOanM3wGwNGsZN2cVTYPArw454xnXj5qmWF1bEoAc4+cU/ol7GVh7odevjp1FNHduHc3KZMcFduxU5Xc6uJRQ==", + "dev": true, + "requires": { + "yocto-queue": "^0.1.0" + } + }, + "p-locate": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-5.0.0.tgz", + "integrity": "sha512-LaNjtRWUBY++zB5nE/NwcaoMylSPk+S+ZHNB1TzdbMJMny6dynpAGt7X/tl/QYq3TIeE6nxHppbo2LGymrG5Pw==", + "dev": true, + "requires": { + "p-limit": "^3.0.2" + } + }, + "p-try": { + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/p-try/-/p-try-2.2.0.tgz", + "integrity": "sha512-R4nPAVTAU0B9D35/Gk3uJf/7XYbQcyohSKdvAxIRSNghFl4e71hVoGnBNQz9cWaXxO2I10KTC+3jMdvvoKw6dQ==", + "dev": true + }, + "parent-module": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/parent-module/-/parent-module-1.0.1.tgz", + "integrity": "sha512-GQ2EWRpQV8/o+Aw8YqtfZZPfNRWZYkbidE9k5rpl/hC3vtHHBfGm2Ifi6qWV+coDGkrUKZAxE3Lot5kcsRlh+g==", + "dev": true, + "requires": { + "callsites": "^3.0.0" + } + }, + "path-exists": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/path-exists/-/path-exists-4.0.0.tgz", + "integrity": "sha512-ak9Qy5Q7jYb2Wwcey5Fpvg2KoAc/ZIhLSLOSBmRmygPsGwkVVt0fZa0qrtMz+m6tJTAHfZQ8FnmB4MG4LWy7/w==", + "dev": true + }, + "path-is-absolute": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/path-is-absolute/-/path-is-absolute-1.0.1.tgz", + "integrity": "sha512-AVbw3UJ2e9bq64vSaS9Am0fje1Pa8pbGqTTsmXfaIiMpnr5DlDhfJOuLj9Sf95ZPVDAUerDfEk88MPmPe7UCQg==", + "dev": true + }, + "path-key": { + "version": "3.1.1", + "resolved": "https://registry.npmjs.org/path-key/-/path-key-3.1.1.tgz", + "integrity": "sha512-ojmeN0qd+y0jszEtoY48r0Peq5dwMEkIlCOu6Q5f41lfkswXuKtYrhgoTpLnyIcHm24Uhqx+5Tqm2InSwLhE6Q==", + "dev": true + }, + "path-parse": { + "version": "1.0.7", + "resolved": "https://registry.npmjs.org/path-parse/-/path-parse-1.0.7.tgz", + "integrity": "sha512-LDJzPVEEEPR+y48z93A0Ed0yXb8pAByGWo/k5YYdYgpY2/2EsOsksJrq7lOHxryrVOn1ejG6oAp8ahvOIQD8sw==", + "dev": true + }, + "picocolors": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/picocolors/-/picocolors-1.0.1.tgz", + "integrity": "sha512-anP1Z8qwhkbmu7MFP5iTt+wQKXgwzf7zTyGlcdzabySa9vd0Xt392U0rVmz9poOaBj0uHJKyyo9/upk0HrEQew==", + "dev": true + }, + "picomatch": { + "version": "2.3.1", + "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-2.3.1.tgz", + "integrity": "sha512-JU3teHTNjmE2VCGFzuY8EXzCDVwEqB2a8fsIvwaStHhAWJEeVd1o1QD80CU6+ZdEXXSLbSsuLwJjkCBWqRQUVA==", + "dev": true + }, + "pkg-dir": { + "version": "4.2.0", + "resolved": "https://registry.npmjs.org/pkg-dir/-/pkg-dir-4.2.0.tgz", + "integrity": "sha512-HRDzbaKjC+AOWVXxAU/x54COGeIv9eb+6CkDSQoNTt4XyWoIJvuPsXizxu/Fr23EiekbtZwmh1IcIG/l/a10GQ==", + "dev": true, + "requires": { + "find-up": "^4.0.0" + }, + "dependencies": { + "find-up": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/find-up/-/find-up-4.1.0.tgz", + "integrity": "sha512-PpOwAdQ/YlXQ2vj8a3h8IipDuYRi3wceVQQGYWxNINccq40Anw7BlsEXCMbt1Zt+OLA6Fq9suIpIWD0OsnISlw==", + "dev": true, + "requires": { + "locate-path": "^5.0.0", + "path-exists": "^4.0.0" + } + }, + "locate-path": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-5.0.0.tgz", + "integrity": "sha512-t7hw9pI+WvuwNJXwk5zVHpyhIqzg2qTlklJOf0mVxGSbe3Fp2VieZcduNYjaLDoy6p9uGpQEGWG87WpMKlNq8g==", + "dev": true, + "requires": { + "p-locate": "^4.1.0" + } + }, + "p-limit": { + "version": "2.3.0", + "resolved": "https://registry.npmjs.org/p-limit/-/p-limit-2.3.0.tgz", + "integrity": "sha512-//88mFWSJx8lxCzwdAABTJL2MyWB12+eIY7MDL2SqLmAkeKU9qxRvWuSyTjm3FUmpBEMuFfckAIqEaVGUDxb6w==", + "dev": true, + "requires": { + "p-try": "^2.0.0" + } + }, + "p-locate": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-4.1.0.tgz", + "integrity": "sha512-R79ZZ/0wAxKGu3oYMlz8jy/kbhsNrS7SKZ7PxEHBgJ5+F2mtFW2fK2cOtBh1cHYkQsbzFV7I+EoRKe6Yt0oK7A==", + "dev": true, + "requires": { + "p-limit": "^2.2.0" + } + } + } + }, + "postcss": { + "version": "8.4.41", + "resolved": "https://registry.npmjs.org/postcss/-/postcss-8.4.41.tgz", + "integrity": "sha512-TesUflQ0WKZqAvg52PWL6kHgLKP6xB6heTOdoYM0Wt2UHyxNa4K25EZZMgKns3BH1RLVbZCREPpLY0rhnNoHVQ==", + "dev": true, + "requires": { + "nanoid": "^3.3.7", + "picocolors": "^1.0.1", + "source-map-js": "^1.2.0" + } + }, + "postcss-modules-extract-imports": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/postcss-modules-extract-imports/-/postcss-modules-extract-imports-3.1.0.tgz", + "integrity": "sha512-k3kNe0aNFQDAZGbin48pL2VNidTF0w4/eASDsxlyspobzU3wZQLOGj7L9gfRe0Jo9/4uud09DsjFNH7winGv8Q==", + "dev": true, + "peer": true, + "requires": {} + }, + "postcss-modules-local-by-default": { + "version": "4.0.5", + "resolved": "https://registry.npmjs.org/postcss-modules-local-by-default/-/postcss-modules-local-by-default-4.0.5.tgz", + "integrity": "sha512-6MieY7sIfTK0hYfafw1OMEG+2bg8Q1ocHCpoWLqOKj3JXlKu4G7btkmM/B7lFubYkYWmRSPLZi5chid63ZaZYw==", + "dev": true, + "peer": true, + "requires": { + "icss-utils": "^5.0.0", + "postcss-selector-parser": "^6.0.2", + "postcss-value-parser": "^4.1.0" + } + }, + "postcss-modules-scope": { + "version": "3.2.0", + "resolved": "https://registry.npmjs.org/postcss-modules-scope/-/postcss-modules-scope-3.2.0.tgz", + "integrity": "sha512-oq+g1ssrsZOsx9M96c5w8laRmvEu9C3adDSjI8oTcbfkrTE8hx/zfyobUoWIxaKPO8bt6S62kxpw5GqypEw1QQ==", + "dev": true, + "peer": true, + "requires": { + "postcss-selector-parser": "^6.0.4" + } + }, + "postcss-modules-values": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/postcss-modules-values/-/postcss-modules-values-4.0.0.tgz", + "integrity": "sha512-RDxHkAiEGI78gS2ofyvCsu7iycRv7oqw5xMWn9iMoR0N/7mf9D50ecQqUo5BZ9Zh2vH4bCUR/ktCqbB9m8vJjQ==", + "dev": true, + "peer": true, + "requires": { + "icss-utils": "^5.0.0" + } + }, + "postcss-selector-parser": { + "version": "6.1.2", + "resolved": "https://registry.npmjs.org/postcss-selector-parser/-/postcss-selector-parser-6.1.2.tgz", + "integrity": "sha512-Q8qQfPiZ+THO/3ZrOrO0cJJKfpYCagtMUkXbnEfmgUjwXg6z/WBeOyS9APBBPCTSiDV+s4SwQGu8yFsiMRIudg==", + "dev": true, + "requires": { + "cssesc": "^3.0.0", + "util-deprecate": "^1.0.2" + } + }, + "postcss-value-parser": { + "version": "4.2.0", + "resolved": "https://registry.npmjs.org/postcss-value-parser/-/postcss-value-parser-4.2.0.tgz", + "integrity": "sha512-1NNCs6uurfkVbeXG4S8JFT9t19m45ICnif8zWLd5oPSZ50QnwMfK+H3jv408d4jw/7Bttv5axS5IiHoLaVNHeQ==", + "dev": true, + "peer": true + }, + "prelude-ls": { + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/prelude-ls/-/prelude-ls-1.2.1.tgz", + "integrity": "sha512-vkcDPrRZo1QZLbn5RLGPpg/WmIQ65qoWWhcGKf/b5eplkkarX0m9z8ppCat4mlOqUsWpyNuYgO3VRyrYHSzX5g==", + "dev": true + }, + "prettier": { + "version": "2.8.8", + "resolved": "https://registry.npmjs.org/prettier/-/prettier-2.8.8.tgz", + "integrity": "sha512-tdN8qQGvNjw4CHbY+XXk0JgCXn9QiF21a55rBe5LJAU+kDyC4WQn4+awm2Xfk2lQMk5fKup9XgzTZtGkjBdP9Q==", + "dev": true, + "optional": true + }, + "pseudomap": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/pseudomap/-/pseudomap-1.0.2.tgz", + "integrity": "sha512-b/YwNhb8lk1Zz2+bXXpS/LK9OisiZZ1SNsSLxN1x2OXVEhW2Ckr/7mWE5vrC1ZTiJlD9g19jWszTmJsB+oEpFQ==", + "dev": true + }, + "punycode": { + "version": "2.3.1", + "resolved": "https://registry.npmjs.org/punycode/-/punycode-2.3.1.tgz", + "integrity": "sha512-vYt7UD1U9Wg6138shLtLOvdAu+8DsC/ilFtEVHcH+wydcSpNE20AfSOduf6MkRFahL5FY7X1oU7nKVZFtfq8Fg==", + "dev": true + }, + "queue-microtask": { + "version": "1.2.3", + "resolved": "https://registry.npmjs.org/queue-microtask/-/queue-microtask-1.2.3.tgz", + "integrity": "sha512-NuaNSa6flKT5JaSYQzJok04JzTL1CA6aGhv5rfLW3PgqA+M2ChpZQnAC8h8i4ZFkBS8X5RqkDBHA7r4hej3K9A==", + "dev": true + }, + "randombytes": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/randombytes/-/randombytes-2.1.0.tgz", + "integrity": "sha512-vYl3iOX+4CKUWuxGi9Ukhie6fsqXqS9FE2Zaic4tNFD2N2QQaXOMFbuKK4QmDHC0JO6B1Zp41J0LpT0oR68amQ==", + "dev": true, + "requires": { + "safe-buffer": "^5.1.0" + } + }, + "readdirp": { + "version": "3.6.0", + "resolved": "https://registry.npmjs.org/readdirp/-/readdirp-3.6.0.tgz", + "integrity": "sha512-hOS089on8RduqdbhvQ5Z37A0ESjsqz6qnRcffsMU3495FuTdqSm+7bhJ29JvIOsBDEEnan5DPu9t3To9VRlMzA==", + "dev": true, + "requires": { + "picomatch": "^2.2.1" + } + }, + "rechoir": { + "version": "0.7.1", + "resolved": "https://registry.npmjs.org/rechoir/-/rechoir-0.7.1.tgz", + "integrity": "sha512-/njmZ8s1wVeR6pjTZ+0nCnv8SpZNRMT2D1RLOJQESlYFDBvwpTA4KWJpZ+sBJ4+vhjILRcK7JIFdGCdxEAAitg==", + "dev": true, + "requires": { + "resolve": "^1.9.0" + } + }, + "require-from-string": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/require-from-string/-/require-from-string-2.0.2.tgz", + "integrity": "sha512-Xf0nWe6RseziFMu+Ap9biiUbmplq6S9/p+7w7YXP/JBHhrUDDUhwa+vANyubuqfZWTveU//DYVGsDG7RKL/vEw==", + "dev": true + }, + "resolve": { + "version": "1.22.8", + "resolved": "https://registry.npmjs.org/resolve/-/resolve-1.22.8.tgz", + "integrity": "sha512-oKWePCxqpd6FlLvGV1VU0x7bkPmmCNolxzjMf4NczoDnQcIWrAF+cPtZn5i6n+RfD2d9i0tzpKnG6Yk168yIyw==", + "dev": true, + "requires": { + "is-core-module": "^2.13.0", + "path-parse": "^1.0.7", + "supports-preserve-symlinks-flag": "^1.0.0" + } + }, + "resolve-cwd": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/resolve-cwd/-/resolve-cwd-3.0.0.tgz", + "integrity": "sha512-OrZaX2Mb+rJCpH/6CpSqt9xFVpN++x01XnN2ie9g6P5/3xelLAkXWVADpdz1IHD/KFfEXyE6V0U01OQ3UO2rEg==", + "dev": true, + "requires": { + "resolve-from": "^5.0.0" + }, + "dependencies": { + "resolve-from": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/resolve-from/-/resolve-from-5.0.0.tgz", + "integrity": "sha512-qYg9KP24dD5qka9J47d0aVky0N+b4fTU89LN9iDnjB5waksiC49rvMB0PrUJQGoTmH50XPiqOvAjDfaijGxYZw==", + "dev": true + } + } + }, + "resolve-from": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/resolve-from/-/resolve-from-4.0.0.tgz", + "integrity": "sha512-pb/MYmXstAkysRFx8piNI1tGFNQIFA3vkE3Gq4EuA1dF6gHp/+vgZqsCGJapvy8N3Q+4o7FwvquPJcnZ7RYy4g==", + "dev": true + }, + "reusify": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/reusify/-/reusify-1.0.4.tgz", + "integrity": "sha512-U9nH88a3fc/ekCF1l0/UP1IosiuIjyTh7hBvXVMHYgVcfGvt897Xguj2UOLDeI5BG2m7/uwyaLVT6fbtCwTyzw==", + "dev": true + }, + "rimraf": { + "version": "3.0.2", + "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-3.0.2.tgz", + "integrity": "sha512-JZkJMZkAGFFPP2YqXZXPbMlMBgsxzE8ILs4lMIX/2o0L9UBw9O/Y3o6wFw/i9YLapcUJWwqbi3kdxIPdC62TIA==", + "dev": true, + "requires": { + "glob": "^7.1.3" + } + }, + "run-parallel": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/run-parallel/-/run-parallel-1.2.0.tgz", + "integrity": "sha512-5l4VyZR86LZ/lDxZTR6jqL8AFE2S0IFLMP26AbjsLVADxHdhB/c0GUsH+y39UfCi3dzz8OlQuPmnaJOMoDHQBA==", + "dev": true, + "requires": { + "queue-microtask": "^1.2.2" + } + }, + "safe-buffer": { + "version": "5.2.1", + "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.2.1.tgz", + "integrity": "sha512-rp3So07KcdmmKbGvgaNxQSJr7bGVSVk5S9Eq1F+ppbRo70+YeaDxkw5Dd8NPN+GD6bjnYm2VuPuCXmpuYvmCXQ==", + "dev": true + }, + "sass": { + "version": "1.77.8", + "resolved": "https://registry.npmjs.org/sass/-/sass-1.77.8.tgz", + "integrity": "sha512-4UHg6prsrycW20fqLGPShtEvo/WyHRVRHwOP4DzkUrObWoWI05QBSfzU71TVB7PFaL104TwNaHpjlWXAZbQiNQ==", + "dev": true, + "requires": { + "chokidar": ">=3.0.0 <4.0.0", + "immutable": "^4.0.0", + "source-map-js": ">=0.6.2 <2.0.0" + } + }, + "sass-loader": { + "version": "16.0.0", + "resolved": "https://registry.npmjs.org/sass-loader/-/sass-loader-16.0.0.tgz", + "integrity": "sha512-n13Z+3rU9A177dk4888czcVFiC8CL9dii4qpXWUg3YIIgZEvi9TCFKjOQcbK0kJM7DJu9VucrZFddvNfYCPwtw==", + "dev": true, + "requires": { + "neo-async": "^2.6.2" + } + }, + "schema-utils": { + "version": "2.7.1", + "resolved": "https://registry.npmjs.org/schema-utils/-/schema-utils-2.7.1.tgz", + "integrity": "sha512-SHiNtMOUGWBQJwzISiVYKu82GiV4QYGePp3odlY1tuKO7gPtphAT5R/py0fA6xtbgLL/RvtJZnU9b8s0F1q0Xg==", + "dev": true, + "requires": { + "@types/json-schema": "^7.0.5", + "ajv": "^6.12.4", + "ajv-keywords": "^3.5.2" + } + }, + "semver": { + "version": "6.3.1", + "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.1.tgz", + "integrity": "sha512-BR7VvDCVHO+q2xBEWskxS6DJE1qRnb7DxzUrogb71CWoSficBxYsiAGd+Kl0mmq/MprG9yArRkyrQxTO6XjMzA==", + "dev": true + }, + "serialize-javascript": { + "version": "6.0.2", + "resolved": "https://registry.npmjs.org/serialize-javascript/-/serialize-javascript-6.0.2.tgz", + "integrity": "sha512-Saa1xPByTTq2gdeFZYLLo+RFE35NHZkAbqZeWNd3BpzppeVisAqpDjcp8dyf6uIvEqJRd46jemmyA4iFIeVk8g==", + "dev": true, + "requires": { + "randombytes": "^2.1.0" + } + }, + "shallow-clone": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/shallow-clone/-/shallow-clone-3.0.1.tgz", + "integrity": "sha512-/6KqX+GVUdqPuPPd2LxDDxzX6CAbjJehAAOKlNpqqUpAqPM6HeL8f+o3a+JsyGjn2lv0WY8UsTgUJjU9Ok55NA==", + "dev": true, + "requires": { + "kind-of": "^6.0.2" + } + }, + "shebang-command": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/shebang-command/-/shebang-command-2.0.0.tgz", + "integrity": "sha512-kHxr2zZpYtdmrN1qDjrrX/Z1rR1kG8Dx+gkpK1G4eXmvXswmcE1hTWBWYUzlraYw1/yZp6YuDY77YtvbN0dmDA==", + "dev": true, + "requires": { + "shebang-regex": "^3.0.0" + } + }, + "shebang-regex": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/shebang-regex/-/shebang-regex-3.0.0.tgz", + "integrity": "sha512-7++dFhtcx3353uBaq8DDR4NuxBetBzC7ZQOhmTQInHEd6bSrXdiEyzCvG07Z44UYdLShWUyXt5M/yhz8ekcb1A==", + "dev": true + }, + "source-map": { + "version": "0.6.1", + "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz", + "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", + "dev": true + }, + "source-map-js": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/source-map-js/-/source-map-js-1.2.0.tgz", + "integrity": "sha512-itJW8lvSA0TXEphiRoawsCksnlf8SyvmFzIhltqAHluXd88pkCd+cXJVHTDwdCr0IzwptSm035IHQktUu1QUMg==", + "dev": true + }, + "source-map-support": { + "version": "0.5.21", + "resolved": "https://registry.npmjs.org/source-map-support/-/source-map-support-0.5.21.tgz", + "integrity": "sha512-uBHU3L3czsIyYXKX88fdrGovxdSCoTGDRZ6SYXtSRxLZUzHg5P/66Ht6uoUlHu9EZod+inXhKo3qQgwXUT/y1w==", + "dev": true, + "requires": { + "buffer-from": "^1.0.0", + "source-map": "^0.6.0" + } + }, + "strip-ansi": { + "version": "6.0.1", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.1.tgz", + "integrity": "sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==", + "dev": true, + "requires": { + "ansi-regex": "^5.0.1" + } + }, + "strip-json-comments": { + "version": "3.1.1", + "resolved": "https://registry.npmjs.org/strip-json-comments/-/strip-json-comments-3.1.1.tgz", + "integrity": "sha512-6fPc+R4ihwqP6N/aIv2f1gMH8lOVtWQHoqC4yK6oSDVVocumAsfCqjkXnqiYMhmMwS/mEHLp7Vehlt3ql6lEig==", + "dev": true + }, + "supports-color": { + "version": "5.5.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-5.5.0.tgz", + "integrity": "sha512-QjVjwdXIt408MIiAqCX4oUKsgU2EqAGzs2Ppkm4aQYbjm+ZEWEcW4SfFNTr4uMNZma0ey4f5lgLrkB0aX0QMow==", + "dev": true, + "peer": true, + "requires": { + "has-flag": "^3.0.0" + } + }, + "supports-preserve-symlinks-flag": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/supports-preserve-symlinks-flag/-/supports-preserve-symlinks-flag-1.0.0.tgz", + "integrity": "sha512-ot0WnXS9fgdkgIcePe6RHNk1WA8+muPa6cSjeR3V8K27q9BB1rTE3R1p7Hv0z1ZyAc8s6Vvv8DIyWf681MAt0w==", + "dev": true + }, + "tapable": { + "version": "2.2.1", + "resolved": "https://registry.npmjs.org/tapable/-/tapable-2.2.1.tgz", + "integrity": "sha512-GNzQvQTOIP6RyTfE2Qxb8ZVlNmw0n88vp1szwWRimP02mnTsx3Wtn5qRdqY9w2XduFNUgvOwhNnQsjwCp+kqaQ==", + "dev": true + }, + "terser": { + "version": "5.31.6", + "resolved": "https://registry.npmjs.org/terser/-/terser-5.31.6.tgz", + "integrity": "sha512-PQ4DAriWzKj+qgehQ7LK5bQqCFNMmlhjR2PFFLuqGCpuCAauxemVBWwWOxo3UIwWQx8+Pr61Df++r76wDmkQBg==", + "dev": true, + "requires": { + "@jridgewell/source-map": "^0.3.3", + "acorn": "^8.8.2", + "commander": "^2.20.0", + "source-map-support": "~0.5.20" + } + }, + "terser-webpack-plugin": { + "version": "5.3.10", + "resolved": "https://registry.npmjs.org/terser-webpack-plugin/-/terser-webpack-plugin-5.3.10.tgz", + "integrity": "sha512-BKFPWlPDndPs+NGGCr1U59t0XScL5317Y0UReNrHaw9/FwhPENlq6bfgs+4yPfyP51vqC1bQ4rp1EfXW5ZSH9w==", + "dev": true, + "requires": { + "@jridgewell/trace-mapping": "^0.3.20", + "jest-worker": "^27.4.5", + "schema-utils": "^3.1.1", + "serialize-javascript": "^6.0.1", + "terser": "^5.26.0" + }, + "dependencies": { + "has-flag": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", + "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", + "dev": true + }, + "jest-worker": { + "version": "27.5.1", + "resolved": "https://registry.npmjs.org/jest-worker/-/jest-worker-27.5.1.tgz", + "integrity": "sha512-7vuh85V5cdDofPyxn58nrPjBktZo0u9x1g8WtjQol+jZDaE+fhN+cIvTj11GndBnMnyfrUOG1sZQxCdjKh+DKg==", + "dev": true, + "requires": { + "@types/node": "*", + "merge-stream": "^2.0.0", + "supports-color": "^8.0.0" + } + }, + "schema-utils": { + "version": "3.3.0", + "resolved": "https://registry.npmjs.org/schema-utils/-/schema-utils-3.3.0.tgz", + "integrity": "sha512-pN/yOAvcC+5rQ5nERGuwrjLlYvLTbCibnZ1I7B1LaiAz9BRBlE9GMgE/eqV30P7aJQUf7Ddimy/RsbYO/GrVGg==", + "dev": true, + "requires": { + "@types/json-schema": "^7.0.8", + "ajv": "^6.12.5", + "ajv-keywords": "^3.5.2" + } + }, + "supports-color": { + "version": "8.1.1", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-8.1.1.tgz", + "integrity": "sha512-MpUEN2OodtUzxvKQl72cUF7RQ5EiHsGvSsVG0ia9c5RbWGL2CI4C7EpPS8UTBIplnlzZiNuV56w+FuNxy3ty2Q==", + "dev": true, + "requires": { + "has-flag": "^4.0.0" + } + } + } + }, + "text-table": { + "version": "0.2.0", + "resolved": "https://registry.npmjs.org/text-table/-/text-table-0.2.0.tgz", + "integrity": "sha512-N+8UisAXDGk8PFXP4HAzVR9nbfmVJ3zYLAWiTIoqC5v5isinhr+r5uaO8+7r3BMfuNIufIsA7RdpVgacC2cSpw==", + "dev": true + }, + "to-fast-properties": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/to-fast-properties/-/to-fast-properties-2.0.0.tgz", + "integrity": "sha512-/OaKK0xYrs3DmxRYqL/yDc+FxFUVYhDlXMhRmv3z915w2HF1tnN1omB354j8VUGO/hbRzyD6Y3sA7v7GS/ceog==", + "dev": true + }, + "to-regex-range": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/to-regex-range/-/to-regex-range-5.0.1.tgz", + "integrity": "sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ==", + "dev": true, + "requires": { + "is-number": "^7.0.0" + } + }, + "type-check": { + "version": "0.4.0", + "resolved": "https://registry.npmjs.org/type-check/-/type-check-0.4.0.tgz", + "integrity": "sha512-XleUoc9uwGXqjWwXaUTZAmzMcFZ5858QA2vvx1Ur5xIcixXIP+8LnFDgRplU30us6teqdlskFfu+ae4K79Ooew==", + "dev": true, + "requires": { + "prelude-ls": "^1.2.1" + } + }, + "type-fest": { + "version": "0.20.2", + "resolved": "https://registry.npmjs.org/type-fest/-/type-fest-0.20.2.tgz", + "integrity": "sha512-Ne+eE4r0/iWnpAxD852z3A+N0Bt5RN//NjJwRd2VFHEmrywxf5vsZlh4R6lixl6B+wz/8d+maTSAkN1FIkI3LQ==", + "dev": true + }, + "undici-types": { + "version": "6.18.2", + "resolved": "https://registry.npmjs.org/undici-types/-/undici-types-6.18.2.tgz", + "integrity": "sha512-5ruQbENj95yDYJNS3TvcaxPMshV7aizdv/hWYjGIKoANWKjhWNBsr2YEuYZKodQulB1b8l7ILOuDQep3afowQQ==", + "dev": true + }, + "update-browserslist-db": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/update-browserslist-db/-/update-browserslist-db-1.1.0.tgz", + "integrity": "sha512-EdRAaAyk2cUE1wOf2DkEhzxqOQvFOoRJFNS6NeyJ01Gp2beMRpBAINjM2iDXE3KCuKhwnvHIQCJm6ThL2Z+HzQ==", + "dev": true, + "requires": { + "escalade": "^3.1.2", + "picocolors": "^1.0.1" + } + }, + "uri-js": { + "version": "4.4.1", + "resolved": "https://registry.npmjs.org/uri-js/-/uri-js-4.4.1.tgz", + "integrity": "sha512-7rKUyy33Q1yc98pQ1DAmLtwX109F7TIfWlW1Ydo8Wl1ii1SeHieeh0HHfPeL2fMXK6z0s8ecKs9frCuLJvndBg==", + "dev": true, + "requires": { + "punycode": "^2.1.0" + } + }, + "util-deprecate": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/util-deprecate/-/util-deprecate-1.0.2.tgz", + "integrity": "sha512-EPD5q1uXyFxJpCrLnCc1nHnq3gOa6DZBocAIiI2TaSCA7VCJ1UJDMagCzIkXNsUYfD1daK//LTEQ8xiIbrHtcw==", + "dev": true + }, + "vue": { + "version": "2.7.16", + "resolved": "https://registry.npmjs.org/vue/-/vue-2.7.16.tgz", + "integrity": "sha512-4gCtFXaAA3zYZdTp5s4Hl2sozuySsgz4jy1EnpBHNfpMa9dK1ZCG7viqBPCwXtmgc8nHqUsAu3G4gtmXkkY3Sw==", + "dev": true, + "requires": { + "@vue/compiler-sfc": "2.7.16", + "csstype": "^3.1.0" + } + }, + "vue-eslint-parser": { + "version": "8.3.0", + "resolved": "https://registry.npmjs.org/vue-eslint-parser/-/vue-eslint-parser-8.3.0.tgz", + "integrity": "sha512-dzHGG3+sYwSf6zFBa0Gi9ZDshD7+ad14DGOdTLjruRVgZXe2J+DcZ9iUhyR48z5g1PqRa20yt3Njna/veLJL/g==", + "dev": true, + "requires": { + "debug": "^4.3.2", + "eslint-scope": "^7.0.0", + "eslint-visitor-keys": "^3.1.0", + "espree": "^9.0.0", + "esquery": "^1.4.0", + "lodash": "^4.17.21", + "semver": "^7.3.5" + }, + "dependencies": { + "semver": { + "version": "7.6.3", + "resolved": "https://registry.npmjs.org/semver/-/semver-7.6.3.tgz", + "integrity": "sha512-oVekP1cKtI+CTDvHWYFUcMtsK/00wmAEfyqKfNdARm8u1wNVhSgaX7A8d4UuIlUI5e84iEwOhs7ZPYRmzU9U6A==", + "dev": true + } + } + }, + "vue-hot-reload-api": { + "version": "2.3.4", + "resolved": "https://registry.npmjs.org/vue-hot-reload-api/-/vue-hot-reload-api-2.3.4.tgz", + "integrity": "sha512-BXq3jwIagosjgNVae6tkHzzIk6a8MHFtzAdwhnV5VlvPTFxDCvIttgSiHWjdGoTJvXtmRu5HacExfdarRcFhog==", + "dev": true + }, + "vue-loader": { + "version": "15.11.1", + "resolved": "https://registry.npmjs.org/vue-loader/-/vue-loader-15.11.1.tgz", + "integrity": "sha512-0iw4VchYLePqJfJu9s62ACWUXeSqM30SQqlIftbYWM3C+jpPcEHKSPUZBLjSF9au4HTHQ/naF6OGnO3Q/qGR3Q==", + "dev": true, + "requires": { + "@vue/component-compiler-utils": "^3.1.0", + "hash-sum": "^1.0.2", + "loader-utils": "^1.1.0", + "vue-hot-reload-api": "^2.3.0", + "vue-style-loader": "^4.1.0" + }, + "dependencies": { + "json5": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/json5/-/json5-1.0.2.tgz", + "integrity": "sha512-g1MWMLBiz8FKi1e4w0UyVL3w+iJceWAFBAaBnnGKOpNa5f8TLktkbre1+s6oICydWAm+HRUGTmI+//xv2hvXYA==", + "dev": true, + "requires": { + "minimist": "^1.2.0" + } + }, + "loader-utils": { + "version": "1.4.2", + "resolved": "https://registry.npmjs.org/loader-utils/-/loader-utils-1.4.2.tgz", + "integrity": "sha512-I5d00Pd/jwMD2QCduo657+YM/6L3KZu++pmX9VFncxaxvHcru9jx1lBaFft+r4Mt2jK0Yhp41XlRAihzPxHNCg==", + "dev": true, + "requires": { + "big.js": "^5.2.2", + "emojis-list": "^3.0.0", + "json5": "^1.0.1" + } + } + } + }, + "vue-style-loader": { + "version": "4.1.3", + "resolved": "https://registry.npmjs.org/vue-style-loader/-/vue-style-loader-4.1.3.tgz", + "integrity": "sha512-sFuh0xfbtpRlKfm39ss/ikqs9AbKCoXZBpHeVZ8Tx650o0k0q/YCM7FRvigtxpACezfq6af+a7JeqVTWvncqDg==", + "dev": true, + "requires": { + "hash-sum": "^1.0.2", + "loader-utils": "^1.0.2" + }, + "dependencies": { + "json5": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/json5/-/json5-1.0.2.tgz", + "integrity": "sha512-g1MWMLBiz8FKi1e4w0UyVL3w+iJceWAFBAaBnnGKOpNa5f8TLktkbre1+s6oICydWAm+HRUGTmI+//xv2hvXYA==", + "dev": true, + "requires": { + "minimist": "^1.2.0" + } + }, + "loader-utils": { + "version": "1.4.2", + "resolved": "https://registry.npmjs.org/loader-utils/-/loader-utils-1.4.2.tgz", + "integrity": "sha512-I5d00Pd/jwMD2QCduo657+YM/6L3KZu++pmX9VFncxaxvHcru9jx1lBaFft+r4Mt2jK0Yhp41XlRAihzPxHNCg==", + "dev": true, + "requires": { + "big.js": "^5.2.2", + "emojis-list": "^3.0.0", + "json5": "^1.0.1" + } + } + } + }, + "vue-template-compiler": { + "version": "2.7.16", + "resolved": "https://registry.npmjs.org/vue-template-compiler/-/vue-template-compiler-2.7.16.tgz", + "integrity": "sha512-AYbUWAJHLGGQM7+cNTELw+KsOG9nl2CnSv467WobS5Cv9uk3wFcnr1Etsz2sEIHEZvw1U+o9mRlEO6QbZvUPGQ==", + "dev": true, + "requires": { + "de-indent": "^1.0.2", + "he": "^1.2.0" + } + }, + "vue-template-es2015-compiler": { + "version": "1.9.1", + "resolved": "https://registry.npmjs.org/vue-template-es2015-compiler/-/vue-template-es2015-compiler-1.9.1.tgz", + "integrity": "sha512-4gDntzrifFnCEvyoO8PqyJDmguXgVPxKiIxrBKjIowvL9l+N66196+72XVYR8BBf1Uv1Fgt3bGevJ+sEmxfZzw==", + "dev": true + }, + "watchpack": { + "version": "2.4.2", + "resolved": "https://registry.npmjs.org/watchpack/-/watchpack-2.4.2.tgz", + "integrity": "sha512-TnbFSbcOCcDgjZ4piURLCbJ3nJhznVh9kw6F6iokjiFPl8ONxe9A6nMDVXDiNbrSfLILs6vB07F7wLBrwPYzJw==", + "dev": true, + "requires": { + "glob-to-regexp": "^0.4.1", + "graceful-fs": "^4.1.2" + } + }, + "webpack": { + "version": "5.94.0", + "resolved": "https://registry.npmjs.org/webpack/-/webpack-5.94.0.tgz", + "integrity": "sha512-KcsGn50VT+06JH/iunZJedYGUJS5FGjow8wb9c0v5n1Om8O1g4L6LjtfxwlXIATopoQu+vOXXa7gYisWxCoPyg==", + "dev": true, + "requires": { + "@types/estree": "^1.0.5", + "@webassemblyjs/ast": "^1.12.1", + "@webassemblyjs/wasm-edit": "^1.12.1", + "@webassemblyjs/wasm-parser": "^1.12.1", + "acorn": "^8.7.1", + "acorn-import-attributes": "^1.9.5", + "browserslist": "^4.21.10", + "chrome-trace-event": "^1.0.2", + "enhanced-resolve": "^5.17.1", + "es-module-lexer": "^1.2.1", + "eslint-scope": "5.1.1", + "events": "^3.2.0", + "glob-to-regexp": "^0.4.1", + "graceful-fs": "^4.2.11", + "json-parse-even-better-errors": "^2.3.1", + "loader-runner": "^4.2.0", + "mime-types": "^2.1.27", + "neo-async": "^2.6.2", + "schema-utils": "^3.2.0", + "tapable": "^2.1.1", + "terser-webpack-plugin": "^5.3.10", + "watchpack": "^2.4.1", + "webpack-sources": "^3.2.3" + }, + "dependencies": { + "eslint-scope": { + "version": "5.1.1", + "resolved": "https://registry.npmjs.org/eslint-scope/-/eslint-scope-5.1.1.tgz", + "integrity": "sha512-2NxwbF/hZ0KpepYN0cNbo+FN6XoK7GaHlQhgx/hIZl6Va0bF45RQOOwhLIy8lQDbuCiadSLCBnH2CFYquit5bw==", + "dev": true, + "requires": { + "esrecurse": "^4.3.0", + "estraverse": "^4.1.1" + } + }, + "estraverse": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/estraverse/-/estraverse-4.3.0.tgz", + "integrity": "sha512-39nnKffWz8xN1BU/2c79n9nB9HDzo0niYUqx6xyqUnyoAnQyyWpOTdZEeiCch8BBu515t4wp9ZmgVfVhn9EBpw==", + "dev": true + }, + "schema-utils": { + "version": "3.3.0", + "resolved": "https://registry.npmjs.org/schema-utils/-/schema-utils-3.3.0.tgz", + "integrity": "sha512-pN/yOAvcC+5rQ5nERGuwrjLlYvLTbCibnZ1I7B1LaiAz9BRBlE9GMgE/eqV30P7aJQUf7Ddimy/RsbYO/GrVGg==", + "dev": true, + "requires": { + "@types/json-schema": "^7.0.8", + "ajv": "^6.12.5", + "ajv-keywords": "^3.5.2" + } + } + } + }, + "webpack-cli": { + "version": "4.10.0", + "resolved": "https://registry.npmjs.org/webpack-cli/-/webpack-cli-4.10.0.tgz", + "integrity": "sha512-NLhDfH/h4O6UOy+0LSso42xvYypClINuMNBVVzX4vX98TmTaTUxwRbXdhucbFMd2qLaCTcLq/PdYrvi8onw90w==", + "dev": true, + "requires": { + "@discoveryjs/json-ext": "^0.5.0", + "@webpack-cli/configtest": "^1.2.0", + "@webpack-cli/info": "^1.5.0", + "@webpack-cli/serve": "^1.7.0", + "colorette": "^2.0.14", + "commander": "^7.0.0", + "cross-spawn": "^7.0.3", + "fastest-levenshtein": "^1.0.12", + "import-local": "^3.0.2", + "interpret": "^2.2.0", + "rechoir": "^0.7.0", + "webpack-merge": "^5.7.3" + }, + "dependencies": { + "commander": { + "version": "7.2.0", + "resolved": "https://registry.npmjs.org/commander/-/commander-7.2.0.tgz", + "integrity": "sha512-QrWXB+ZQSVPmIWIhtEO9H+gwHaMGYiF5ChvoJ+K9ZGHG/sVsa6yiesAD1GC/x46sET00Xlwo1u49RVVVzvcSkw==", + "dev": true + } + } + }, + "webpack-merge": { + "version": "5.10.0", + "resolved": "https://registry.npmjs.org/webpack-merge/-/webpack-merge-5.10.0.tgz", + "integrity": "sha512-+4zXKdx7UnO+1jaN4l2lHVD+mFvnlZQP/6ljaJVb4SZiwIKeUnrT5l0gkT8z+n4hKpC+jpOv6O9R+gLtag7pSA==", + "dev": true, + "requires": { + "clone-deep": "^4.0.1", + "flat": "^5.0.2", + "wildcard": "^2.0.0" + } + }, + "webpack-sources": { + "version": "3.2.3", + "resolved": "https://registry.npmjs.org/webpack-sources/-/webpack-sources-3.2.3.tgz", + "integrity": "sha512-/DyMEOrDgLKKIG0fmvtz+4dUX/3Ghozwgm6iPp8KRhvn+eQf9+Q7GWxVNMk3+uCPWfdXYC4ExGBckIXdFEfH1w==", + "dev": true + }, + "which": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/which/-/which-2.0.2.tgz", + "integrity": "sha512-BLI3Tl1TW3Pvl70l3yq3Y64i+awpwXqsGBYWkkqMtnbXgrMD+yj7rhW0kuEDxzJaYXGjEW5ogapKNMEKNMjibA==", + "dev": true, + "requires": { + "isexe": "^2.0.0" + } + }, + "wildcard": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/wildcard/-/wildcard-2.0.1.tgz", + "integrity": "sha512-CC1bOL87PIWSBhDcTrdeLo6eGT7mCFtrg0uIJtqJUFyK+eJnzl8A1niH56uu7KMa5XFrtiV+AQuHO3n7DsHnLQ==", + "dev": true + }, + "word-wrap": { + "version": "1.2.5", + "resolved": "https://registry.npmjs.org/word-wrap/-/word-wrap-1.2.5.tgz", + "integrity": "sha512-BN22B5eaMMI9UMtjrGd5g5eCYPpCPDUy0FJXbYsaT5zYxjFOckS53SQDE3pWkVoWpHXVb3BrYcEN4Twa55B5cA==", + "dev": true + }, + "wrappy": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/wrappy/-/wrappy-1.0.2.tgz", + "integrity": "sha512-l4Sp/DRseor9wL6EvV2+TuQn63dMkPjZ/sp9XkghTEbV9KlPS1xUsZ3u7/IQO4wxtcFB4bgpQPRcR3QCvezPcQ==", + "dev": true + }, + "yallist": { + "version": "3.1.1", + "resolved": "https://registry.npmjs.org/yallist/-/yallist-3.1.1.tgz", + "integrity": "sha512-a4UGQaWPH59mOXUYnAG2ewncQS4i4F43Tv3JoAM+s2VDAmS9NsK8GpDMLrCHPksFT7h3K6TOoUNn2pb7RoXx4g==", + "dev": true, + "peer": true + }, + "yocto-queue": { + "version": "0.1.0", + "resolved": "https://registry.npmjs.org/yocto-queue/-/yocto-queue-0.1.0.tgz", + "integrity": "sha512-rVksvsnNCdJ/ohGc6xgPwyN8eheCxsiLM8mxuE/t/mOVqJewPuO1miLpTHQiRgTKCLexL4MeAFVagts7HmNZ2Q==", + "dev": true + } } } diff --git a/webapp/package.json b/webapp/package.json index 4563ae02caf..10db92123f2 100644 --- a/webapp/package.json +++ b/webapp/package.json @@ -1,6 +1,6 @@ { "name": "meeds-social", - "version": "6.2.0", + "version": "7.0.0", "description": "Meeds Space poetlets", "repository": { "type": "git", diff --git a/webapp/src/main/java/io/meeds/social/SocialApplication.java b/webapp/src/main/java/io/meeds/social/SocialApplication.java new file mode 100644 index 00000000000..23d7d67737b --- /dev/null +++ b/webapp/src/main/java/io/meeds/social/SocialApplication.java @@ -0,0 +1,42 @@ +/* + * This file is part of the Meeds project (https://meeds.io/). + * Copyright (C) 2020 - 2022 Meeds Association contact@meeds.io + * This program is free software; you can redistribute it and/or + * modify it under the terms of the GNU Lesser General Public + * License as published by the Free Software Foundation; either + * version 3 of the License, or (at your option) any later version. + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU + * Lesser General Public License for more details. + * You should have received a copy of the GNU Lesser General Public License + * along with this program; if not, write to the Free Software Foundation, + * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. + */ +package io.meeds.social; + +import org.springframework.boot.autoconfigure.SpringBootApplication; +import org.springframework.boot.autoconfigure.liquibase.LiquibaseAutoConfiguration; +import org.springframework.context.annotation.PropertySource; +import org.springframework.data.jpa.repository.config.EnableJpaRepositories; + +import io.meeds.spring.AvailableIntegration; +import io.meeds.spring.kernel.PortalApplicationContextInitializer; + +@SpringBootApplication(scanBasePackages = { + SocialApplication.MODULE_NAME, + AvailableIntegration.KERNEL_MODULE, + AvailableIntegration.JPA_MODULE, + AvailableIntegration.WEB_MODULE, +}, exclude = { + LiquibaseAutoConfiguration.class, +}) +@EnableJpaRepositories(basePackages = SocialApplication.MODULE_NAME) +@PropertySource("classpath:application.properties") +@PropertySource("classpath:application-common.properties") +@PropertySource("classpath:social.properties") +public class SocialApplication extends PortalApplicationContextInitializer { + + public static final String MODULE_NAME = "io.meeds.social"; + +} diff --git a/webapp/src/main/resources/locale/portlet/SpaceTemplatesManagement_en.properties b/webapp/src/main/resources/locale/portlet/SpaceTemplatesManagement_en.properties new file mode 100644 index 00000000000..265cbd60032 --- /dev/null +++ b/webapp/src/main/resources/locale/portlet/SpaceTemplatesManagement_en.properties @@ -0,0 +1,33 @@ +spaceTemplates.title=Space Templates +spaceTemplates.add=Add +spaceTemplates.filter.placeholder=Filter by name, description +spaceTemplates.label.name=Name +spaceTemplates.label.description=Description +spaceTemplates.label.permissions=Creation +spaceTemplates.label.spacesCount=Spaces +spaceTemplates.label.status=Status +spaceTemplates.label.actions=Actions +spaceTemplates.menu.open=Open Menu +spaceTemplate.label.preview=Preview of {0} template +spaceTemplate.status.update.success=Template status successfully updated +spaceTemplate.status.update.error=An unknown error occurred while updating template status. Please contact the administrator or try agan later. +spaceTemplate.layout.update.success=Template layout successfully updated +spaceTemplate.layout.update.error=An unknown error occurred while updating template layout. Please contact the administrator or try agan later. +spaceTemplate.delete.success=Template successfully deleted +spaceTemplate.delete.error=An unknown error occurred while deleting template. Please contact the administrator or try agan later. +spaceTemplate.label.delete=Delete +spaceTemplate.label.editLayout=Edit Layout +spaceTemplate.label.editProperties=Edit Properties +spaceTemplate.label.duplicate=Duplicate +spaceTemplate.label.templateMenu={0} +spaceTemplate.label.closeMenu=Close Menu +spaceTemplate.label.close=Close +spaceTemplate.label.openIllustrationPreview=Open illustration Preview +spaceTemplate.label.enableTemplate=Enable Template +spaceTemplate.label.disableTemplate=Disable Template +spaceTemplate.label.system.noDelete=This product template cannot be deleted +spaceTemplate.label.system.noEditLayout=This product template's layout cannot be updated +spaceTemplate.label.confirmDeleteTitle=Delete space template? +spaceTemplate.label.confirmDeleteMessage=Would you like to delete space template: {0} +spaceTemplate.label.confirm=Confirm +spaceTemplate.label.cancel=Cancel diff --git a/webapp/src/main/resources/social.properties b/webapp/src/main/resources/social.properties new file mode 100644 index 00000000000..bdf205b419e --- /dev/null +++ b/webapp/src/main/resources/social.properties @@ -0,0 +1,18 @@ +# +# This file is part of the Meeds project (https://meeds.io/). +# +# Copyright (C) 2024 Meeds Association contact@meeds.io +# +# This program is free software; you can redistribute it and/or +# modify it under the terms of the GNU Lesser General Public +# License as published by the Free Software Foundation; either +# version 3 of the License, or (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU +# Lesser General Public License for more details. +# +# You should have received a copy of the GNU Lesser General Public License +# along with this program; if not, write to the Free Software Foundation, +# Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. diff --git a/webapp/src/main/webapp/WEB-INF/gatein-resources.xml b/webapp/src/main/webapp/WEB-INF/gatein-resources.xml index ebd4521927e..65e13dca69e 100644 --- a/webapp/src/main/webapp/WEB-INF/gatein-resources.xml +++ b/webapp/src/main/webapp/WEB-INF/gatein-resources.xml @@ -2046,6 +2046,38 @@ + + SpaceTemplateManagement + + spaceTemplateManagement + + + vue + + + vuetify + + + translationField + + + commonVueComponents + + + applicationToolbarComponent + + + eXoVueI18n + + + extensionRegistry + + + + siteDetailsComponent vueGRP diff --git a/webapp/src/main/webapp/WEB-INF/portlet.xml b/webapp/src/main/webapp/WEB-INF/portlet.xml index 6dfc8971117..81f620b384b 100644 --- a/webapp/src/main/webapp/WEB-INF/portlet.xml +++ b/webapp/src/main/webapp/WEB-INF/portlet.xml @@ -1087,4 +1087,26 @@ Organizational Chart + + + SpaceTemplateManagement + Space Template Management + org.exoplatform.commons.api.portlet.GenericDispatchedViewPortlet + + portlet-view-dispatched-file-path + /html/spaceTemplateManagement.html + + -1 + PUBLIC + + text/html + + en + locale.portlet.SpaceTemplateManagement + + Space Template Management + Space Template Management + + + diff --git a/webapp/src/main/webapp/html/spaceTemplateManagement.html b/webapp/src/main/webapp/html/spaceTemplateManagement.html new file mode 100644 index 00000000000..2cbaacc41a2 --- /dev/null +++ b/webapp/src/main/webapp/html/spaceTemplateManagement.html @@ -0,0 +1,9 @@ +
+
+ +
+
diff --git a/webapp/src/main/webapp/vue-apps/common/js/SpaceService.js b/webapp/src/main/webapp/vue-apps/common/js/SpaceService.js index f33d1b26614..896005926d5 100644 --- a/webapp/src/main/webapp/vue-apps/common/js/SpaceService.js +++ b/webapp/src/main/webapp/vue-apps/common/js/SpaceService.js @@ -13,6 +13,19 @@ export function getSpaceTemplates() { }); } +export function getSpacesCountByTemplates() { + return fetch(`${eXo.env.portal.context}/${eXo.env.portal.rest}/v1/social/spaces/countByTemplate`, { + method: 'GET', + credentials: 'include', + }).then(resp => { + if (resp?.ok) { + return resp.json(); + } else { + throw new Error('Response code indicates a server error', resp); + } + }); +} + export function findSpaceExternalInvitationsBySpaceId(spaceId) { return fetch(`${eXo.env.portal.context}/${eXo.env.portal.rest}/v1/social/spaces/${spaceId}/externalInvitations`, { method: 'GET', diff --git a/webapp/src/main/webapp/vue-apps/space-templates-management/components/SpaceTemplatesManagement.vue b/webapp/src/main/webapp/vue-apps/space-templates-management/components/SpaceTemplatesManagement.vue new file mode 100644 index 00000000000..354d0237131 --- /dev/null +++ b/webapp/src/main/webapp/vue-apps/space-templates-management/components/SpaceTemplatesManagement.vue @@ -0,0 +1,43 @@ + + + diff --git a/webapp/src/main/webapp/vue-apps/space-templates-management/components/header/Toolbar.vue b/webapp/src/main/webapp/vue-apps/space-templates-management/components/header/Toolbar.vue new file mode 100644 index 00000000000..61e6456c697 --- /dev/null +++ b/webapp/src/main/webapp/vue-apps/space-templates-management/components/header/Toolbar.vue @@ -0,0 +1,50 @@ + + + diff --git a/webapp/src/main/webapp/vue-apps/space-templates-management/components/list/SpaceTemplateItem.vue b/webapp/src/main/webapp/vue-apps/space-templates-management/components/list/SpaceTemplateItem.vue new file mode 100644 index 00000000000..c0553da532b --- /dev/null +++ b/webapp/src/main/webapp/vue-apps/space-templates-management/components/list/SpaceTemplateItem.vue @@ -0,0 +1,103 @@ + + \ No newline at end of file diff --git a/webapp/src/main/webapp/vue-apps/space-templates-management/components/list/SpaceTemplateItemMenu.vue b/webapp/src/main/webapp/vue-apps/space-templates-management/components/list/SpaceTemplateItemMenu.vue new file mode 100644 index 00000000000..568e22c8d86 --- /dev/null +++ b/webapp/src/main/webapp/vue-apps/space-templates-management/components/list/SpaceTemplateItemMenu.vue @@ -0,0 +1,158 @@ + + \ No newline at end of file diff --git a/webapp/src/main/webapp/vue-apps/space-templates-management/components/list/SpaceTemplates.vue b/webapp/src/main/webapp/vue-apps/space-templates-management/components/list/SpaceTemplates.vue new file mode 100644 index 00000000000..d2c76ffc3d4 --- /dev/null +++ b/webapp/src/main/webapp/vue-apps/space-templates-management/components/list/SpaceTemplates.vue @@ -0,0 +1,169 @@ + + diff --git a/webapp/src/main/webapp/vue-apps/space-templates-management/initComponents.js b/webapp/src/main/webapp/vue-apps/space-templates-management/initComponents.js new file mode 100644 index 00000000000..b8ca05ff2f5 --- /dev/null +++ b/webapp/src/main/webapp/vue-apps/space-templates-management/initComponents.js @@ -0,0 +1,37 @@ +/* + * This file is part of the Meeds project (https://meeds.io/). + * + * Copyright (C) 2020 - 2024 Meeds Association contact@meeds.io + * + * This program is free software; you can redistribute it and/or + * modify it under the terms of the GNU Lesser General Public + * License as published by the Free Software Foundation; either + * version 3 of the License, or (at your option) any later version. + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU + * Lesser General Public License for more details. + * + * You should have received a copy of the GNU Lesser General Public License + * along with this program; if not, write to the Free Software Foundation, + * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. + */ +import SpaceTemplatesManagement from './components/SpaceTemplatesManagement.vue'; + +import Toolbar from './components/header/Toolbar.vue'; +import SpaceTemplates from './components/list/SpaceTemplates.vue'; + +import SpaceTemplateItem from './components/list/SpaceTemplateItem.vue'; +import SpaceTemplateItemMenu from './components/list/SpaceTemplateItemMenu.vue'; + +const components = { + 'space-templates-management': SpaceTemplatesManagement, + 'space-templates-management-toolbar': Toolbar, + 'space-templates-management-list': SpaceTemplates, + 'space-templates-management-item': SpaceTemplateItem, + 'space-templates-management-item-menu': SpaceTemplateItemMenu, +}; + +for (const key in components) { + Vue.component(key, components[key]); +} diff --git a/webapp/src/main/webapp/vue-apps/space-templates-management/js/SpaceTemplateService.js b/webapp/src/main/webapp/vue-apps/space-templates-management/js/SpaceTemplateService.js new file mode 100644 index 00000000000..2da60797bac --- /dev/null +++ b/webapp/src/main/webapp/vue-apps/space-templates-management/js/SpaceTemplateService.js @@ -0,0 +1,87 @@ +/* + * This file is part of the Meeds project (https://meeds.io/). + * + * Copyright (C) 2020 - 2024 Meeds Association contact@meeds.io + * + * This program is free software; you can redistribute it and/or + * modify it under the terms of the GNU Lesser General Public + * License as published by the Free Software Foundation; either + * version 3 of the License, or (at your option) any later version. + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU + * Lesser General Public License for more details. + * + * You should have received a copy of the GNU Lesser General Public License + * along with this program; if not, write to the Free Software Foundation, + * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. + */ + +export function getSpaceTemplates(includeDisabled) { + return fetch(`/social/rest/space/templates?includeDisabled=${includeDisabled || false}`, { + method: 'GET', + credentials: 'include', + }).then(resp => { + if (!resp?.ok) { + throw new Error('Error when retrieving space templates'); + } else { + return resp.json(); + } + }); +} + +export function getSpaceTemplate(id) { + return fetch(`/social/rest/space/templates/${id}`, { + method: 'GET', + credentials: 'include', + }).then(resp => { + if (!resp?.ok) { + throw new Error('Error when retrieving space template'); + } else { + return resp.json(); + } + }); +} + +export function createSpaceTemplate(spaceTemplate) { + return fetch('/social/rest/space/templates', { + credentials: 'include', + method: 'POST', + headers: { + 'Content-Type': 'application/json', + }, + body: JSON.stringify(spaceTemplate), + }).then((resp) => { + if (resp?.ok) { + return resp.json(); + } else { + throw new Error('Error when creating space template'); + } + }); +} + +export function updateSpaceTemplate(spaceTemplate) { + return fetch(`/social/rest/space/templates/${spaceTemplate.id}`, { + credentials: 'include', + method: 'PUT', + headers: { + 'Content-Type': 'application/json', + }, + body: JSON.stringify(spaceTemplate), + }).then((resp) => { + if (!resp?.ok) { + throw new Error('Error when updating space template'); + } + }); +} + +export function deleteSpaceTemplate(id) { + return fetch(`/social/rest/space/templates/${id}`, { + credentials: 'include', + method: 'DELETE', + }).then((resp) => { + if (!resp?.ok) { + throw new Error('Error when deleting space template'); + } + }); +} diff --git a/webapp/src/main/webapp/vue-apps/space-templates-management/main.js b/webapp/src/main/webapp/vue-apps/space-templates-management/main.js new file mode 100644 index 00000000000..2468efe72ee --- /dev/null +++ b/webapp/src/main/webapp/vue-apps/space-templates-management/main.js @@ -0,0 +1,57 @@ +/* + * This file is part of the Meeds project (https://meeds.io/). + * + * Copyright (C) 2020 - 2024 Meeds Association contact@meeds.io + * + * This program is free software; you can redistribute it and/or + * modify it under the terms of the GNU Lesser General Public + * License as published by the Free Software Foundation; either + * version 3 of the License, or (at your option) any later version. + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU + * Lesser General Public License for more details. + * + * You should have received a copy of the GNU Lesser General Public License + * along with this program; if not, write to the Free Software Foundation, + * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. + */ + +import './initComponents.js'; +import './services.js'; + +// get overridden components if exists +if (extensionRegistry) { + const components = extensionRegistry.loadComponents('SpaceTemplatesManagement'); + if (components && components.length > 0) { + components.forEach(cmp => { + Vue.component(cmp.componentName, cmp.componentOptions); + }); + } +} + +const lang = eXo?.env.portal.language || 'en'; +const url = `/social/i18n/locale.portlet.SpaceTemplatesManagement?lang=${lang}`; + +const appId = 'SpaceTemplatesManagement'; +export function init() { + exoi18n.loadLanguageAsync(lang, url) + .then(i18n => + Vue.createApp({ + data: { + spacesCountByTemplates: null, + }, + computed: { + isMobile() { + return this.$vuetify.breakpoint.mobile; + }, + }, + async created() { + this.spacesCountByTemplates = await this.$spaceService.getSpacesCountByTemplates(); + }, + template: ``, + vuetify: Vue.prototype.vuetifyOptions, + i18n, + }, `#${appId}`, 'Space Templates') + ); +} diff --git a/webapp/src/main/webapp/vue-apps/space-templates-management/services.js b/webapp/src/main/webapp/vue-apps/space-templates-management/services.js new file mode 100644 index 00000000000..692c8dbd8aa --- /dev/null +++ b/webapp/src/main/webapp/vue-apps/space-templates-management/services.js @@ -0,0 +1,26 @@ +/* + * This file is part of the Meeds project (https://meeds.io/). + * + * Copyright (C) 2020 - 2024 Meeds Association contact@meeds.io + * + * This program is free software; you can redistribute it and/or + * modify it under the terms of the GNU Lesser General Public + * License as published by the Free Software Foundation; either + * version 3 of the License, or (at your option) any later version. + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU + * Lesser General Public License for more details. + * + * You should have received a copy of the GNU Lesser General Public License + * along with this program; if not, write to the Free Software Foundation, + * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. + */ + +import * as spaceTemplateService from './js/SpaceTemplateService.js'; + +if (!Vue.prototype.$spaceTemplateService) { + window.Object.defineProperty(Vue.prototype, '$spaceTemplateService', { + value: spaceTemplateService, + }); +} diff --git a/webapp/webpack.common.js b/webapp/webpack.common.js index 1697a3aaf57..9385e440d3e 100644 --- a/webapp/webpack.common.js +++ b/webapp/webpack.common.js @@ -37,6 +37,7 @@ let config = { spacesListComponents: './src/main/webapp/vue-apps/spaces-list/initComponents.js', spacesList: './src/main/webapp/vue-apps/spaces-list/main.js', spacesOverview: './src/main/webapp/vue-apps/spaces-overview/main.js', + spaceTemplateManagement: './src/main/webapp/vue-apps/space-templates-management/main.js', suggestions: './src/main/webapp/vue-apps/suggestions-people-space/main.js', peopleListComponents: './src/main/webapp/vue-apps/people-list/main-components.js', peopleList: './src/main/webapp/vue-apps/people-list/main.js',