From fe3949d77dd81946b035525677e78f3f3f59a7b1 Mon Sep 17 00:00:00 2001 From: Giancarlo Panichi Date: Tue, 30 Nov 2021 16:47:42 +0100 Subject: [PATCH] First Import --- .editorconfig | 23 + .eslintignore | 9 + .eslintrc.json | 24 + .gitattributes | 150 + .gitignore | 155 + .huskyrc | 5 + .lintstagedrc.js | 3 + .mvn/wrapper/MavenWrapperDownloader.java | 117 + .mvn/wrapper/maven-wrapper.jar | Bin 0 -> 50710 bytes .mvn/wrapper/maven-wrapper.properties | 2 + .prettierignore | 8 + .prettierrc | 12 + .yo-rc.json | 39 + README-Orig.md | 254 + README.md | 20 + angular.json | 36 + checkstyle.xml | 21 + mvnw | 310 + mvnw.cmd | 182 + package-lock.json | 24027 ++++++++++++++++ package.json | 123 + pom.xml | 1202 + postcss.config.js | 5 + proxy.conf.json | 7 + sonar-project.properties | 30 + src/main/docker/app.yml | 17 + .../grafana/provisioning/dashboards/JVM.json | 3778 +++ .../provisioning/dashboards/dashboard.yml | 11 + .../provisioning/datasources/datasource.yml | 50 + src/main/docker/monitoring.yml | 26 + src/main/docker/postgresql.yml | 12 + src/main/docker/prometheus/prometheus.yml | 31 + src/main/docker/sonar.yml | 7 + src/main/docker/swagger-editor.yml | 6 + .../cnr/isti/epasmed/ApplicationWebXml.java | 20 + .../java/it/cnr/isti/epasmed/EpasmedApp.java | 108 + .../epasmed/aop/logging/LoggingAspect.java | 113 + .../epasmed/config/ApplicationProperties.java | 942 + .../epasmed/config/AsyncConfiguration.java | 47 + .../config/CloudDatabaseConfiguration.java | 28 + .../it/cnr/isti/epasmed/config/Constants.java | 17 + .../config/DateTimeFormatConfiguration.java | 22 + .../EpasMedDataSourceConfiguration.java | 202 + .../config/EpasRestTemplateConfig.java | 36 + .../epasmed/config/JacksonConfiguration.java | 52 + .../epasmed/config/LocaleConfiguration.java | 27 + .../config/LoggingAspectConfiguration.java | 19 + .../epasmed/config/LoggingConfiguration.java | 50 + .../epasmed/config/OpenApiConfiguration.java | 72 + .../epasmed/config/SecurityConfiguration.java | 98 + ...emaInformativoDataSourceConfiguration.java | 88 + .../StaticResourcesWebConfiguration.java | 49 + .../isti/epasmed/config/WebConfigurer.java | 138 + .../config/audit/AuditEventConverter.java | 86 + .../epasmed/config/audit/package-info.java | 4 + .../cnr/isti/epasmed/config/package-info.java | 4 + .../domain/AbstractAuditingEntity.java | 77 + .../it/cnr/isti/epasmed/domain/Authority.java | 58 + .../epasmed/domain/PersistentAuditEvent.java | 108 + .../isti/epasmed/domain/PersistentToken.java | 128 + .../it/cnr/isti/epasmed/domain/TabsSI.java | 106 + .../java/it/cnr/isti/epasmed/domain/User.java | 240 + .../cnr/isti/epasmed/domain/package-info.java | 4 + .../epas/client/EPASAffiliationsCient.java | 108 + .../epas/client/EPASContractsCient.java | 117 + .../epasmed/epas/client/EPASGroupsCient.java | 83 + .../epasmed/epas/client/EPASPersonsCient.java | 127 + .../epas/client/EPASWorkinTipeTypesCient.java | 48 + .../epasmed/epas/dto/EPASAffiliationsDTO.java | 25 + .../epasmed/epas/dto/EPASContractsDTO.java | 26 + .../isti/epasmed/epas/dto/EPASGroupsDTO.java | 24 + .../isti/epasmed/epas/dto/EPASPersonsDTO.java | 31 + .../epas/mapper/EPASAffiliationsMapper.java | 47 + .../epas/mapper/EPASContractsMapper.java | 45 + .../epasmed/epas/mapper/EPASGroupsMapper.java | 47 + .../epas/mapper/EPASPersonsMapper.java | 51 + .../epasmed/epas/model/EPASAffiliations.java | 26 + .../epasmed/epas/model/EPASContracts.java | 28 + .../isti/epasmed/epas/model/EPASGroups.java | 25 + .../epasmed/epas/model/EPASGroupsManager.java | 24 + .../isti/epasmed/epas/model/EPASOffice.java | 22 + .../isti/epasmed/epas/model/EPASPersons.java | 31 + .../epas/model/EPASWorkingTimeForPerson.java | 20 + .../epas/model/EPASWorkingTimeTypeDays.java | 27 + .../epas/model/EPASWorkingTimeTypes.java | 25 + .../epas/service/EPASAffiliationsService.java | 48 + .../epas/service/EPASContractsService.java | 52 + .../epas/service/EPASGroupsService.java | 40 + .../epas/service/EPASPersonsService.java | 54 + .../service/EPASWorkingTimeTypesService.java | 26 + .../repository/AuthorityRepository.java | 11 + .../CustomAuditEventRepository.java | 89 + .../PersistenceAuditEventRepository.java | 23 + .../repository/PersistentTokenRepository.java | 19 + .../epasmed/repository/TabsSIRepository.java | 19 + .../epasmed/repository/UserRepository.java | 38 + .../isti/epasmed/repository/package-info.java | 4 + .../security/AuthoritiesConstants.java | 18 + .../security/DomainUserDetailsService.java | 62 + .../PersistentTokenRememberMeServices.java | 228 + .../isti/epasmed/security/SecurityUtils.java | 73 + .../security/SpringSecurityAuditorAware.java | 20 + .../security/UserNotActivatedException.java | 19 + .../isti/epasmed/security/package-info.java | 4 + .../epasmed/service/AuditEventService.java | 77 + .../service/EmailAlreadyUsedException.java | 11 + .../service/InvalidPasswordException.java | 11 + .../cnr/isti/epasmed/service/MailService.java | 106 + .../isti/epasmed/service/TabsSIService.java | 125 + .../cnr/isti/epasmed/service/UserService.java | 303 + .../service/UsernameAlreadyUsedException.java | 11 + .../service/dto/PasswordChangeDTO.java | 35 + .../cnr/isti/epasmed/service/dto/UserDTO.java | 197 + .../epasmed/service/dto/package-info.java | 4 + .../epasmed/service/mapper/UserMapper.java | 81 + .../epasmed/service/mapper/package-info.java | 4 + .../isti/epasmed/service/package-info.java | 4 + .../model/SIAnagrafico.java | 30 + .../sistemainformativo/model/SIEmail.java | 25 + .../sistemainformativo/model/SIGruppi.java | 23 + .../model/SIGruppiPers.java | 30 + .../sistemainformativo/model/SIPosizioni.java | 35 + .../sistemainformativo/model/SIProroghe.java | 29 + .../sistemainformativo/model/SITelefoni.java | 25 + .../repository/SIAnagraficoRepository.java | 71 + .../repository/SIEmailRepository.java | 59 + .../repository/SIGruppiPersRepository.java | 63 + .../repository/SIGruppiRepository.java | 57 + .../repository/SIMasterLogRepository.java | 124 + .../repository/SIPosizioniRepository.java | 69 + .../repository/SIProrogheRepository.java | 63 + .../repository/SITelefoniRepository.java | 59 + .../service/SIAnagraficoService.java | 37 + .../service/SIEmailService.java | 37 + .../service/SIGruppiPersService.java | 37 + .../service/SIGruppiService.java | 37 + .../service/SIMasterLogService.java | 52 + .../service/SIPosizioniService.java | 37 + .../service/SIProrogheService.java | 37 + .../service/SITelefoniService.java | 37 + .../it/cnr/isti/epasmed/sync/SyncService.java | 405 + .../epasmed/web/rest/AccountResource.java | 240 + .../isti/epasmed/web/rest/AuditResource.java | 79 + .../web/rest/ClientForwardController.java | 17 + .../isti/epasmed/web/rest/SyncResource.java | 70 + .../isti/epasmed/web/rest/TabsSIResource.java | 172 + .../isti/epasmed/web/rest/UserResource.java | 200 + .../rest/epas/EPASAffiliationsResource.java | 185 + .../web/rest/epas/EPASContractsResource.java | 192 + .../web/rest/epas/EPASGroupsResource.java | 158 + .../web/rest/epas/EPASPersonsResource.java | 202 + .../epas/EPASWorkingTimeTypesResource.java | 67 + .../rest/errors/BadRequestAlertException.java | 42 + .../errors/EmailAlreadyUsedException.java | 10 + .../web/rest/errors/ErrorConstants.java | 18 + .../web/rest/errors/ExceptionTranslator.java | 200 + .../epasmed/web/rest/errors/FieldErrorVM.java | 33 + .../rest/errors/InvalidPasswordException.java | 13 + .../errors/LoginAlreadyUsedException.java | 10 + .../epasmed/web/rest/errors/package-info.java | 6 + .../isti/epasmed/web/rest/package-info.java | 4 + .../SIMasterLogResource.java | 29 + .../epasmed/web/rest/vm/KeyAndPasswordVM.java | 27 + .../epasmed/web/rest/vm/ManagedUserVM.java | 35 + .../epasmed/web/rest/vm/package-info.java | 4 + src/main/javasecurity/java.security | 1223 + src/main/jib/entrypoint.sh | 10 + src/main/resources/.h2.server.properties | 6 + src/main/resources/banner.txt | 11 + src/main/resources/config/application-dev.yml | 132 + .../resources/config/application-prod.yml | 164 + src/main/resources/config/application-tls.yml | 19 + src/main/resources/config/application.yml | 141 + .../00000000000000_initial_schema.xml | 202 + .../liquibase/epasmed/data/authority.csv | 4 + .../config/liquibase/epasmed/data/tabssi.csv | 17 + .../config/liquibase/epasmed/data/user.csv | 6 + .../liquibase/epasmed/data/user_authority.csv | 8 + .../config/liquibase/epasmed/master.xml | 20 + src/main/resources/config/tls/keystore.p12 | Bin 0 -> 2607 bytes src/main/resources/i18n/messages.properties | 21 + src/main/resources/logback-spring.xml | 68 + src/main/resources/swagger/api.yml | 33 + src/main/resources/templates/error.html | 87 + .../templates/mail/activationEmail.html | 25 + .../templates/mail/creationEmail.html | 25 + .../templates/mail/passwordResetEmail.html | 25 + src/main/webapp/404.html | 61 + src/main/webapp/WEB-INF/web.xml | 13 + src/main/webapp/app/account/account.module.ts | 29 + src/main/webapp/app/account/account.route.ts | 26 + .../account/activate/activate.component.html | 16 + .../account/activate/activate.component.ts | 28 + .../app/account/activate/activate.route.ts | 12 + .../app/account/activate/activate.service.ts | 16 + .../password-reset-finish.component.html | 84 + .../finish/password-reset-finish.component.ts | 69 + .../finish/password-reset-finish.route.ts | 12 + .../finish/password-reset-finish.service.ts | 14 + .../init/password-reset-init.component.html | 49 + .../init/password-reset-init.component.ts | 30 + .../init/password-reset-init.route.ts | 12 + .../init/password-reset-init.service.ts | 14 + .../password-strength-bar.component.ts | 83 + .../password/password-strength-bar.scss | 23 + .../account/password/password.component.html | 87 + .../account/password/password.component.ts | 45 + .../app/account/password/password.route.ts | 15 + .../app/account/password/password.service.ts | 14 + .../account/register/register.component.html | 144 + .../account/register/register.component.ts | 78 + .../app/account/register/register.route.ts | 12 + .../app/account/register/register.service.ts | 15 + .../app/account/sessions/session.model.ts | 3 + .../account/sessions/sessions.component.html | 38 + .../account/sessions/sessions.component.ts | 38 + .../app/account/sessions/sessions.route.ts | 15 + .../app/account/sessions/sessions.service.ts | 21 + .../account/settings/settings.component.html | 91 + .../account/settings/settings.component.ts | 49 + .../app/account/settings/settings.route.ts | 15 + .../webapp/app/admin/admin-routing.module.ts | 44 + .../app/admin/audits/audit-data.model.ts | 3 + .../webapp/app/admin/audits/audit.model.ts | 5 + .../app/admin/audits/audits.component.html | 67 + .../app/admin/audits/audits.component.ts | 115 + .../webapp/app/admin/audits/audits.module.ts | 13 + .../webapp/app/admin/audits/audits.route.ts | 12 + .../webapp/app/admin/audits/audits.service.ts | 28 + .../configuration.component.html | 50 + .../configuration/configuration.component.ts | 32 + .../configuration/configuration.module.ts | 13 + .../configuration/configuration.route.ts | 11 + .../configuration/configuration.service.ts | 68 + .../webapp/app/admin/docs/docs.component.html | 2 + .../webapp/app/admin/docs/docs.component.ts | 8 + src/main/webapp/app/admin/docs/docs.module.ts | 13 + src/main/webapp/app/admin/docs/docs.route.ts | 11 + src/main/webapp/app/admin/docs/docs.scss | 6 + .../admin/health/health-modal.component.html | 36 + .../admin/health/health-modal.component.ts | 37 + .../app/admin/health/health.component.html | 38 + .../app/admin/health/health.component.ts | 44 + .../webapp/app/admin/health/health.module.ts | 15 + .../webapp/app/admin/health/health.route.ts | 11 + .../webapp/app/admin/health/health.service.ts | 30 + src/main/webapp/app/admin/logs/log.model.ts | 15 + .../webapp/app/admin/logs/logs.component.html | 28 + .../webapp/app/admin/logs/logs.component.ts | 34 + src/main/webapp/app/admin/logs/logs.module.ts | 13 + src/main/webapp/app/admin/logs/logs.route.ts | 11 + .../webapp/app/admin/logs/logs.service.ts | 19 + .../app/admin/metrics/metrics.component.html | 65 + .../app/admin/metrics/metrics.component.ts | 47 + .../app/admin/metrics/metrics.module.ts | 13 + .../webapp/app/admin/metrics/metrics.route.ts | 11 + .../app/admin/metrics/metrics.service.ts | 23 + ...er-management-delete-dialog.component.html | 23 + ...user-management-delete-dialog.component.ts | 27 + .../user-management-detail.component.html | 55 + .../user-management-detail.component.ts | 18 + .../user-management-update.component.html | 119 + .../user-management-update.component.ts | 105 + .../user-management.component.html | 85 + .../user-management.component.ts | 106 + .../user-management/user-management.module.ts | 21 + .../user-management/user-management.route.ts | 53 + src/main/webapp/app/app-routing.module.ts | 40 + src/main/webapp/app/app.constants.ts | 8 + src/main/webapp/app/app.main.ts | 16 + src/main/webapp/app/app.module.ts | 30 + .../webapp/app/blocks/config/prod.config.ts | 9 + .../blocks/config/uib-pagination.config.ts | 13 + .../interceptor/auth-expired.interceptor.ts | 36 + .../interceptor/errorhandler.interceptor.ts | 20 + .../interceptor/notification.interceptor.ts | 30 + .../webapp/app/core/auth/account.service.ts | 82 + .../app/core/auth/auth-session.service.ts | 36 + src/main/webapp/app/core/auth/csrf.service.ts | 11 + .../app/core/auth/state-storage.service.ts | 21 + .../core/auth/user-route-access-service.ts | 53 + src/main/webapp/app/core/core.module.ts | 63 + .../app/core/icons/font-awesome-icons.ts | 75 + .../app/core/login/login-modal.service.ts | 20 + src/main/webapp/app/core/login/login.model.ts | 3 + .../webapp/app/core/login/login.service.ts | 29 + .../webapp/app/core/user/account.model.ts | 12 + src/main/webapp/app/core/user/user.model.ts | 33 + src/main/webapp/app/core/user/user.service.ts | 39 + src/main/webapp/app/entities/entity.module.ts | 18 + .../tabssi-delete-dialog.component.html | 23 + .../tabssi/tabssi-delete-dialog.component.ts | 27 + .../tabssi/tabssi-detail.component.html | 30 + .../tabssi/tabssi-detail.component.ts | 18 + .../tabssi/tabssi-update.component.html | 80 + .../tabssi/tabssi-update.component.ts | 80 + .../app/entities/tabssi/tabssi.component.html | 75 + .../app/entities/tabssi/tabssi.component.ts | 102 + .../app/entities/tabssi/tabssi.model.ts | 11 + .../app/entities/tabssi/tabssi.module.ts | 16 + .../app/entities/tabssi/tabssi.route.ts | 53 + .../app/entities/tabssi/tabssi.service.ts | 35 + src/main/webapp/app/home/home.component.html | 32 + src/main/webapp/app/home/home.component.ts | 36 + src/main/webapp/app/home/home.module.ts | 12 + src/main/webapp/app/home/home.route.ts | 12 + src/main/webapp/app/home/home.scss | 23 + .../app/layouts/error/error.component.html | 15 + .../app/layouts/error/error.component.ts | 20 + .../webapp/app/layouts/error/error.route.ts | 36 + .../app/layouts/footer/footer.component.html | 3 + .../app/layouts/footer/footer.component.ts | 7 + .../app/layouts/main/main.component.html | 13 + .../webapp/app/layouts/main/main.component.ts | 43 + .../app/layouts/navbar/navbar.component.html | 146 + .../app/layouts/navbar/navbar.component.ts | 63 + .../webapp/app/layouts/navbar/navbar.route.ts | 9 + .../webapp/app/layouts/navbar/navbar.scss | 42 + .../layouts/profiles/page-ribbon.component.ts | 23 + .../app/layouts/profiles/page-ribbon.scss | 31 + .../layouts/profiles/profile-info.model.ts | 15 + .../app/layouts/profiles/profile.service.ts | 43 + src/main/webapp/app/polyfills.ts | 3 + .../app/shared/alert/alert-error.component.ts | 113 + .../app/shared/alert/alert-error.model.ts | 3 + .../app/shared/alert/alert.component.ts | 39 + .../auth/has-any-authority.directive.ts | 47 + .../shared/constants/authority.constants.ts | 4 + .../app/shared/constants/error.constants.ts | 3 + .../app/shared/constants/input.constants.ts | 2 + .../shared/constants/pagination.constants.ts | 1 + .../app/shared/login/login.component.html | 47 + .../app/shared/login/login.component.ts | 73 + .../webapp/app/shared/shared-libs.module.ts | 12 + src/main/webapp/app/shared/shared.module.ts | 14 + .../app/shared/util/datepicker-adapter.ts | 23 + .../webapp/app/shared/util/request-util.ts | 33 + src/main/webapp/app/vendor.ts | 2 + src/main/webapp/content/css/loading.css | 152 + .../webapp/content/images/ePASMedLogo.png | Bin 0 -> 7181 bytes src/main/webapp/content/images/plumber_0.svg | 70 + .../content/images/plumber_0_head_192.png | Bin 0 -> 20533 bytes .../content/images/plumber_0_head_256.png | Bin 0 -> 31122 bytes .../content/images/plumber_0_head_384.png | Bin 0 -> 59863 bytes .../content/images/plumber_0_head_512.png | Bin 0 -> 89302 bytes .../content/scss/_bootstrap-variables.scss | 45 + src/main/webapp/content/scss/global.scss | 195 + src/main/webapp/content/scss/vendor.scss | 12 + src/main/webapp/favicon.ico | Bin 0 -> 16958 bytes src/main/webapp/index.html | 107 + src/main/webapp/manifest.webapp | 31 + src/main/webapp/robots.txt | 11 + .../swagger-ui/dist/images/throbber.gif | Bin 0 -> 9257 bytes src/main/webapp/swagger-ui/index.html | 71 + .../java/it/cnr/isti/epasmed/ArchTest.java | 29 + .../epasmed/config/NoOpMailConfiguration.java | 24 + .../StaticResourcesWebConfigurerTest.java | 80 + .../epasmed/config/WebConfigurerTest.java | 165 + .../config/WebConfigurerTestController.java | 16 + .../CustomAuditEventRepositoryIT.java | 158 + .../security/DomainUserDetailsServiceIT.java | 115 + .../security/SecurityUtilsUnitTest.java | 62 + .../epasmed/service/AuditEventServiceIT.java | 73 + .../isti/epasmed/service/MailServiceIT.java | 247 + .../isti/epasmed/service/UserServiceIT.java | 235 + .../service/mapper/UserMapperTest.java | 136 + .../epasmed/web/rest/AccountResourceIT.java | 570 + .../epasmed/web/rest/AuditResourceIT.java | 135 + .../web/rest/ClientForwardControllerTest.java | 64 + .../isti/epasmed/web/rest/SyncResourceIT.java | 57 + .../cnr/isti/epasmed/web/rest/TestUtil.java | 157 + .../isti/epasmed/web/rest/UserResourceIT.java | 590 + .../web/rest/WithUnauthenticatedMockUser.java | 24 + .../rest/epas/EPASAffiliationsResourceIT.java | 318 + .../rest/epas/EPASContractsResourceIT.java | 330 + .../web/rest/epas/EPASGroupsResourceIT.java | 302 + .../web/rest/epas/EPASPersonsResourceIT.java | 299 + .../epas/EPASWorkingTimeTypesResourceIT.java | 91 + .../rest/errors/ExceptionTranslatorIT.java | 111 + .../ExceptionTranslatorTestController.java | 72 + src/test/javascript/jest-global-mocks.ts | 3 + src/test/javascript/jest.conf.js | 58 + src/test/javascript/jest.ts | 2 + .../activate/activate.component.spec.ts | 72 + .../password-reset-finish.component.spec.ts | 102 + .../password-reset-init.component.spec.ts | 66 + .../password-strength-bar.component.spec.ts | 48 + .../password/password.component.spec.ts | 102 + .../register/register.component.spec.ts | 134 + .../sessions/sessions.component.spec.ts | 112 + .../settings/settings.component.spec.ts | 88 + .../app/admin/audits/audits.component.spec.ts | 204 + .../app/admin/audits/audits.service.spec.ts | 70 + .../configuration.component.spec.ts | 68 + .../configuration.service.spec.ts | 81 + .../app/admin/health/health.component.spec.ts | 67 + .../app/admin/logs/logs.component.spec.ts | 85 + .../spec/app/admin/logs/logs.service.spec.ts | 44 + .../admin/metrics/metrics.component.spec.ts | 54 + .../app/admin/metrics/metrics.service.spec.ts | 68 + ...management-delete-dialog.component.spec.ts | 56 + .../user-management-detail.component.spec.ts | 62 + .../user-management-update.component.spec.ts | 104 + .../user-management.component.spec.ts | 86 + .../core/login/login-modal.service.spec.ts | 48 + .../app/core/user/account.service.spec.ts | 209 + .../spec/app/core/user/user.service.spec.ts | 79 + .../spec/app/home/home.component.spec.ts | 55 + .../app/layouts/main/main.component.spec.ts | 108 + .../layouts/navbar/navbar.component.spec.ts | 52 + .../alert/alert-error.component.spec.ts | 140 + .../app/shared/alert/alert.component.spec.ts | 44 + .../app/shared/login/login.component.spec.ts | 116 + .../spec/helpers/mock-account.service.ts | 31 + .../spec/helpers/mock-active-modal.service.ts | 15 + .../spec/helpers/mock-alert.service.ts | 13 + .../helpers/mock-event-manager.service.ts | 13 + .../spec/helpers/mock-login-modal.service.ts | 14 + .../spec/helpers/mock-login.service.ts | 27 + .../spec/helpers/mock-route.service.ts | 50 + .../helpers/mock-state-storage.service.ts | 21 + src/test/javascript/spec/helpers/spyobject.ts | 48 + src/test/javascript/spec/test.module.ts | 67 + .../config/application-testcontainers.yml | 22 + src/test/resources/config/application.yml | 126 + .../resources/i18n/messages_en.properties | 1 + .../web/rest/epas/affiliazioniRepartiISTI.csv | 445 + .../epas/affiliazioniRepartiISTIRimanenti.csv | 71 + .../isti/epasmed/web/rest/epas/personale.csv | 247 + .../epasmed/web/rest/epas/posizioniISTI.csv | 167 + src/test/resources/logback.xml | 47 + .../resources/templates/mail/testEmail.html | 1 + tsconfig.app.json | 4 + tsconfig.base.json | 31 + tsconfig.json | 11 + tslint.json | 17 + webpack/ePASMedLogo.png | Bin 0 -> 7181 bytes webpack/logo-jhipster.png | Bin 0 -> 3326 bytes webpack/utils.js | 39 + webpack/webpack.common.js | 100 + webpack/webpack.dev.js | 120 + webpack/webpack.prod.js | 147 + 442 files changed, 57106 insertions(+) create mode 100644 .editorconfig create mode 100644 .eslintignore create mode 100644 .eslintrc.json create mode 100644 .gitattributes create mode 100644 .gitignore create mode 100644 .huskyrc create mode 100644 .lintstagedrc.js create mode 100644 .mvn/wrapper/MavenWrapperDownloader.java create mode 100644 .mvn/wrapper/maven-wrapper.jar create mode 100644 .mvn/wrapper/maven-wrapper.properties create mode 100644 .prettierignore create mode 100644 .prettierrc create mode 100644 .yo-rc.json create mode 100644 README-Orig.md create mode 100644 README.md create mode 100644 angular.json create mode 100644 checkstyle.xml create mode 100755 mvnw create mode 100644 mvnw.cmd create mode 100644 package-lock.json create mode 100644 package.json create mode 100644 pom.xml create mode 100644 postcss.config.js create mode 100644 proxy.conf.json create mode 100644 sonar-project.properties create mode 100644 src/main/docker/app.yml create mode 100644 src/main/docker/grafana/provisioning/dashboards/JVM.json create mode 100644 src/main/docker/grafana/provisioning/dashboards/dashboard.yml create mode 100644 src/main/docker/grafana/provisioning/datasources/datasource.yml create mode 100644 src/main/docker/monitoring.yml create mode 100644 src/main/docker/postgresql.yml create mode 100644 src/main/docker/prometheus/prometheus.yml create mode 100644 src/main/docker/sonar.yml create mode 100644 src/main/docker/swagger-editor.yml create mode 100644 src/main/java/it/cnr/isti/epasmed/ApplicationWebXml.java create mode 100644 src/main/java/it/cnr/isti/epasmed/EpasmedApp.java create mode 100644 src/main/java/it/cnr/isti/epasmed/aop/logging/LoggingAspect.java create mode 100644 src/main/java/it/cnr/isti/epasmed/config/ApplicationProperties.java create mode 100644 src/main/java/it/cnr/isti/epasmed/config/AsyncConfiguration.java create mode 100644 src/main/java/it/cnr/isti/epasmed/config/CloudDatabaseConfiguration.java create mode 100644 src/main/java/it/cnr/isti/epasmed/config/Constants.java create mode 100644 src/main/java/it/cnr/isti/epasmed/config/DateTimeFormatConfiguration.java create mode 100644 src/main/java/it/cnr/isti/epasmed/config/EpasMedDataSourceConfiguration.java create mode 100644 src/main/java/it/cnr/isti/epasmed/config/EpasRestTemplateConfig.java create mode 100644 src/main/java/it/cnr/isti/epasmed/config/JacksonConfiguration.java create mode 100644 src/main/java/it/cnr/isti/epasmed/config/LocaleConfiguration.java create mode 100644 src/main/java/it/cnr/isti/epasmed/config/LoggingAspectConfiguration.java create mode 100644 src/main/java/it/cnr/isti/epasmed/config/LoggingConfiguration.java create mode 100644 src/main/java/it/cnr/isti/epasmed/config/OpenApiConfiguration.java create mode 100644 src/main/java/it/cnr/isti/epasmed/config/SecurityConfiguration.java create mode 100644 src/main/java/it/cnr/isti/epasmed/config/SistemaInformativoDataSourceConfiguration.java create mode 100644 src/main/java/it/cnr/isti/epasmed/config/StaticResourcesWebConfiguration.java create mode 100644 src/main/java/it/cnr/isti/epasmed/config/WebConfigurer.java create mode 100644 src/main/java/it/cnr/isti/epasmed/config/audit/AuditEventConverter.java create mode 100644 src/main/java/it/cnr/isti/epasmed/config/audit/package-info.java create mode 100644 src/main/java/it/cnr/isti/epasmed/config/package-info.java create mode 100644 src/main/java/it/cnr/isti/epasmed/domain/AbstractAuditingEntity.java create mode 100644 src/main/java/it/cnr/isti/epasmed/domain/Authority.java create mode 100644 src/main/java/it/cnr/isti/epasmed/domain/PersistentAuditEvent.java create mode 100644 src/main/java/it/cnr/isti/epasmed/domain/PersistentToken.java create mode 100644 src/main/java/it/cnr/isti/epasmed/domain/TabsSI.java create mode 100644 src/main/java/it/cnr/isti/epasmed/domain/User.java create mode 100644 src/main/java/it/cnr/isti/epasmed/domain/package-info.java create mode 100644 src/main/java/it/cnr/isti/epasmed/epas/client/EPASAffiliationsCient.java create mode 100644 src/main/java/it/cnr/isti/epasmed/epas/client/EPASContractsCient.java create mode 100644 src/main/java/it/cnr/isti/epasmed/epas/client/EPASGroupsCient.java create mode 100644 src/main/java/it/cnr/isti/epasmed/epas/client/EPASPersonsCient.java create mode 100644 src/main/java/it/cnr/isti/epasmed/epas/client/EPASWorkinTipeTypesCient.java create mode 100644 src/main/java/it/cnr/isti/epasmed/epas/dto/EPASAffiliationsDTO.java create mode 100644 src/main/java/it/cnr/isti/epasmed/epas/dto/EPASContractsDTO.java create mode 100644 src/main/java/it/cnr/isti/epasmed/epas/dto/EPASGroupsDTO.java create mode 100644 src/main/java/it/cnr/isti/epasmed/epas/dto/EPASPersonsDTO.java create mode 100644 src/main/java/it/cnr/isti/epasmed/epas/mapper/EPASAffiliationsMapper.java create mode 100644 src/main/java/it/cnr/isti/epasmed/epas/mapper/EPASContractsMapper.java create mode 100644 src/main/java/it/cnr/isti/epasmed/epas/mapper/EPASGroupsMapper.java create mode 100644 src/main/java/it/cnr/isti/epasmed/epas/mapper/EPASPersonsMapper.java create mode 100644 src/main/java/it/cnr/isti/epasmed/epas/model/EPASAffiliations.java create mode 100644 src/main/java/it/cnr/isti/epasmed/epas/model/EPASContracts.java create mode 100644 src/main/java/it/cnr/isti/epasmed/epas/model/EPASGroups.java create mode 100644 src/main/java/it/cnr/isti/epasmed/epas/model/EPASGroupsManager.java create mode 100644 src/main/java/it/cnr/isti/epasmed/epas/model/EPASOffice.java create mode 100644 src/main/java/it/cnr/isti/epasmed/epas/model/EPASPersons.java create mode 100644 src/main/java/it/cnr/isti/epasmed/epas/model/EPASWorkingTimeForPerson.java create mode 100644 src/main/java/it/cnr/isti/epasmed/epas/model/EPASWorkingTimeTypeDays.java create mode 100644 src/main/java/it/cnr/isti/epasmed/epas/model/EPASWorkingTimeTypes.java create mode 100644 src/main/java/it/cnr/isti/epasmed/epas/service/EPASAffiliationsService.java create mode 100644 src/main/java/it/cnr/isti/epasmed/epas/service/EPASContractsService.java create mode 100644 src/main/java/it/cnr/isti/epasmed/epas/service/EPASGroupsService.java create mode 100644 src/main/java/it/cnr/isti/epasmed/epas/service/EPASPersonsService.java create mode 100644 src/main/java/it/cnr/isti/epasmed/epas/service/EPASWorkingTimeTypesService.java create mode 100644 src/main/java/it/cnr/isti/epasmed/repository/AuthorityRepository.java create mode 100644 src/main/java/it/cnr/isti/epasmed/repository/CustomAuditEventRepository.java create mode 100644 src/main/java/it/cnr/isti/epasmed/repository/PersistenceAuditEventRepository.java create mode 100644 src/main/java/it/cnr/isti/epasmed/repository/PersistentTokenRepository.java create mode 100644 src/main/java/it/cnr/isti/epasmed/repository/TabsSIRepository.java create mode 100644 src/main/java/it/cnr/isti/epasmed/repository/UserRepository.java create mode 100644 src/main/java/it/cnr/isti/epasmed/repository/package-info.java create mode 100644 src/main/java/it/cnr/isti/epasmed/security/AuthoritiesConstants.java create mode 100644 src/main/java/it/cnr/isti/epasmed/security/DomainUserDetailsService.java create mode 100644 src/main/java/it/cnr/isti/epasmed/security/PersistentTokenRememberMeServices.java create mode 100644 src/main/java/it/cnr/isti/epasmed/security/SecurityUtils.java create mode 100644 src/main/java/it/cnr/isti/epasmed/security/SpringSecurityAuditorAware.java create mode 100644 src/main/java/it/cnr/isti/epasmed/security/UserNotActivatedException.java create mode 100644 src/main/java/it/cnr/isti/epasmed/security/package-info.java create mode 100644 src/main/java/it/cnr/isti/epasmed/service/AuditEventService.java create mode 100644 src/main/java/it/cnr/isti/epasmed/service/EmailAlreadyUsedException.java create mode 100644 src/main/java/it/cnr/isti/epasmed/service/InvalidPasswordException.java create mode 100644 src/main/java/it/cnr/isti/epasmed/service/MailService.java create mode 100644 src/main/java/it/cnr/isti/epasmed/service/TabsSIService.java create mode 100644 src/main/java/it/cnr/isti/epasmed/service/UserService.java create mode 100644 src/main/java/it/cnr/isti/epasmed/service/UsernameAlreadyUsedException.java create mode 100644 src/main/java/it/cnr/isti/epasmed/service/dto/PasswordChangeDTO.java create mode 100644 src/main/java/it/cnr/isti/epasmed/service/dto/UserDTO.java create mode 100644 src/main/java/it/cnr/isti/epasmed/service/dto/package-info.java create mode 100644 src/main/java/it/cnr/isti/epasmed/service/mapper/UserMapper.java create mode 100644 src/main/java/it/cnr/isti/epasmed/service/mapper/package-info.java create mode 100644 src/main/java/it/cnr/isti/epasmed/service/package-info.java create mode 100644 src/main/java/it/cnr/isti/epasmed/sistemainformativo/model/SIAnagrafico.java create mode 100644 src/main/java/it/cnr/isti/epasmed/sistemainformativo/model/SIEmail.java create mode 100644 src/main/java/it/cnr/isti/epasmed/sistemainformativo/model/SIGruppi.java create mode 100644 src/main/java/it/cnr/isti/epasmed/sistemainformativo/model/SIGruppiPers.java create mode 100644 src/main/java/it/cnr/isti/epasmed/sistemainformativo/model/SIPosizioni.java create mode 100644 src/main/java/it/cnr/isti/epasmed/sistemainformativo/model/SIProroghe.java create mode 100644 src/main/java/it/cnr/isti/epasmed/sistemainformativo/model/SITelefoni.java create mode 100644 src/main/java/it/cnr/isti/epasmed/sistemainformativo/repository/SIAnagraficoRepository.java create mode 100644 src/main/java/it/cnr/isti/epasmed/sistemainformativo/repository/SIEmailRepository.java create mode 100644 src/main/java/it/cnr/isti/epasmed/sistemainformativo/repository/SIGruppiPersRepository.java create mode 100644 src/main/java/it/cnr/isti/epasmed/sistemainformativo/repository/SIGruppiRepository.java create mode 100644 src/main/java/it/cnr/isti/epasmed/sistemainformativo/repository/SIMasterLogRepository.java create mode 100644 src/main/java/it/cnr/isti/epasmed/sistemainformativo/repository/SIPosizioniRepository.java create mode 100644 src/main/java/it/cnr/isti/epasmed/sistemainformativo/repository/SIProrogheRepository.java create mode 100644 src/main/java/it/cnr/isti/epasmed/sistemainformativo/repository/SITelefoniRepository.java create mode 100644 src/main/java/it/cnr/isti/epasmed/sistemainformativo/service/SIAnagraficoService.java create mode 100644 src/main/java/it/cnr/isti/epasmed/sistemainformativo/service/SIEmailService.java create mode 100644 src/main/java/it/cnr/isti/epasmed/sistemainformativo/service/SIGruppiPersService.java create mode 100644 src/main/java/it/cnr/isti/epasmed/sistemainformativo/service/SIGruppiService.java create mode 100644 src/main/java/it/cnr/isti/epasmed/sistemainformativo/service/SIMasterLogService.java create mode 100644 src/main/java/it/cnr/isti/epasmed/sistemainformativo/service/SIPosizioniService.java create mode 100644 src/main/java/it/cnr/isti/epasmed/sistemainformativo/service/SIProrogheService.java create mode 100644 src/main/java/it/cnr/isti/epasmed/sistemainformativo/service/SITelefoniService.java create mode 100644 src/main/java/it/cnr/isti/epasmed/sync/SyncService.java create mode 100644 src/main/java/it/cnr/isti/epasmed/web/rest/AccountResource.java create mode 100644 src/main/java/it/cnr/isti/epasmed/web/rest/AuditResource.java create mode 100644 src/main/java/it/cnr/isti/epasmed/web/rest/ClientForwardController.java create mode 100644 src/main/java/it/cnr/isti/epasmed/web/rest/SyncResource.java create mode 100644 src/main/java/it/cnr/isti/epasmed/web/rest/TabsSIResource.java create mode 100644 src/main/java/it/cnr/isti/epasmed/web/rest/UserResource.java create mode 100644 src/main/java/it/cnr/isti/epasmed/web/rest/epas/EPASAffiliationsResource.java create mode 100644 src/main/java/it/cnr/isti/epasmed/web/rest/epas/EPASContractsResource.java create mode 100644 src/main/java/it/cnr/isti/epasmed/web/rest/epas/EPASGroupsResource.java create mode 100644 src/main/java/it/cnr/isti/epasmed/web/rest/epas/EPASPersonsResource.java create mode 100644 src/main/java/it/cnr/isti/epasmed/web/rest/epas/EPASWorkingTimeTypesResource.java create mode 100644 src/main/java/it/cnr/isti/epasmed/web/rest/errors/BadRequestAlertException.java create mode 100644 src/main/java/it/cnr/isti/epasmed/web/rest/errors/EmailAlreadyUsedException.java create mode 100644 src/main/java/it/cnr/isti/epasmed/web/rest/errors/ErrorConstants.java create mode 100644 src/main/java/it/cnr/isti/epasmed/web/rest/errors/ExceptionTranslator.java create mode 100644 src/main/java/it/cnr/isti/epasmed/web/rest/errors/FieldErrorVM.java create mode 100644 src/main/java/it/cnr/isti/epasmed/web/rest/errors/InvalidPasswordException.java create mode 100644 src/main/java/it/cnr/isti/epasmed/web/rest/errors/LoginAlreadyUsedException.java create mode 100644 src/main/java/it/cnr/isti/epasmed/web/rest/errors/package-info.java create mode 100644 src/main/java/it/cnr/isti/epasmed/web/rest/package-info.java create mode 100644 src/main/java/it/cnr/isti/epasmed/web/rest/sistemainformativo/SIMasterLogResource.java create mode 100644 src/main/java/it/cnr/isti/epasmed/web/rest/vm/KeyAndPasswordVM.java create mode 100644 src/main/java/it/cnr/isti/epasmed/web/rest/vm/ManagedUserVM.java create mode 100644 src/main/java/it/cnr/isti/epasmed/web/rest/vm/package-info.java create mode 100644 src/main/javasecurity/java.security create mode 100644 src/main/jib/entrypoint.sh create mode 100644 src/main/resources/.h2.server.properties create mode 100644 src/main/resources/banner.txt create mode 100644 src/main/resources/config/application-dev.yml create mode 100644 src/main/resources/config/application-prod.yml create mode 100644 src/main/resources/config/application-tls.yml create mode 100644 src/main/resources/config/application.yml create mode 100644 src/main/resources/config/liquibase/epasmed/changelog/00000000000000_initial_schema.xml create mode 100644 src/main/resources/config/liquibase/epasmed/data/authority.csv create mode 100644 src/main/resources/config/liquibase/epasmed/data/tabssi.csv create mode 100644 src/main/resources/config/liquibase/epasmed/data/user.csv create mode 100644 src/main/resources/config/liquibase/epasmed/data/user_authority.csv create mode 100644 src/main/resources/config/liquibase/epasmed/master.xml create mode 100644 src/main/resources/config/tls/keystore.p12 create mode 100644 src/main/resources/i18n/messages.properties create mode 100644 src/main/resources/logback-spring.xml create mode 100644 src/main/resources/swagger/api.yml create mode 100644 src/main/resources/templates/error.html create mode 100644 src/main/resources/templates/mail/activationEmail.html create mode 100644 src/main/resources/templates/mail/creationEmail.html create mode 100644 src/main/resources/templates/mail/passwordResetEmail.html create mode 100644 src/main/webapp/404.html create mode 100644 src/main/webapp/WEB-INF/web.xml create mode 100644 src/main/webapp/app/account/account.module.ts create mode 100644 src/main/webapp/app/account/account.route.ts create mode 100644 src/main/webapp/app/account/activate/activate.component.html create mode 100644 src/main/webapp/app/account/activate/activate.component.ts create mode 100644 src/main/webapp/app/account/activate/activate.route.ts create mode 100644 src/main/webapp/app/account/activate/activate.service.ts create mode 100644 src/main/webapp/app/account/password-reset/finish/password-reset-finish.component.html create mode 100644 src/main/webapp/app/account/password-reset/finish/password-reset-finish.component.ts create mode 100644 src/main/webapp/app/account/password-reset/finish/password-reset-finish.route.ts create mode 100644 src/main/webapp/app/account/password-reset/finish/password-reset-finish.service.ts create mode 100644 src/main/webapp/app/account/password-reset/init/password-reset-init.component.html create mode 100644 src/main/webapp/app/account/password-reset/init/password-reset-init.component.ts create mode 100644 src/main/webapp/app/account/password-reset/init/password-reset-init.route.ts create mode 100644 src/main/webapp/app/account/password-reset/init/password-reset-init.service.ts create mode 100644 src/main/webapp/app/account/password/password-strength-bar.component.ts create mode 100644 src/main/webapp/app/account/password/password-strength-bar.scss create mode 100644 src/main/webapp/app/account/password/password.component.html create mode 100644 src/main/webapp/app/account/password/password.component.ts create mode 100644 src/main/webapp/app/account/password/password.route.ts create mode 100644 src/main/webapp/app/account/password/password.service.ts create mode 100644 src/main/webapp/app/account/register/register.component.html create mode 100644 src/main/webapp/app/account/register/register.component.ts create mode 100644 src/main/webapp/app/account/register/register.route.ts create mode 100644 src/main/webapp/app/account/register/register.service.ts create mode 100644 src/main/webapp/app/account/sessions/session.model.ts create mode 100644 src/main/webapp/app/account/sessions/sessions.component.html create mode 100644 src/main/webapp/app/account/sessions/sessions.component.ts create mode 100644 src/main/webapp/app/account/sessions/sessions.route.ts create mode 100644 src/main/webapp/app/account/sessions/sessions.service.ts create mode 100644 src/main/webapp/app/account/settings/settings.component.html create mode 100644 src/main/webapp/app/account/settings/settings.component.ts create mode 100644 src/main/webapp/app/account/settings/settings.route.ts create mode 100644 src/main/webapp/app/admin/admin-routing.module.ts create mode 100644 src/main/webapp/app/admin/audits/audit-data.model.ts create mode 100644 src/main/webapp/app/admin/audits/audit.model.ts create mode 100644 src/main/webapp/app/admin/audits/audits.component.html create mode 100644 src/main/webapp/app/admin/audits/audits.component.ts create mode 100644 src/main/webapp/app/admin/audits/audits.module.ts create mode 100644 src/main/webapp/app/admin/audits/audits.route.ts create mode 100644 src/main/webapp/app/admin/audits/audits.service.ts create mode 100644 src/main/webapp/app/admin/configuration/configuration.component.html create mode 100644 src/main/webapp/app/admin/configuration/configuration.component.ts create mode 100644 src/main/webapp/app/admin/configuration/configuration.module.ts create mode 100644 src/main/webapp/app/admin/configuration/configuration.route.ts create mode 100644 src/main/webapp/app/admin/configuration/configuration.service.ts create mode 100644 src/main/webapp/app/admin/docs/docs.component.html create mode 100644 src/main/webapp/app/admin/docs/docs.component.ts create mode 100644 src/main/webapp/app/admin/docs/docs.module.ts create mode 100644 src/main/webapp/app/admin/docs/docs.route.ts create mode 100644 src/main/webapp/app/admin/docs/docs.scss create mode 100644 src/main/webapp/app/admin/health/health-modal.component.html create mode 100644 src/main/webapp/app/admin/health/health-modal.component.ts create mode 100644 src/main/webapp/app/admin/health/health.component.html create mode 100644 src/main/webapp/app/admin/health/health.component.ts create mode 100644 src/main/webapp/app/admin/health/health.module.ts create mode 100644 src/main/webapp/app/admin/health/health.route.ts create mode 100644 src/main/webapp/app/admin/health/health.service.ts create mode 100644 src/main/webapp/app/admin/logs/log.model.ts create mode 100644 src/main/webapp/app/admin/logs/logs.component.html create mode 100644 src/main/webapp/app/admin/logs/logs.component.ts create mode 100644 src/main/webapp/app/admin/logs/logs.module.ts create mode 100644 src/main/webapp/app/admin/logs/logs.route.ts create mode 100644 src/main/webapp/app/admin/logs/logs.service.ts create mode 100644 src/main/webapp/app/admin/metrics/metrics.component.html create mode 100644 src/main/webapp/app/admin/metrics/metrics.component.ts create mode 100644 src/main/webapp/app/admin/metrics/metrics.module.ts create mode 100644 src/main/webapp/app/admin/metrics/metrics.route.ts create mode 100644 src/main/webapp/app/admin/metrics/metrics.service.ts create mode 100644 src/main/webapp/app/admin/user-management/user-management-delete-dialog.component.html create mode 100644 src/main/webapp/app/admin/user-management/user-management-delete-dialog.component.ts create mode 100644 src/main/webapp/app/admin/user-management/user-management-detail.component.html create mode 100644 src/main/webapp/app/admin/user-management/user-management-detail.component.ts create mode 100644 src/main/webapp/app/admin/user-management/user-management-update.component.html create mode 100644 src/main/webapp/app/admin/user-management/user-management-update.component.ts create mode 100644 src/main/webapp/app/admin/user-management/user-management.component.html create mode 100644 src/main/webapp/app/admin/user-management/user-management.component.ts create mode 100644 src/main/webapp/app/admin/user-management/user-management.module.ts create mode 100644 src/main/webapp/app/admin/user-management/user-management.route.ts create mode 100644 src/main/webapp/app/app-routing.module.ts create mode 100644 src/main/webapp/app/app.constants.ts create mode 100644 src/main/webapp/app/app.main.ts create mode 100644 src/main/webapp/app/app.module.ts create mode 100644 src/main/webapp/app/blocks/config/prod.config.ts create mode 100644 src/main/webapp/app/blocks/config/uib-pagination.config.ts create mode 100644 src/main/webapp/app/blocks/interceptor/auth-expired.interceptor.ts create mode 100644 src/main/webapp/app/blocks/interceptor/errorhandler.interceptor.ts create mode 100644 src/main/webapp/app/blocks/interceptor/notification.interceptor.ts create mode 100644 src/main/webapp/app/core/auth/account.service.ts create mode 100644 src/main/webapp/app/core/auth/auth-session.service.ts create mode 100644 src/main/webapp/app/core/auth/csrf.service.ts create mode 100644 src/main/webapp/app/core/auth/state-storage.service.ts create mode 100644 src/main/webapp/app/core/auth/user-route-access-service.ts create mode 100644 src/main/webapp/app/core/core.module.ts create mode 100644 src/main/webapp/app/core/icons/font-awesome-icons.ts create mode 100644 src/main/webapp/app/core/login/login-modal.service.ts create mode 100644 src/main/webapp/app/core/login/login.model.ts create mode 100644 src/main/webapp/app/core/login/login.service.ts create mode 100644 src/main/webapp/app/core/user/account.model.ts create mode 100644 src/main/webapp/app/core/user/user.model.ts create mode 100644 src/main/webapp/app/core/user/user.service.ts create mode 100644 src/main/webapp/app/entities/entity.module.ts create mode 100644 src/main/webapp/app/entities/tabssi/tabssi-delete-dialog.component.html create mode 100644 src/main/webapp/app/entities/tabssi/tabssi-delete-dialog.component.ts create mode 100644 src/main/webapp/app/entities/tabssi/tabssi-detail.component.html create mode 100644 src/main/webapp/app/entities/tabssi/tabssi-detail.component.ts create mode 100644 src/main/webapp/app/entities/tabssi/tabssi-update.component.html create mode 100644 src/main/webapp/app/entities/tabssi/tabssi-update.component.ts create mode 100644 src/main/webapp/app/entities/tabssi/tabssi.component.html create mode 100644 src/main/webapp/app/entities/tabssi/tabssi.component.ts create mode 100644 src/main/webapp/app/entities/tabssi/tabssi.model.ts create mode 100644 src/main/webapp/app/entities/tabssi/tabssi.module.ts create mode 100644 src/main/webapp/app/entities/tabssi/tabssi.route.ts create mode 100644 src/main/webapp/app/entities/tabssi/tabssi.service.ts create mode 100644 src/main/webapp/app/home/home.component.html create mode 100644 src/main/webapp/app/home/home.component.ts create mode 100644 src/main/webapp/app/home/home.module.ts create mode 100644 src/main/webapp/app/home/home.route.ts create mode 100644 src/main/webapp/app/home/home.scss create mode 100644 src/main/webapp/app/layouts/error/error.component.html create mode 100644 src/main/webapp/app/layouts/error/error.component.ts create mode 100644 src/main/webapp/app/layouts/error/error.route.ts create mode 100644 src/main/webapp/app/layouts/footer/footer.component.html create mode 100644 src/main/webapp/app/layouts/footer/footer.component.ts create mode 100644 src/main/webapp/app/layouts/main/main.component.html create mode 100644 src/main/webapp/app/layouts/main/main.component.ts create mode 100644 src/main/webapp/app/layouts/navbar/navbar.component.html create mode 100644 src/main/webapp/app/layouts/navbar/navbar.component.ts create mode 100644 src/main/webapp/app/layouts/navbar/navbar.route.ts create mode 100644 src/main/webapp/app/layouts/navbar/navbar.scss create mode 100644 src/main/webapp/app/layouts/profiles/page-ribbon.component.ts create mode 100644 src/main/webapp/app/layouts/profiles/page-ribbon.scss create mode 100644 src/main/webapp/app/layouts/profiles/profile-info.model.ts create mode 100644 src/main/webapp/app/layouts/profiles/profile.service.ts create mode 100644 src/main/webapp/app/polyfills.ts create mode 100644 src/main/webapp/app/shared/alert/alert-error.component.ts create mode 100644 src/main/webapp/app/shared/alert/alert-error.model.ts create mode 100644 src/main/webapp/app/shared/alert/alert.component.ts create mode 100644 src/main/webapp/app/shared/auth/has-any-authority.directive.ts create mode 100644 src/main/webapp/app/shared/constants/authority.constants.ts create mode 100644 src/main/webapp/app/shared/constants/error.constants.ts create mode 100644 src/main/webapp/app/shared/constants/input.constants.ts create mode 100644 src/main/webapp/app/shared/constants/pagination.constants.ts create mode 100644 src/main/webapp/app/shared/login/login.component.html create mode 100644 src/main/webapp/app/shared/login/login.component.ts create mode 100644 src/main/webapp/app/shared/shared-libs.module.ts create mode 100644 src/main/webapp/app/shared/shared.module.ts create mode 100644 src/main/webapp/app/shared/util/datepicker-adapter.ts create mode 100644 src/main/webapp/app/shared/util/request-util.ts create mode 100644 src/main/webapp/app/vendor.ts create mode 100644 src/main/webapp/content/css/loading.css create mode 100644 src/main/webapp/content/images/ePASMedLogo.png create mode 100644 src/main/webapp/content/images/plumber_0.svg create mode 100644 src/main/webapp/content/images/plumber_0_head_192.png create mode 100644 src/main/webapp/content/images/plumber_0_head_256.png create mode 100644 src/main/webapp/content/images/plumber_0_head_384.png create mode 100644 src/main/webapp/content/images/plumber_0_head_512.png create mode 100644 src/main/webapp/content/scss/_bootstrap-variables.scss create mode 100644 src/main/webapp/content/scss/global.scss create mode 100644 src/main/webapp/content/scss/vendor.scss create mode 100644 src/main/webapp/favicon.ico create mode 100644 src/main/webapp/index.html create mode 100644 src/main/webapp/manifest.webapp create mode 100644 src/main/webapp/robots.txt create mode 100644 src/main/webapp/swagger-ui/dist/images/throbber.gif create mode 100644 src/main/webapp/swagger-ui/index.html create mode 100644 src/test/java/it/cnr/isti/epasmed/ArchTest.java create mode 100644 src/test/java/it/cnr/isti/epasmed/config/NoOpMailConfiguration.java create mode 100644 src/test/java/it/cnr/isti/epasmed/config/StaticResourcesWebConfigurerTest.java create mode 100644 src/test/java/it/cnr/isti/epasmed/config/WebConfigurerTest.java create mode 100644 src/test/java/it/cnr/isti/epasmed/config/WebConfigurerTestController.java create mode 100644 src/test/java/it/cnr/isti/epasmed/repository/CustomAuditEventRepositoryIT.java create mode 100644 src/test/java/it/cnr/isti/epasmed/security/DomainUserDetailsServiceIT.java create mode 100644 src/test/java/it/cnr/isti/epasmed/security/SecurityUtilsUnitTest.java create mode 100644 src/test/java/it/cnr/isti/epasmed/service/AuditEventServiceIT.java create mode 100644 src/test/java/it/cnr/isti/epasmed/service/MailServiceIT.java create mode 100644 src/test/java/it/cnr/isti/epasmed/service/UserServiceIT.java create mode 100644 src/test/java/it/cnr/isti/epasmed/service/mapper/UserMapperTest.java create mode 100644 src/test/java/it/cnr/isti/epasmed/web/rest/AccountResourceIT.java create mode 100644 src/test/java/it/cnr/isti/epasmed/web/rest/AuditResourceIT.java create mode 100644 src/test/java/it/cnr/isti/epasmed/web/rest/ClientForwardControllerTest.java create mode 100644 src/test/java/it/cnr/isti/epasmed/web/rest/SyncResourceIT.java create mode 100644 src/test/java/it/cnr/isti/epasmed/web/rest/TestUtil.java create mode 100644 src/test/java/it/cnr/isti/epasmed/web/rest/UserResourceIT.java create mode 100644 src/test/java/it/cnr/isti/epasmed/web/rest/WithUnauthenticatedMockUser.java create mode 100644 src/test/java/it/cnr/isti/epasmed/web/rest/epas/EPASAffiliationsResourceIT.java create mode 100644 src/test/java/it/cnr/isti/epasmed/web/rest/epas/EPASContractsResourceIT.java create mode 100644 src/test/java/it/cnr/isti/epasmed/web/rest/epas/EPASGroupsResourceIT.java create mode 100644 src/test/java/it/cnr/isti/epasmed/web/rest/epas/EPASPersonsResourceIT.java create mode 100644 src/test/java/it/cnr/isti/epasmed/web/rest/epas/EPASWorkingTimeTypesResourceIT.java create mode 100644 src/test/java/it/cnr/isti/epasmed/web/rest/errors/ExceptionTranslatorIT.java create mode 100644 src/test/java/it/cnr/isti/epasmed/web/rest/errors/ExceptionTranslatorTestController.java create mode 100644 src/test/javascript/jest-global-mocks.ts create mode 100644 src/test/javascript/jest.conf.js create mode 100644 src/test/javascript/jest.ts create mode 100644 src/test/javascript/spec/app/account/activate/activate.component.spec.ts create mode 100644 src/test/javascript/spec/app/account/password-reset/finish/password-reset-finish.component.spec.ts create mode 100644 src/test/javascript/spec/app/account/password-reset/init/password-reset-init.component.spec.ts create mode 100644 src/test/javascript/spec/app/account/password/password-strength-bar.component.spec.ts create mode 100644 src/test/javascript/spec/app/account/password/password.component.spec.ts create mode 100644 src/test/javascript/spec/app/account/register/register.component.spec.ts create mode 100644 src/test/javascript/spec/app/account/sessions/sessions.component.spec.ts create mode 100644 src/test/javascript/spec/app/account/settings/settings.component.spec.ts create mode 100644 src/test/javascript/spec/app/admin/audits/audits.component.spec.ts create mode 100644 src/test/javascript/spec/app/admin/audits/audits.service.spec.ts create mode 100644 src/test/javascript/spec/app/admin/configuration/configuration.component.spec.ts create mode 100644 src/test/javascript/spec/app/admin/configuration/configuration.service.spec.ts create mode 100644 src/test/javascript/spec/app/admin/health/health.component.spec.ts create mode 100644 src/test/javascript/spec/app/admin/logs/logs.component.spec.ts create mode 100644 src/test/javascript/spec/app/admin/logs/logs.service.spec.ts create mode 100644 src/test/javascript/spec/app/admin/metrics/metrics.component.spec.ts create mode 100644 src/test/javascript/spec/app/admin/metrics/metrics.service.spec.ts create mode 100644 src/test/javascript/spec/app/admin/user-management/user-management-delete-dialog.component.spec.ts create mode 100644 src/test/javascript/spec/app/admin/user-management/user-management-detail.component.spec.ts create mode 100644 src/test/javascript/spec/app/admin/user-management/user-management-update.component.spec.ts create mode 100644 src/test/javascript/spec/app/admin/user-management/user-management.component.spec.ts create mode 100644 src/test/javascript/spec/app/core/login/login-modal.service.spec.ts create mode 100644 src/test/javascript/spec/app/core/user/account.service.spec.ts create mode 100644 src/test/javascript/spec/app/core/user/user.service.spec.ts create mode 100644 src/test/javascript/spec/app/home/home.component.spec.ts create mode 100644 src/test/javascript/spec/app/layouts/main/main.component.spec.ts create mode 100644 src/test/javascript/spec/app/layouts/navbar/navbar.component.spec.ts create mode 100644 src/test/javascript/spec/app/shared/alert/alert-error.component.spec.ts create mode 100644 src/test/javascript/spec/app/shared/alert/alert.component.spec.ts create mode 100644 src/test/javascript/spec/app/shared/login/login.component.spec.ts create mode 100644 src/test/javascript/spec/helpers/mock-account.service.ts create mode 100644 src/test/javascript/spec/helpers/mock-active-modal.service.ts create mode 100644 src/test/javascript/spec/helpers/mock-alert.service.ts create mode 100644 src/test/javascript/spec/helpers/mock-event-manager.service.ts create mode 100644 src/test/javascript/spec/helpers/mock-login-modal.service.ts create mode 100644 src/test/javascript/spec/helpers/mock-login.service.ts create mode 100644 src/test/javascript/spec/helpers/mock-route.service.ts create mode 100644 src/test/javascript/spec/helpers/mock-state-storage.service.ts create mode 100644 src/test/javascript/spec/helpers/spyobject.ts create mode 100644 src/test/javascript/spec/test.module.ts create mode 100644 src/test/resources/config/application-testcontainers.yml create mode 100644 src/test/resources/config/application.yml create mode 100644 src/test/resources/i18n/messages_en.properties create mode 100644 src/test/resources/it/cnr/isti/epasmed/web/rest/epas/affiliazioniRepartiISTI.csv create mode 100644 src/test/resources/it/cnr/isti/epasmed/web/rest/epas/affiliazioniRepartiISTIRimanenti.csv create mode 100644 src/test/resources/it/cnr/isti/epasmed/web/rest/epas/personale.csv create mode 100644 src/test/resources/it/cnr/isti/epasmed/web/rest/epas/posizioniISTI.csv create mode 100644 src/test/resources/logback.xml create mode 100644 src/test/resources/templates/mail/testEmail.html create mode 100644 tsconfig.app.json create mode 100644 tsconfig.base.json create mode 100644 tsconfig.json create mode 100644 tslint.json create mode 100644 webpack/ePASMedLogo.png create mode 100755 webpack/logo-jhipster.png create mode 100644 webpack/utils.js create mode 100644 webpack/webpack.common.js create mode 100644 webpack/webpack.dev.js create mode 100644 webpack/webpack.prod.js diff --git a/.editorconfig b/.editorconfig new file mode 100644 index 0000000..0439866 --- /dev/null +++ b/.editorconfig @@ -0,0 +1,23 @@ +# EditorConfig helps developers define and maintain consistent +# coding styles between different editors and IDEs +# editorconfig.org + +root = true + +[*] + +# We recommend you to keep these unchanged +end_of_line = lf +charset = utf-8 +trim_trailing_whitespace = true +insert_final_newline = true + +# Change these settings to your own preference +indent_style = space +indent_size = 4 + +[*.{ts,tsx,js,jsx,json,css,scss,yml}] +indent_size = 2 + +[*.md] +trim_trailing_whitespace = false diff --git a/.eslintignore b/.eslintignore new file mode 100644 index 0000000..c8d1f55 --- /dev/null +++ b/.eslintignore @@ -0,0 +1,9 @@ +node_modules/ +src/main/docker/ +src/test/javascript/protractor.conf.js +src/test/javascript/jest.conf.js +webpack/ +target/ +build/ +node/ +postcss.config.js diff --git a/.eslintrc.json b/.eslintrc.json new file mode 100644 index 0000000..43c1cf2 --- /dev/null +++ b/.eslintrc.json @@ -0,0 +1,24 @@ +{ + "plugins": ["@typescript-eslint/tslint"], + "extends": ["jhipster"], + "parserOptions": { + "project": "./tsconfig.base.json" + }, + "rules": { + "@typescript-eslint/tslint/config": [ + "error", + { + "lintFile": "./tslint.json" + } + ], + "@typescript-eslint/no-unused-vars": [ + "warn", + { + "vars": "all", + "args": "after-used", + "ignoreRestSiblings": false + } + ], + "@typescript-eslint/no-non-null-assertion": "off" + } +} diff --git a/.gitattributes b/.gitattributes new file mode 100644 index 0000000..ca61722 --- /dev/null +++ b/.gitattributes @@ -0,0 +1,150 @@ +# This file is inspired by https://github.com/alexkaratarakis/gitattributes +# +# Auto detect text files and perform LF normalization +# http://davidlaing.com/2012/09/19/customise-your-gitattributes-to-become-a-git-ninja/ +* text=auto + +# The above will handle all files NOT found below +# These files are text and should be normalized (Convert crlf => lf) + +*.bat text eol=crlf +*.cmd text eol=crlf +*.ps1 text eol=crlf +*.coffee text +*.css text +*.cql text +*.df text +*.ejs text +*.html text +*.java text +*.js text +*.json text +*.less text +*.properties text +*.sass text +*.scss text +*.sh text eol=lf +*.sql text +*.txt text +*.ts text +*.xml text +*.yaml text +*.yml text + +# Documents +*.doc diff=astextplain +*.DOC diff=astextplain +*.docx diff=astextplain +*.DOCX diff=astextplain +*.dot diff=astextplain +*.DOT diff=astextplain +*.pdf diff=astextplain +*.PDF diff=astextplain +*.rtf diff=astextplain +*.RTF diff=astextplain +*.markdown text +*.md text +*.adoc text +*.textile text +*.mustache text +*.csv text +*.tab text +*.tsv text +*.txt text +AUTHORS text +CHANGELOG text +CHANGES text +CONTRIBUTING text +COPYING text +copyright text +*COPYRIGHT* text +INSTALL text +license text +LICENSE text +NEWS text +readme text +*README* text +TODO text + +# Graphics +*.png binary +*.jpg binary +*.jpeg binary +*.gif binary +*.tif binary +*.tiff binary +*.ico binary +# SVG treated as an asset (binary) by default. If you want to treat it as text, +# comment-out the following line and uncomment the line after. +*.svg binary +#*.svg text +*.eps binary + +# These files are binary and should be left untouched +# (binary is a macro for -text -diff) +*.class binary +*.jar binary +*.war binary + +## LINTERS +.csslintrc text +.eslintrc text +.jscsrc text +.jshintrc text +.jshintignore text +.stylelintrc text + +## CONFIGS +*.conf text +*.config text +.editorconfig text +.gitattributes text +.gitconfig text +.gitignore text +.htaccess text +*.npmignore text + +## HEROKU +Procfile text +.slugignore text + +## AUDIO +*.kar binary +*.m4a binary +*.mid binary +*.midi binary +*.mp3 binary +*.ogg binary +*.ra binary + +## VIDEO +*.3gpp binary +*.3gp binary +*.as binary +*.asf binary +*.asx binary +*.fla binary +*.flv binary +*.m4v binary +*.mng binary +*.mov binary +*.mp4 binary +*.mpeg binary +*.mpg binary +*.swc binary +*.swf binary +*.webm binary + +## ARCHIVES +*.7z binary +*.gz binary +*.rar binary +*.tar binary +*.zip binary + +## FONTS +*.ttf binary +*.eot binary +*.otf binary +*.woff binary +*.woff2 binary diff --git a/.gitignore b/.gitignore new file mode 100644 index 0000000..2391902 --- /dev/null +++ b/.gitignore @@ -0,0 +1,155 @@ +###################### +# Project Specific +###################### +/src/main/webapp/content/css/main.css +/target/classes/static/** +/src/test/javascript/coverage/ + +###################### +# Node +###################### +/node/ +node_tmp/ +node_modules/ +npm-debug.log.* +/.awcache/* +/.cache-loader/* + +###################### +# SASS +###################### +.sass-cache/ + +###################### +# Eclipse +###################### +*.pydevproject +.project +.metadata +tmp/ +tmp/**/* +*.tmp +*.bak +*.swp +*~.nib +local.properties +.classpath +.settings/ +.loadpath +.factorypath +/src/main/resources/rebel.xml + +# External tool builders +.externalToolBuilders/** + +# Locally stored "Eclipse launch configurations" +*.launch + +# CDT-specific +.cproject + +# PDT-specific +.buildpath + +# STS-specific +/.sts4-cache/* + +###################### +# IntelliJ +###################### +.idea/ +*.iml +*.iws +*.ipr +*.ids +*.orig +classes/ +out/ + +###################### +# Visual Studio Code +###################### +.vscode/ + +###################### +# Maven +###################### +/log/ +/target/ + +###################### +# Gradle +###################### +.gradle/ +/build/ + +###################### +# Package Files +###################### +*.jar +*.war +*.ear +*.db + +###################### +# Windows +###################### +# Windows image file caches +Thumbs.db + +# Folder config file +Desktop.ini + +###################### +# Mac OSX +###################### +.DS_Store +.svn + +# Thumbnails +._* + +# Files that might appear on external disk +.Spotlight-V100 +.Trashes + +###################### +# Directories +###################### +/bin/ +/deploy/ + +###################### +# Logs +###################### +*.log* + +###################### +# Others +###################### +*.class +*.*~ +*~ +.merge_file* + +###################### +# Gradle Wrapper +###################### +!gradle/wrapper/gradle-wrapper.jar + +###################### +# Maven Wrapper +###################### +!.mvn/wrapper/maven-wrapper.jar + +###################### +# ESLint +###################### +.eslintcache +/.attach_pid1184 +/.attach_pid14281 +/.attach_pid27581 +/.attach_pid28007 +/.attach_pid29719 +/.attach_pid31807 +/.attach_pid4789 diff --git a/.huskyrc b/.huskyrc new file mode 100644 index 0000000..4d077c8 --- /dev/null +++ b/.huskyrc @@ -0,0 +1,5 @@ +{ + "hooks": { + "pre-commit": "lint-staged" + } +} diff --git a/.lintstagedrc.js b/.lintstagedrc.js new file mode 100644 index 0000000..00c792a --- /dev/null +++ b/.lintstagedrc.js @@ -0,0 +1,3 @@ +module.exports = { + '{,src/**/}*.{json,md,yml,ts,css,scss}': ['prettier --write', 'git add'] +}; diff --git a/.mvn/wrapper/MavenWrapperDownloader.java b/.mvn/wrapper/MavenWrapperDownloader.java new file mode 100644 index 0000000..b901097 --- /dev/null +++ b/.mvn/wrapper/MavenWrapperDownloader.java @@ -0,0 +1,117 @@ +/* + * Copyright 2007-present the original author or authors. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ +import java.net.*; +import java.io.*; +import java.nio.channels.*; +import java.util.Properties; + +public class MavenWrapperDownloader { + + private static final String WRAPPER_VERSION = "0.5.6"; + /** + * Default URL to download the maven-wrapper.jar from, if no 'downloadUrl' is provided. + */ + private static final String DEFAULT_DOWNLOAD_URL = "https://repo.maven.apache.org/maven2/io/takari/maven-wrapper/" + + WRAPPER_VERSION + "/maven-wrapper-" + WRAPPER_VERSION + ".jar"; + + /** + * Path to the maven-wrapper.properties file, which might contain a downloadUrl property to + * use instead of the default one. + */ + private static final String MAVEN_WRAPPER_PROPERTIES_PATH = + ".mvn/wrapper/maven-wrapper.properties"; + + /** + * Path where the maven-wrapper.jar will be saved to. + */ + private static final String MAVEN_WRAPPER_JAR_PATH = + ".mvn/wrapper/maven-wrapper.jar"; + + /** + * Name of the property which should be used to override the default download url for the wrapper. + */ + private static final String PROPERTY_NAME_WRAPPER_URL = "wrapperUrl"; + + public static void main(String args[]) { + System.out.println("- Downloader started"); + File baseDirectory = new File(args[0]); + System.out.println("- Using base directory: " + baseDirectory.getAbsolutePath()); + + // If the maven-wrapper.properties exists, read it and check if it contains a custom + // wrapperUrl parameter. + File mavenWrapperPropertyFile = new File(baseDirectory, MAVEN_WRAPPER_PROPERTIES_PATH); + String url = DEFAULT_DOWNLOAD_URL; + if(mavenWrapperPropertyFile.exists()) { + FileInputStream mavenWrapperPropertyFileInputStream = null; + try { + mavenWrapperPropertyFileInputStream = new FileInputStream(mavenWrapperPropertyFile); + Properties mavenWrapperProperties = new Properties(); + mavenWrapperProperties.load(mavenWrapperPropertyFileInputStream); + url = mavenWrapperProperties.getProperty(PROPERTY_NAME_WRAPPER_URL, url); + } catch (IOException e) { + System.out.println("- ERROR loading '" + MAVEN_WRAPPER_PROPERTIES_PATH + "'"); + } finally { + try { + if(mavenWrapperPropertyFileInputStream != null) { + mavenWrapperPropertyFileInputStream.close(); + } + } catch (IOException e) { + // Ignore ... + } + } + } + System.out.println("- Downloading from: " + url); + + File outputFile = new File(baseDirectory.getAbsolutePath(), MAVEN_WRAPPER_JAR_PATH); + if(!outputFile.getParentFile().exists()) { + if(!outputFile.getParentFile().mkdirs()) { + System.out.println( + "- ERROR creating output directory '" + outputFile.getParentFile().getAbsolutePath() + "'"); + } + } + System.out.println("- Downloading to: " + outputFile.getAbsolutePath()); + try { + downloadFileFromURL(url, outputFile); + System.out.println("Done"); + System.exit(0); + } catch (Throwable e) { + System.out.println("- Error downloading"); + e.printStackTrace(); + System.exit(1); + } + } + + private static void downloadFileFromURL(String urlString, File destination) throws Exception { + if (System.getenv("MVNW_USERNAME") != null && System.getenv("MVNW_PASSWORD") != null) { + String username = System.getenv("MVNW_USERNAME"); + char[] password = System.getenv("MVNW_PASSWORD").toCharArray(); + Authenticator.setDefault(new Authenticator() { + @Override + protected PasswordAuthentication getPasswordAuthentication() { + return new PasswordAuthentication(username, password); + } + }); + } + URL website = new URL(urlString); + ReadableByteChannel rbc; + rbc = Channels.newChannel(website.openStream()); + FileOutputStream fos = new FileOutputStream(destination); + fos.getChannel().transferFrom(rbc, 0, Long.MAX_VALUE); + fos.close(); + rbc.close(); + } + +} diff --git a/.mvn/wrapper/maven-wrapper.jar b/.mvn/wrapper/maven-wrapper.jar new file mode 100644 index 0000000000000000000000000000000000000000..2cc7d4a55c0cd0092912bf49ae38b3a9e3fd0054 GIT binary patch literal 50710 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 EfAhh=DF6Tf literal 0 HcmV?d00001 diff --git a/.mvn/wrapper/maven-wrapper.properties b/.mvn/wrapper/maven-wrapper.properties new file mode 100644 index 0000000..642d572 --- /dev/null +++ b/.mvn/wrapper/maven-wrapper.properties @@ -0,0 +1,2 @@ +distributionUrl=https://repo.maven.apache.org/maven2/org/apache/maven/apache-maven/3.6.3/apache-maven-3.6.3-bin.zip +wrapperUrl=https://repo.maven.apache.org/maven2/io/takari/maven-wrapper/0.5.6/maven-wrapper-0.5.6.jar diff --git a/.prettierignore b/.prettierignore new file mode 100644 index 0000000..ab0567f --- /dev/null +++ b/.prettierignore @@ -0,0 +1,8 @@ +node_modules +target +build +package-lock.json +.git +.mvn +gradle +.gradle diff --git a/.prettierrc b/.prettierrc new file mode 100644 index 0000000..b749286 --- /dev/null +++ b/.prettierrc @@ -0,0 +1,12 @@ +# Prettier configuration + +printWidth: 140 +singleQuote: true +tabWidth: 2 +useTabs: false + +# js and ts rules: +arrowParens: avoid + +# jsx and tsx rules: +jsxBracketSameLine: false diff --git a/.yo-rc.json b/.yo-rc.json new file mode 100644 index 0000000..fa4fa5f --- /dev/null +++ b/.yo-rc.json @@ -0,0 +1,39 @@ +{ + "generator-jhipster": { + "promptValues": { + "packageName": "it.cnr.isti.epasmed" + }, + "jhipsterVersion": "6.10.4", + "applicationType": "monolith", + "baseName": "epasmed", + "packageName": "it.cnr.isti.epasmed", + "packageFolder": "it/cnr/isti/epasmed", + "serverPort": "8080", + "authenticationType": "session", + "cacheProvider": "no", + "websocket": false, + "databaseType": "sql", + "devDatabaseType": "h2Disk", + "prodDatabaseType": "postgresql", + "searchEngine": false, + "messageBroker": false, + "serviceDiscoveryType": false, + "buildTool": "maven", + "enableSwaggerCodegen": true, + "rememberMeKey": "4130cb9045e4ce9479bec67eb3d78c04384bd6a04ccfb79510ff9aa8e1dd9233930cb396a534b842ab4537d3d21744127bc7", + "embeddableLaunchScript": false, + "useSass": true, + "clientPackageManager": "npm", + "clientFramework": "angularX", + "clientTheme": "none", + "clientThemeVariant": "", + "creationTimestamp": 1606233754700, + "testFrameworks": [], + "jhiPrefix": "jhi", + "entitySuffix": "", + "dtoSuffix": "DTO", + "otherModules": [], + "enableTranslation": false, + "blueprints": [] + } +} diff --git a/README-Orig.md b/README-Orig.md new file mode 100644 index 0000000..5551f75 --- /dev/null +++ b/README-Orig.md @@ -0,0 +1,254 @@ +# epasmed + +This application was generated using JHipster 6.10.4, you can find documentation and help at [https://www.jhipster.tech/documentation-archive/v6.10.4](https://www.jhipster.tech/documentation-archive/v6.10.4). + +## Development + +Before you can build this project, you must install and configure the following dependencies on your machine: + +1. [Node.js][]: We use Node to run a development web server and build the project. + Depending on your system, you can install Node either from source or as a pre-packaged bundle. + +After installing Node, you should be able to run the following command to install development tools. +You will only need to run this command when dependencies change in [package.json](package.json). + +``` +npm install +``` + +We use npm scripts and [Webpack][] as our build system. + +Run the following commands in two separate terminals to create a blissful development experience where your browser +auto-refreshes when files change on your hard drive. + +``` + +./mvnw + + +npm start +``` + +Npm is also used to manage CSS and JavaScript dependencies used in this application. You can upgrade dependencies by +specifying a newer version in [package.json](package.json). You can also run `npm update` and `npm install` to manage dependencies. +Add the `help` flag on any command to see how you can use it. For example, `npm help update`. + +The `npm run` command will list all of the scripts available to run for this project. + +### PWA Support + +JHipster ships with PWA (Progressive Web App) support, and it's turned off by default. One of the main components of a PWA is a service worker. + +The service worker initialization code is commented out by default. To enable it, uncomment the following code in `src/main/webapp/index.html`: + +```html + +``` + +Note: [Workbox](https://developers.google.com/web/tools/workbox/) powers JHipster's service worker. It dynamically generates the `service-worker.js` file. + +### Managing dependencies + +For example, to add [Leaflet][] library as a runtime dependency of your application, you would run following command: + +``` +npm install --save --save-exact leaflet +``` + +To benefit from TypeScript type definitions from [DefinitelyTyped][] repository in development, you would run following command: + +``` +npm install --save-dev --save-exact @types/leaflet +``` + +Then you would import the JS and CSS files specified in library's installation instructions so that [Webpack][] knows about them: +Edit [src/main/webapp/app/vendor.ts](src/main/webapp/app/vendor.ts) file: + +``` +import 'leaflet/dist/leaflet.js'; +``` + +Edit [src/main/webapp/content/scss/vendor.scss](src/main/webapp/content/scss/vendor.scss) file: + +``` +@import '~leaflet/dist/leaflet.css'; +``` + +Note: There are still a few other things remaining to do for Leaflet that we won't detail here. + +For further instructions on how to develop with JHipster, have a look at [Using JHipster in development][]. + +### Using Angular CLI + +You can also use [Angular CLI][] to generate some custom client code. + +For example, the following command: + +``` +ng generate component my-component +``` + +will generate few files: + +``` +create src/main/webapp/app/my-component/my-component.component.html +create src/main/webapp/app/my-component/my-component.component.ts +update src/main/webapp/app/app.module.ts +``` + +### Doing API-First development using openapi-generator + +[OpenAPI-Generator]() is configured for this application. You can generate API code from the `src/main/resources/swagger/api.yml` definition file by running: + +```bash +./mvnw generate-sources +``` + +Then implements the generated delegate classes with `@Service` classes. + +To edit the `api.yml` definition file, you can use a tool such as [Swagger-Editor](). Start a local instance of the swagger-editor using docker by running: `docker-compose -f src/main/docker/swagger-editor.yml up -d`. The editor will then be reachable at [http://localhost:7742](http://localhost:7742). + +Refer to [Doing API-First development][] for more details. + +## Building for production + +### Packaging as jar + +To build the final jar and optimize the epasmed application for production, run: + +``` + +./mvnw -Pprod clean verify + + +``` + +This will concatenate and minify the client CSS and JavaScript files. It will also modify `index.html` so it references these new files. +To ensure everything worked, run: + +``` + +java -jar target/*.jar + + +``` + +Then navigate to [http://localhost:8080](http://localhost:8080) in your browser. + +Refer to [Using JHipster in production][] for more details. + +### Packaging as war + +To package your application as a war in order to deploy it to an application server, run: + +``` + +./mvnw -Pprod,war clean verify + + +``` + +## Testing + +To launch your application's tests, run: + +``` +./mvnw verify +``` + +### Client tests + +Unit tests are run by [Jest][] and written with [Jasmine][]. They're located in [src/test/javascript/](src/test/javascript/) and can be run with: + +``` +npm test +``` + +For more information, refer to the [Running tests page][]. + +### Code quality + +Sonar is used to analyse code quality. You can start a local Sonar server (accessible on http://localhost:9001) with: + +``` +docker-compose -f src/main/docker/sonar.yml up -d +``` + +You can run a Sonar analysis with using the [sonar-scanner](https://docs.sonarqube.org/display/SCAN/Analyzing+with+SonarQube+Scanner) or by using the maven plugin. + +Then, run a Sonar analysis: + +``` +./mvnw -Pprod clean verify sonar:sonar +``` + +If you need to re-run the Sonar phase, please be sure to specify at least the `initialize` phase since Sonar properties are loaded from the sonar-project.properties file. + +``` +./mvnw initialize sonar:sonar +``` + +For more information, refer to the [Code quality page][]. + +## Using Docker to simplify development (optional) + +You can use Docker to improve your JHipster development experience. A number of docker-compose configuration are available in the [src/main/docker](src/main/docker) folder to launch required third party services. + +For example, to start a postgresql database in a docker container, run: + +``` +docker-compose -f src/main/docker/postgresql.yml up -d +``` + +To stop it and remove the container, run: + +``` +docker-compose -f src/main/docker/postgresql.yml down +``` + +You can also fully dockerize your application and all the services that it depends on. +To achieve this, first build a docker image of your app by running: + +``` +./mvnw -Pprod verify jib:dockerBuild +``` + +Then run: + +``` +docker-compose -f src/main/docker/app.yml up -d +``` + +For more information refer to [Using Docker and Docker-Compose][], this page also contains information on the docker-compose sub-generator (`jhipster docker-compose`), which is able to generate docker configurations for one or several JHipster applications. + +## Continuous Integration (optional) + +To configure CI for your project, run the ci-cd sub-generator (`jhipster ci-cd`), this will let you generate configuration files for a number of Continuous Integration systems. Consult the [Setting up Continuous Integration][] page for more information. + +[jhipster homepage and latest documentation]: https://www.jhipster.tech +[jhipster 6.10.4 archive]: https://www.jhipster.tech/documentation-archive/v6.10.4 +[using jhipster in development]: https://www.jhipster.tech/documentation-archive/v6.10.4/development/ +[using docker and docker-compose]: https://www.jhipster.tech/documentation-archive/v6.10.4/docker-compose +[using jhipster in production]: https://www.jhipster.tech/documentation-archive/v6.10.4/production/ +[running tests page]: https://www.jhipster.tech/documentation-archive/v6.10.4/running-tests/ +[code quality page]: https://www.jhipster.tech/documentation-archive/v6.10.4/code-quality/ +[setting up continuous integration]: https://www.jhipster.tech/documentation-archive/v6.10.4/setting-up-ci/ +[node.js]: https://nodejs.org/ +[yarn]: https://yarnpkg.org/ +[webpack]: https://webpack.github.io/ +[angular cli]: https://cli.angular.io/ +[browsersync]: https://www.browsersync.io/ +[jest]: https://facebook.github.io/jest/ +[jasmine]: https://jasmine.github.io/2.0/introduction.html +[protractor]: https://angular.github.io/protractor/ +[leaflet]: https://leafletjs.com/ +[definitelytyped]: https://definitelytyped.org/ +[openapi-generator]: https://openapi-generator.tech +[swagger-editor]: https://editor.swagger.io +[doing api-first development]: https://www.jhipster.tech/documentation-archive/v6.10.4/doing-api-first-development/ diff --git a/README.md b/README.md new file mode 100644 index 0000000..b17c228 --- /dev/null +++ b/README.md @@ -0,0 +1,20 @@ +# ePASMed + +ePASMed è il servizio di mediazione tra ePAS il software di rilevazione delle presenze e il Sistema Informativo ISTI + +## Structure of the project + +- The source code is present in the src folder. + +## Built With + +- [OpenJDK](https://openjdk.java.net/) - The JDK used +- [Maven](https://maven.apache.org/) - Dependency Management + +## Authors + +- **Giancarlo Panichi** ([ORCID](http://orcid.org/0000-0001-8375-6644)) - [ISTI-CNR Infrascience Group](http://nemis.isti.cnr.it/groups/infrascience) + +## License + +This project is licensed under the EUPL V.1.1 License - see the [LICENSE.md](LICENSE.md) file for details. diff --git a/angular.json b/angular.json new file mode 100644 index 0000000..cefc033 --- /dev/null +++ b/angular.json @@ -0,0 +1,36 @@ +{ + "$schema": "./node_modules/@angular/cli/lib/config/schema.json", + "version": 1, + "newProjectRoot": "projects", + "projects": { + "epasmed": { + "root": "", + "sourceRoot": "src/main/webapp", + "projectType": "application", + "schematics": { + "@schematics/angular:component": { + "skipTests": true, + "style": "scss" + }, + "@schematics/angular:directive": { + "skipTests": true + }, + "@schematics/angular:guard": { + "skipTests": true + }, + "@schematics/angular:pipe": { + "skipTests": true + }, + "@schematics/angular:service": { + "skipTests": true + } + }, + "prefix": "jhi", + "architect": {} + } + }, + "defaultProject": "epasmed", + "cli": { + "packageManager": "npm" + } +} diff --git a/checkstyle.xml b/checkstyle.xml new file mode 100644 index 0000000..6e3dbaf --- /dev/null +++ b/checkstyle.xml @@ -0,0 +1,21 @@ + + + + + + + + + + + + + + + diff --git a/mvnw b/mvnw new file mode 100755 index 0000000..41c0f0c --- /dev/null +++ b/mvnw @@ -0,0 +1,310 @@ +#!/bin/sh +# ---------------------------------------------------------------------------- +# Licensed to the Apache Software Foundation (ASF) under one +# or more contributor license agreements. See the NOTICE file +# distributed with this work for additional information +# regarding copyright ownership. The ASF licenses this file +# to you under the Apache License, Version 2.0 (the +# "License"); you may not use this file except in compliance +# with the License. You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, +# software distributed under the License is distributed on an +# "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY +# KIND, either express or implied. See the License for the +# specific language governing permissions and limitations +# under the License. +# ---------------------------------------------------------------------------- + +# ---------------------------------------------------------------------------- +# Maven Start Up Batch script +# +# Required ENV vars: +# ------------------ +# JAVA_HOME - location of a JDK home dir +# +# Optional ENV vars +# ----------------- +# M2_HOME - location of maven2's installed home dir +# MAVEN_OPTS - parameters passed to the Java VM when running Maven +# e.g. to debug Maven itself, use +# set MAVEN_OPTS=-Xdebug -Xrunjdwp:transport=dt_socket,server=y,suspend=y,address=8000 +# MAVEN_SKIP_RC - flag to disable loading of mavenrc files +# ---------------------------------------------------------------------------- + +if [ -z "$MAVEN_SKIP_RC" ] ; then + + if [ -f /etc/mavenrc ] ; then + . /etc/mavenrc + fi + + if [ -f "$HOME/.mavenrc" ] ; then + . "$HOME/.mavenrc" + fi + +fi + +# OS specific support. $var _must_ be set to either true or false. +cygwin=false; +darwin=false; +mingw=false +case "`uname`" in + CYGWIN*) cygwin=true ;; + MINGW*) mingw=true;; + Darwin*) darwin=true + # Use /usr/libexec/java_home if available, otherwise fall back to /Library/Java/Home + # See https://developer.apple.com/library/mac/qa/qa1170/_index.html + if [ -z "$JAVA_HOME" ]; then + if [ -x "/usr/libexec/java_home" ]; then + export JAVA_HOME="`/usr/libexec/java_home`" + else + export JAVA_HOME="/Library/Java/Home" + fi + fi + ;; +esac + +if [ -z "$JAVA_HOME" ] ; then + if [ -r /etc/gentoo-release ] ; then + JAVA_HOME=`java-config --jre-home` + fi +fi + +if [ -z "$M2_HOME" ] ; then + ## resolve links - $0 may be a link to maven's home + PRG="$0" + + # need this for relative symlinks + while [ -h "$PRG" ] ; do + ls=`ls -ld "$PRG"` + link=`expr "$ls" : '.*-> \(.*\)$'` + if expr "$link" : '/.*' > /dev/null; then + PRG="$link" + else + PRG="`dirname "$PRG"`/$link" + fi + done + + saveddir=`pwd` + + M2_HOME=`dirname "$PRG"`/.. + + # make it fully qualified + M2_HOME=`cd "$M2_HOME" && pwd` + + cd "$saveddir" + # echo Using m2 at $M2_HOME +fi + +# For Cygwin, ensure paths are in UNIX format before anything is touched +if $cygwin ; then + [ -n "$M2_HOME" ] && + M2_HOME=`cygpath --unix "$M2_HOME"` + [ -n "$JAVA_HOME" ] && + JAVA_HOME=`cygpath --unix "$JAVA_HOME"` + [ -n "$CLASSPATH" ] && + CLASSPATH=`cygpath --path --unix "$CLASSPATH"` +fi + +# For Mingw, ensure paths are in UNIX format before anything is touched +if $mingw ; then + [ -n "$M2_HOME" ] && + M2_HOME="`(cd "$M2_HOME"; pwd)`" + [ -n "$JAVA_HOME" ] && + JAVA_HOME="`(cd "$JAVA_HOME"; pwd)`" +fi + +if [ -z "$JAVA_HOME" ]; then + javaExecutable="`which javac`" + if [ -n "$javaExecutable" ] && ! [ "`expr \"$javaExecutable\" : '\([^ ]*\)'`" = "no" ]; then + # readlink(1) is not available as standard on Solaris 10. + readLink=`which readlink` + if [ ! `expr "$readLink" : '\([^ ]*\)'` = "no" ]; then + if $darwin ; then + javaHome="`dirname \"$javaExecutable\"`" + javaExecutable="`cd \"$javaHome\" && pwd -P`/javac" + else + javaExecutable="`readlink -f \"$javaExecutable\"`" + fi + javaHome="`dirname \"$javaExecutable\"`" + javaHome=`expr "$javaHome" : '\(.*\)/bin'` + JAVA_HOME="$javaHome" + export JAVA_HOME + fi + fi +fi + +if [ -z "$JAVACMD" ] ; then + if [ -n "$JAVA_HOME" ] ; then + if [ -x "$JAVA_HOME/jre/sh/java" ] ; then + # IBM's JDK on AIX uses strange locations for the executables + JAVACMD="$JAVA_HOME/jre/sh/java" + else + JAVACMD="$JAVA_HOME/bin/java" + fi + else + JAVACMD="`which java`" + fi +fi + +if [ ! -x "$JAVACMD" ] ; then + echo "Error: JAVA_HOME is not defined correctly." >&2 + echo " We cannot execute $JAVACMD" >&2 + exit 1 +fi + +if [ -z "$JAVA_HOME" ] ; then + echo "Warning: JAVA_HOME environment variable is not set." +fi + +CLASSWORLDS_LAUNCHER=org.codehaus.plexus.classworlds.launcher.Launcher + +# traverses directory structure from process work directory to filesystem root +# first directory with .mvn subdirectory is considered project base directory +find_maven_basedir() { + + if [ -z "$1" ] + then + echo "Path not specified to find_maven_basedir" + return 1 + fi + + basedir="$1" + wdir="$1" + while [ "$wdir" != '/' ] ; do + if [ -d "$wdir"/.mvn ] ; then + basedir=$wdir + break + fi + # workaround for JBEAP-8937 (on Solaris 10/Sparc) + if [ -d "${wdir}" ]; then + wdir=`cd "$wdir/.."; pwd` + fi + # end of workaround + done + echo "${basedir}" +} + +# concatenates all lines of a file +concat_lines() { + if [ -f "$1" ]; then + echo "$(tr -s '\n' ' ' < "$1")" + fi +} + +BASE_DIR=`find_maven_basedir "$(pwd)"` +if [ -z "$BASE_DIR" ]; then + exit 1; +fi + +########################################################################################## +# Extension to allow automatically downloading the maven-wrapper.jar from Maven-central +# This allows using the maven wrapper in projects that prohibit checking in binary data. +########################################################################################## +if [ -r "$BASE_DIR/.mvn/wrapper/maven-wrapper.jar" ]; then + if [ "$MVNW_VERBOSE" = true ]; then + echo "Found .mvn/wrapper/maven-wrapper.jar" + fi +else + if [ "$MVNW_VERBOSE" = true ]; then + echo "Couldn't find .mvn/wrapper/maven-wrapper.jar, downloading it ..." + fi + if [ -n "$MVNW_REPOURL" ]; then + jarUrl="$MVNW_REPOURL/io/takari/maven-wrapper/0.5.6/maven-wrapper-0.5.6.jar" + else + jarUrl="https://repo.maven.apache.org/maven2/io/takari/maven-wrapper/0.5.6/maven-wrapper-0.5.6.jar" + fi + while IFS="=" read key value; do + case "$key" in (wrapperUrl) jarUrl="$value"; break ;; + esac + done < "$BASE_DIR/.mvn/wrapper/maven-wrapper.properties" + if [ "$MVNW_VERBOSE" = true ]; then + echo "Downloading from: $jarUrl" + fi + wrapperJarPath="$BASE_DIR/.mvn/wrapper/maven-wrapper.jar" + if $cygwin; then + wrapperJarPath=`cygpath --path --windows "$wrapperJarPath"` + fi + + if command -v wget > /dev/null; then + if [ "$MVNW_VERBOSE" = true ]; then + echo "Found wget ... using wget" + fi + if [ -z "$MVNW_USERNAME" ] || [ -z "$MVNW_PASSWORD" ]; then + wget "$jarUrl" -O "$wrapperJarPath" + else + wget --http-user=$MVNW_USERNAME --http-password=$MVNW_PASSWORD "$jarUrl" -O "$wrapperJarPath" + fi + elif command -v curl > /dev/null; then + if [ "$MVNW_VERBOSE" = true ]; then + echo "Found curl ... using curl" + fi + if [ -z "$MVNW_USERNAME" ] || [ -z "$MVNW_PASSWORD" ]; then + curl -o "$wrapperJarPath" "$jarUrl" -f + else + curl --user $MVNW_USERNAME:$MVNW_PASSWORD -o "$wrapperJarPath" "$jarUrl" -f + fi + + else + if [ "$MVNW_VERBOSE" = true ]; then + echo "Falling back to using Java to download" + fi + javaClass="$BASE_DIR/.mvn/wrapper/MavenWrapperDownloader.java" + # For Cygwin, switch paths to Windows format before running javac + if $cygwin; then + javaClass=`cygpath --path --windows "$javaClass"` + fi + if [ -e "$javaClass" ]; then + if [ ! -e "$BASE_DIR/.mvn/wrapper/MavenWrapperDownloader.class" ]; then + if [ "$MVNW_VERBOSE" = true ]; then + echo " - Compiling MavenWrapperDownloader.java ..." + fi + # Compiling the Java class + ("$JAVA_HOME/bin/javac" "$javaClass") + fi + if [ -e "$BASE_DIR/.mvn/wrapper/MavenWrapperDownloader.class" ]; then + # Running the downloader + if [ "$MVNW_VERBOSE" = true ]; then + echo " - Running MavenWrapperDownloader.java ..." + fi + ("$JAVA_HOME/bin/java" -cp .mvn/wrapper MavenWrapperDownloader "$MAVEN_PROJECTBASEDIR") + fi + fi + fi +fi +########################################################################################## +# End of extension +########################################################################################## + +export MAVEN_PROJECTBASEDIR=${MAVEN_BASEDIR:-"$BASE_DIR"} +if [ "$MVNW_VERBOSE" = true ]; then + echo $MAVEN_PROJECTBASEDIR +fi +MAVEN_OPTS="$(concat_lines "$MAVEN_PROJECTBASEDIR/.mvn/jvm.config") $MAVEN_OPTS" + +# For Cygwin, switch paths to Windows format before running java +if $cygwin; then + [ -n "$M2_HOME" ] && + M2_HOME=`cygpath --path --windows "$M2_HOME"` + [ -n "$JAVA_HOME" ] && + JAVA_HOME=`cygpath --path --windows "$JAVA_HOME"` + [ -n "$CLASSPATH" ] && + CLASSPATH=`cygpath --path --windows "$CLASSPATH"` + [ -n "$MAVEN_PROJECTBASEDIR" ] && + MAVEN_PROJECTBASEDIR=`cygpath --path --windows "$MAVEN_PROJECTBASEDIR"` +fi + +# Provide a "standardized" way to retrieve the CLI args that will +# work with both Windows and non-Windows executions. +MAVEN_CMD_LINE_ARGS="$MAVEN_CONFIG $@" +export MAVEN_CMD_LINE_ARGS + +WRAPPER_LAUNCHER=org.apache.maven.wrapper.MavenWrapperMain + +exec "$JAVACMD" \ + $MAVEN_OPTS \ + -classpath "$MAVEN_PROJECTBASEDIR/.mvn/wrapper/maven-wrapper.jar" \ + "-Dmaven.home=${M2_HOME}" "-Dmaven.multiModuleProjectDirectory=${MAVEN_PROJECTBASEDIR}" \ + ${WRAPPER_LAUNCHER} $MAVEN_CONFIG "$@" diff --git a/mvnw.cmd b/mvnw.cmd new file mode 100644 index 0000000..8611571 --- /dev/null +++ b/mvnw.cmd @@ -0,0 +1,182 @@ +@REM ---------------------------------------------------------------------------- +@REM Licensed to the Apache Software Foundation (ASF) under one +@REM or more contributor license agreements. See the NOTICE file +@REM distributed with this work for additional information +@REM regarding copyright ownership. The ASF licenses this file +@REM to you under the Apache License, Version 2.0 (the +@REM "License"); you may not use this file except in compliance +@REM with the License. You may obtain a copy of the License at +@REM +@REM http://www.apache.org/licenses/LICENSE-2.0 +@REM +@REM Unless required by applicable law or agreed to in writing, +@REM software distributed under the License is distributed on an +@REM "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY +@REM KIND, either express or implied. See the License for the +@REM specific language governing permissions and limitations +@REM under the License. +@REM ---------------------------------------------------------------------------- + +@REM ---------------------------------------------------------------------------- +@REM Maven Start Up Batch script +@REM +@REM Required ENV vars: +@REM JAVA_HOME - location of a JDK home dir +@REM +@REM Optional ENV vars +@REM M2_HOME - location of maven2's installed home dir +@REM MAVEN_BATCH_ECHO - set to 'on' to enable the echoing of the batch commands +@REM MAVEN_BATCH_PAUSE - set to 'on' to wait for a keystroke before ending +@REM MAVEN_OPTS - parameters passed to the Java VM when running Maven +@REM e.g. to debug Maven itself, use +@REM set MAVEN_OPTS=-Xdebug -Xrunjdwp:transport=dt_socket,server=y,suspend=y,address=8000 +@REM MAVEN_SKIP_RC - flag to disable loading of mavenrc files +@REM ---------------------------------------------------------------------------- + +@REM Begin all REM lines with '@' in case MAVEN_BATCH_ECHO is 'on' +@echo off +@REM set title of command window +title %0 +@REM enable echoing by setting MAVEN_BATCH_ECHO to 'on' +@if "%MAVEN_BATCH_ECHO%" == "on" echo %MAVEN_BATCH_ECHO% + +@REM set %HOME% to equivalent of $HOME +if "%HOME%" == "" (set "HOME=%HOMEDRIVE%%HOMEPATH%") + +@REM Execute a user defined script before this one +if not "%MAVEN_SKIP_RC%" == "" goto skipRcPre +@REM check for pre script, once with legacy .bat ending and once with .cmd ending +if exist "%HOME%\mavenrc_pre.bat" call "%HOME%\mavenrc_pre.bat" +if exist "%HOME%\mavenrc_pre.cmd" call "%HOME%\mavenrc_pre.cmd" +:skipRcPre + +@setlocal + +set ERROR_CODE=0 + +@REM To isolate internal variables from possible post scripts, we use another setlocal +@setlocal + +@REM ==== START VALIDATION ==== +if not "%JAVA_HOME%" == "" goto OkJHome + +echo. +echo Error: JAVA_HOME not found in your environment. >&2 +echo Please set the JAVA_HOME variable in your environment to match the >&2 +echo location of your Java installation. >&2 +echo. +goto error + +:OkJHome +if exist "%JAVA_HOME%\bin\java.exe" goto init + +echo. +echo Error: JAVA_HOME is set to an invalid directory. >&2 +echo JAVA_HOME = "%JAVA_HOME%" >&2 +echo Please set the JAVA_HOME variable in your environment to match the >&2 +echo location of your Java installation. >&2 +echo. +goto error + +@REM ==== END VALIDATION ==== + +:init + +@REM Find the project base dir, i.e. the directory that contains the folder ".mvn". +@REM Fallback to current working directory if not found. + +set MAVEN_PROJECTBASEDIR=%MAVEN_BASEDIR% +IF NOT "%MAVEN_PROJECTBASEDIR%"=="" goto endDetectBaseDir + +set EXEC_DIR=%CD% +set WDIR=%EXEC_DIR% +:findBaseDir +IF EXIST "%WDIR%"\.mvn goto baseDirFound +cd .. +IF "%WDIR%"=="%CD%" goto baseDirNotFound +set WDIR=%CD% +goto findBaseDir + +:baseDirFound +set MAVEN_PROJECTBASEDIR=%WDIR% +cd "%EXEC_DIR%" +goto endDetectBaseDir + +:baseDirNotFound +set MAVEN_PROJECTBASEDIR=%EXEC_DIR% +cd "%EXEC_DIR%" + +:endDetectBaseDir + +IF NOT EXIST "%MAVEN_PROJECTBASEDIR%\.mvn\jvm.config" goto endReadAdditionalConfig + +@setlocal EnableExtensions EnableDelayedExpansion +for /F "usebackq delims=" %%a in ("%MAVEN_PROJECTBASEDIR%\.mvn\jvm.config") do set JVM_CONFIG_MAVEN_PROPS=!JVM_CONFIG_MAVEN_PROPS! %%a +@endlocal & set JVM_CONFIG_MAVEN_PROPS=%JVM_CONFIG_MAVEN_PROPS% + +:endReadAdditionalConfig + +SET MAVEN_JAVA_EXE="%JAVA_HOME%\bin\java.exe" +set WRAPPER_JAR="%MAVEN_PROJECTBASEDIR%\.mvn\wrapper\maven-wrapper.jar" +set WRAPPER_LAUNCHER=org.apache.maven.wrapper.MavenWrapperMain + +set DOWNLOAD_URL="https://repo.maven.apache.org/maven2/io/takari/maven-wrapper/0.5.6/maven-wrapper-0.5.6.jar" + +FOR /F "tokens=1,2 delims==" %%A IN ("%MAVEN_PROJECTBASEDIR%\.mvn\wrapper\maven-wrapper.properties") DO ( + IF "%%A"=="wrapperUrl" SET DOWNLOAD_URL=%%B +) + +@REM Extension to allow automatically downloading the maven-wrapper.jar from Maven-central +@REM This allows using the maven wrapper in projects that prohibit checking in binary data. +if exist %WRAPPER_JAR% ( + if "%MVNW_VERBOSE%" == "true" ( + echo Found %WRAPPER_JAR% + ) +) else ( + if not "%MVNW_REPOURL%" == "" ( + SET DOWNLOAD_URL="%MVNW_REPOURL%/io/takari/maven-wrapper/0.5.6/maven-wrapper-0.5.6.jar" + ) + if "%MVNW_VERBOSE%" == "true" ( + echo Couldn't find %WRAPPER_JAR%, downloading it ... + echo Downloading from: %DOWNLOAD_URL% + ) + + powershell -Command "&{"^ + "$webclient = new-object System.Net.WebClient;"^ + "if (-not ([string]::IsNullOrEmpty('%MVNW_USERNAME%') -and [string]::IsNullOrEmpty('%MVNW_PASSWORD%'))) {"^ + "$webclient.Credentials = new-object System.Net.NetworkCredential('%MVNW_USERNAME%', '%MVNW_PASSWORD%');"^ + "}"^ + "[Net.ServicePointManager]::SecurityProtocol = [Net.SecurityProtocolType]::Tls12; $webclient.DownloadFile('%DOWNLOAD_URL%', '%WRAPPER_JAR%')"^ + "}" + if "%MVNW_VERBOSE%" == "true" ( + echo Finished downloading %WRAPPER_JAR% + ) +) +@REM End of extension + +@REM Provide a "standardized" way to retrieve the CLI args that will +@REM work with both Windows and non-Windows executions. +set MAVEN_CMD_LINE_ARGS=%* + +%MAVEN_JAVA_EXE% %JVM_CONFIG_MAVEN_PROPS% %MAVEN_OPTS% %MAVEN_DEBUG_OPTS% -classpath %WRAPPER_JAR% "-Dmaven.multiModuleProjectDirectory=%MAVEN_PROJECTBASEDIR%" %WRAPPER_LAUNCHER% %MAVEN_CONFIG% %* +if ERRORLEVEL 1 goto error +goto end + +:error +set ERROR_CODE=1 + +:end +@endlocal & set ERROR_CODE=%ERROR_CODE% + +if not "%MAVEN_SKIP_RC%" == "" goto skipRcPost +@REM check for post script, once with legacy .bat ending and once with .cmd ending +if exist "%HOME%\mavenrc_post.bat" call "%HOME%\mavenrc_post.bat" +if exist "%HOME%\mavenrc_post.cmd" call "%HOME%\mavenrc_post.cmd" +:skipRcPost + +@REM pause the script if MAVEN_BATCH_PAUSE is set to 'on' +if "%MAVEN_BATCH_PAUSE%" == "on" pause + +if "%MAVEN_TERMINATE_CMD%" == "on" exit %ERROR_CODE% + +exit /B %ERROR_CODE% diff --git a/package-lock.json b/package-lock.json new file mode 100644 index 0000000..f4c9b64 --- /dev/null +++ b/package-lock.json @@ -0,0 +1,24027 @@ +{ + "name": "epasmed", + "version": "0.0.1-SNAPSHOT", + "lockfileVersion": 1, + "requires": true, + "dependencies": { + "@angular-devkit/architect": { + "version": "0.1000.0", + "resolved": "https://registry.npmjs.org/@angular-devkit/architect/-/architect-0.1000.0.tgz", + "integrity": "sha512-luzBYe7t994ebq6xIfYJudxOkMBO0bywafk6sQqb+bOaBQAran4orF1R/zEx6f8TJzEoXELjUvxm/ePSqZdpKg==", + "dev": true, + "requires": { + "@angular-devkit/core": "10.0.0", + "rxjs": "6.5.5" + } + }, + "@angular-devkit/core": { + "version": "10.0.0", + "resolved": "https://registry.npmjs.org/@angular-devkit/core/-/core-10.0.0.tgz", + "integrity": "sha512-IvX9IMaCjDkN9vDVnYcgWbSBinlUUb7jdFhDGeTtK6rGSnjX1GoLvWneVI2hoccS07fPbnfMoYXBoZLwVxiIxw==", + "dev": true, + "requires": { + "ajv": "6.12.2", + "fast-json-stable-stringify": "2.1.0", + "magic-string": "0.25.7", + "rxjs": "6.5.5", + "source-map": "0.7.3" + }, + "dependencies": { + "source-map": { + "version": "0.7.3", + "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.7.3.tgz", + "integrity": "sha512-CkCj6giN3S+n9qrYiBTX5gystlENnRW5jZeNLHpe6aue+SrHcG5VYwujhW9s4dY31mEGsxBDrHR6oI69fTXsaQ==", + "dev": true + } + } + }, + "@angular-devkit/schematics": { + "version": "10.0.0", + "resolved": "https://registry.npmjs.org/@angular-devkit/schematics/-/schematics-10.0.0.tgz", + "integrity": "sha512-FJ/dY18M+cnAT9RkVjVRJ0PMFZci3ok0WoOosW25Fk68jwNSbGCeF8k8NcD6YE60+CfF4/0LxQWgFagr/wdEhw==", + "dev": true, + "requires": { + "@angular-devkit/core": "10.0.0", + "ora": "4.0.4", + "rxjs": "6.5.5" + } + }, + "@angular/cli": { + "version": "10.0.0", + "resolved": "https://registry.npmjs.org/@angular/cli/-/cli-10.0.0.tgz", + "integrity": "sha512-I+2cltQCmThgrnHwsG5AX0hQ9z6rK/8ysRWWeiJXHtEtqupW9eNzXX1QfXWxWB3o6oIKgijvnLlp04BUlWCyXA==", + "dev": true, + "requires": { + "@angular-devkit/architect": "0.1000.0", + "@angular-devkit/core": "10.0.0", + "@angular-devkit/schematics": "10.0.0", + "@schematics/angular": "10.0.0", + "@schematics/update": "0.1000.0", + "@yarnpkg/lockfile": "1.1.0", + "ansi-colors": "4.1.1", + "debug": "4.1.1", + "ini": "1.3.5", + "inquirer": "7.1.0", + "npm-package-arg": "8.0.1", + "npm-pick-manifest": "6.1.0", + "open": "7.0.4", + "pacote": "9.5.12", + "read-package-tree": "5.3.1", + "rimraf": "3.0.2", + "semver": "7.3.2", + "symbol-observable": "1.2.0", + "universal-analytics": "0.4.20", + "uuid": "8.1.0" + }, + "dependencies": { + "debug": { + "version": "4.1.1", + "resolved": "https://registry.npmjs.org/debug/-/debug-4.1.1.tgz", + "integrity": "sha512-pYAIzeRo8J6KPEaJ0VWOh5Pzkbw/RetuzehGM7QRRX5he4fPHx2rdKMB256ehJCkX+XRQm16eZLqLNS8RSZXZw==", + "dev": true, + "requires": { + "ms": "^2.1.1" + } + }, + "semver": { + "version": "7.3.2", + "resolved": "https://registry.npmjs.org/semver/-/semver-7.3.2.tgz", + "integrity": "sha512-OrOb32TeeambH6UrhtShmF7CRDqhL6/5XpPNp2DuRH6+9QLw/orhp72j87v8Qa1ScDkvrrBNpZcDejAirJmfXQ==", + "dev": true + } + } + }, + "@angular/common": { + "version": "10.0.0", + "resolved": "https://registry.npmjs.org/@angular/common/-/common-10.0.0.tgz", + "integrity": "sha512-zaLHMRQqjZyoIc15tLGrwg5ugcVlggqATiMlMVNo9ddqxwpZ4qHK2fdAvqQy9FfWKaPGi6NpIBSR31RGV0ircw==", + "requires": { + "tslib": "^2.0.0" + } + }, + "@angular/compiler": { + "version": "10.0.0", + "resolved": "https://registry.npmjs.org/@angular/compiler/-/compiler-10.0.0.tgz", + "integrity": "sha512-meyJKPLLhkgjxF4dvdJq/sw7MI7KvpgYADoFt2K/5dt48ExXRj+kDlyQC6FAwm9mbgSea/A+JpDG8HMPU7AK5Q==", + "requires": { + "tslib": "^2.0.0" + } + }, + "@angular/compiler-cli": { + "version": "10.0.0", + "resolved": "https://registry.npmjs.org/@angular/compiler-cli/-/compiler-cli-10.0.0.tgz", + "integrity": "sha512-KusRkS1NKPrjhj3BXoxRgeejuJVy/ra4dcDV3hpscRmR8FSdim3rs2H+lYK2hZ26ISGsYUvC8cHNaMqP2U3X+g==", + "dev": true, + "requires": { + "canonical-path": "1.0.0", + "chokidar": "^3.0.0", + "convert-source-map": "^1.5.1", + "dependency-graph": "^0.7.2", + "fs-extra": "4.0.2", + "magic-string": "^0.25.0", + "minimist": "^1.2.0", + "reflect-metadata": "^0.1.2", + "semver": "^6.3.0", + "source-map": "^0.6.1", + "sourcemap-codec": "^1.4.8", + "tslib": "^2.0.0", + "yargs": "15.3.0" + }, + "dependencies": { + "semver": { + "version": "6.3.0", + "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.0.tgz", + "integrity": "sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw==", + "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 + } + } + }, + "@angular/core": { + "version": "10.0.0", + "resolved": "https://registry.npmjs.org/@angular/core/-/core-10.0.0.tgz", + "integrity": "sha512-N1m6op428ktgcsnXqqspb1xGZ9gp664Jmb4JoVajCD3JXucRfidw+vt3kPOldbWA6M4pIu5ZtZY3IZc2GrK5UQ==", + "requires": { + "tslib": "^2.0.0" + } + }, + "@angular/forms": { + "version": "10.0.0", + "resolved": "https://registry.npmjs.org/@angular/forms/-/forms-10.0.0.tgz", + "integrity": "sha512-Jaqs6WmF3fcaL1mKmeZt8pcFHkMuFvgLjbgGFqiOKcdz4UdGTcYkI3mE+UqHpqqXGXYTHLusup8A4TqD7s0rxg==", + "requires": { + "tslib": "^2.0.0" + } + }, + "@angular/localize": { + "version": "10.0.0", + "resolved": "https://registry.npmjs.org/@angular/localize/-/localize-10.0.0.tgz", + "integrity": "sha512-UWuuJYVYDwMGKu8wQuyKH2v0Qbb9HW5n/3tIuw5s0O9yDlhyreueWqCYk42u51p8YDQ7FY7ToZhbMq73G7A8tA==", + "requires": { + "@babel/core": "7.8.3", + "glob": "7.1.2", + "yargs": "15.3.0" + } + }, + "@angular/platform-browser": { + "version": "10.0.0", + "resolved": "https://registry.npmjs.org/@angular/platform-browser/-/platform-browser-10.0.0.tgz", + "integrity": "sha512-2gp53WjGUrL1uReewU21IZJa4KFpbigCyDYB0j/KwTkIGrdwYCOjD0/04oHYjkpazPe/VIu3aN7Gb3PgUa3X/Q==", + "requires": { + "tslib": "^2.0.0" + } + }, + "@angular/platform-browser-dynamic": { + "version": "10.0.0", + "resolved": "https://registry.npmjs.org/@angular/platform-browser-dynamic/-/platform-browser-dynamic-10.0.0.tgz", + "integrity": "sha512-Ol9dMU/UB8PZ1xQkcILmANC+tA1Y/6YeYc8+MYnZUH7uAMXV6kscg9C5fK9JixSldOOS05AQZAdEty9ESjFNNg==", + "requires": { + "tslib": "^2.0.0" + } + }, + "@angular/router": { + "version": "10.0.0", + "resolved": "https://registry.npmjs.org/@angular/router/-/router-10.0.0.tgz", + "integrity": "sha512-rx9e0eOwXypV/oWLrGsug3wx9lPnD6NUU3i20/VG9O1uMHgXfpnP8RJLcUHqmEMBZEYTLMo59fDE5LEgM1sIFg==", + "requires": { + "tslib": "^2.0.0" + } + }, + "@babel/code-frame": { + "version": "7.10.4", + "resolved": "https://registry.npmjs.org/@babel/code-frame/-/code-frame-7.10.4.tgz", + "integrity": "sha512-vG6SvB6oYEhvgisZNFRmRCUkLz11c7rp+tbNTynGqc6mS1d5ATd/sGyV6W0KZZnXRKMTzZDRgQT3Ou9jhpAfUg==", + "requires": { + "@babel/highlight": "^7.10.4" + } + }, + "@babel/compat-data": { + "version": "7.12.7", + "resolved": "https://registry.npmjs.org/@babel/compat-data/-/compat-data-7.12.7.tgz", + "integrity": "sha512-YaxPMGs/XIWtYqrdEOZOCPsVWfEoriXopnsz3/i7apYPXQ3698UFhS6dVT1KN5qOsWmVgw/FOrmQgpRaZayGsw==", + "dev": true + }, + "@babel/core": { + "version": "7.8.3", + "resolved": "https://registry.npmjs.org/@babel/core/-/core-7.8.3.tgz", + "integrity": "sha512-4XFkf8AwyrEG7Ziu3L2L0Cv+WyY47Tcsp70JFmpftbAA1K7YL/sgE9jh9HyNj08Y/U50ItUchpN0w6HxAoX1rA==", + "requires": { + "@babel/code-frame": "^7.8.3", + "@babel/generator": "^7.8.3", + "@babel/helpers": "^7.8.3", + "@babel/parser": "^7.8.3", + "@babel/template": "^7.8.3", + "@babel/traverse": "^7.8.3", + "@babel/types": "^7.8.3", + "convert-source-map": "^1.7.0", + "debug": "^4.1.0", + "gensync": "^1.0.0-beta.1", + "json5": "^2.1.0", + "lodash": "^4.17.13", + "resolve": "^1.3.2", + "semver": "^5.4.1", + "source-map": "^0.5.0" + } + }, + "@babel/generator": { + "version": "7.12.5", + "resolved": "https://registry.npmjs.org/@babel/generator/-/generator-7.12.5.tgz", + "integrity": "sha512-m16TQQJ8hPt7E+OS/XVQg/7U184MLXtvuGbCdA7na61vha+ImkyyNM/9DDA0unYCVZn3ZOhng+qz48/KBOT96A==", + "requires": { + "@babel/types": "^7.12.5", + "jsesc": "^2.5.1", + "source-map": "^0.5.0" + } + }, + "@babel/helper-annotate-as-pure": { + "version": "7.10.4", + "resolved": "https://registry.npmjs.org/@babel/helper-annotate-as-pure/-/helper-annotate-as-pure-7.10.4.tgz", + "integrity": "sha512-XQlqKQP4vXFB7BN8fEEerrmYvHp3fK/rBkRFz9jaJbzK0B1DSfej9Kc7ZzE8Z/OnId1jpJdNAZ3BFQjWG68rcA==", + "dev": true, + "requires": { + "@babel/types": "^7.10.4" + } + }, + "@babel/helper-builder-binary-assignment-operator-visitor": { + "version": "7.10.4", + "resolved": "https://registry.npmjs.org/@babel/helper-builder-binary-assignment-operator-visitor/-/helper-builder-binary-assignment-operator-visitor-7.10.4.tgz", + "integrity": "sha512-L0zGlFrGWZK4PbT8AszSfLTM5sDU1+Az/En9VrdT8/LmEiJt4zXt+Jve9DCAnQcbqDhCI+29y/L93mrDzddCcg==", + "dev": true, + "requires": { + "@babel/helper-explode-assignable-expression": "^7.10.4", + "@babel/types": "^7.10.4" + } + }, + "@babel/helper-compilation-targets": { + "version": "7.12.5", + "resolved": "https://registry.npmjs.org/@babel/helper-compilation-targets/-/helper-compilation-targets-7.12.5.tgz", + "integrity": "sha512-+qH6NrscMolUlzOYngSBMIOQpKUGPPsc61Bu5W10mg84LxZ7cmvnBHzARKbDoFxVvqqAbj6Tg6N7bSrWSPXMyw==", + "dev": true, + "requires": { + "@babel/compat-data": "^7.12.5", + "@babel/helper-validator-option": "^7.12.1", + "browserslist": "^4.14.5", + "semver": "^5.5.0" + } + }, + "@babel/helper-create-class-features-plugin": { + "version": "7.12.1", + "resolved": "https://registry.npmjs.org/@babel/helper-create-class-features-plugin/-/helper-create-class-features-plugin-7.12.1.tgz", + "integrity": "sha512-hkL++rWeta/OVOBTRJc9a5Azh5mt5WgZUGAKMD8JM141YsE08K//bp1unBBieO6rUKkIPyUE0USQ30jAy3Sk1w==", + "dev": true, + "requires": { + "@babel/helper-function-name": "^7.10.4", + "@babel/helper-member-expression-to-functions": "^7.12.1", + "@babel/helper-optimise-call-expression": "^7.10.4", + "@babel/helper-replace-supers": "^7.12.1", + "@babel/helper-split-export-declaration": "^7.10.4" + } + }, + "@babel/helper-create-regexp-features-plugin": { + "version": "7.12.7", + "resolved": "https://registry.npmjs.org/@babel/helper-create-regexp-features-plugin/-/helper-create-regexp-features-plugin-7.12.7.tgz", + "integrity": "sha512-idnutvQPdpbduutvi3JVfEgcVIHooQnhvhx0Nk9isOINOIGYkZea1Pk2JlJRiUnMefrlvr0vkByATBY/mB4vjQ==", + "dev": true, + "requires": { + "@babel/helper-annotate-as-pure": "^7.10.4", + "regexpu-core": "^4.7.1" + } + }, + "@babel/helper-define-map": { + "version": "7.10.5", + "resolved": "https://registry.npmjs.org/@babel/helper-define-map/-/helper-define-map-7.10.5.tgz", + "integrity": "sha512-fMw4kgFB720aQFXSVaXr79pjjcW5puTCM16+rECJ/plGS+zByelE8l9nCpV1GibxTnFVmUuYG9U8wYfQHdzOEQ==", + "dev": true, + "requires": { + "@babel/helper-function-name": "^7.10.4", + "@babel/types": "^7.10.5", + "lodash": "^4.17.19" + } + }, + "@babel/helper-explode-assignable-expression": { + "version": "7.12.1", + "resolved": "https://registry.npmjs.org/@babel/helper-explode-assignable-expression/-/helper-explode-assignable-expression-7.12.1.tgz", + "integrity": "sha512-dmUwH8XmlrUpVqgtZ737tK88v07l840z9j3OEhCLwKTkjlvKpfqXVIZ0wpK3aeOxspwGrf/5AP5qLx4rO3w5rA==", + "dev": true, + "requires": { + "@babel/types": "^7.12.1" + } + }, + "@babel/helper-function-name": { + "version": "7.10.4", + "resolved": "https://registry.npmjs.org/@babel/helper-function-name/-/helper-function-name-7.10.4.tgz", + "integrity": "sha512-YdaSyz1n8gY44EmN7x44zBn9zQ1Ry2Y+3GTA+3vH6Mizke1Vw0aWDM66FOYEPw8//qKkmqOckrGgTYa+6sceqQ==", + "requires": { + "@babel/helper-get-function-arity": "^7.10.4", + "@babel/template": "^7.10.4", + "@babel/types": "^7.10.4" + } + }, + "@babel/helper-get-function-arity": { + "version": "7.10.4", + "resolved": "https://registry.npmjs.org/@babel/helper-get-function-arity/-/helper-get-function-arity-7.10.4.tgz", + "integrity": "sha512-EkN3YDB+SRDgiIUnNgcmiD361ti+AVbL3f3Henf6dqqUyr5dMsorno0lJWJuLhDhkI5sYEpgj6y9kB8AOU1I2A==", + "requires": { + "@babel/types": "^7.10.4" + } + }, + "@babel/helper-hoist-variables": { + "version": "7.10.4", + "resolved": "https://registry.npmjs.org/@babel/helper-hoist-variables/-/helper-hoist-variables-7.10.4.tgz", + "integrity": "sha512-wljroF5PgCk2juF69kanHVs6vrLwIPNp6DLD+Lrl3hoQ3PpPPikaDRNFA+0t81NOoMt2DL6WW/mdU8k4k6ZzuA==", + "dev": true, + "requires": { + "@babel/types": "^7.10.4" + } + }, + "@babel/helper-member-expression-to-functions": { + "version": "7.12.7", + "resolved": "https://registry.npmjs.org/@babel/helper-member-expression-to-functions/-/helper-member-expression-to-functions-7.12.7.tgz", + "integrity": "sha512-DCsuPyeWxeHgh1Dus7APn7iza42i/qXqiFPWyBDdOFtvS581JQePsc1F/nD+fHrcswhLlRc2UpYS1NwERxZhHw==", + "dev": true, + "requires": { + "@babel/types": "^7.12.7" + } + }, + "@babel/helper-module-imports": { + "version": "7.12.5", + "resolved": "https://registry.npmjs.org/@babel/helper-module-imports/-/helper-module-imports-7.12.5.tgz", + "integrity": "sha512-SR713Ogqg6++uexFRORf/+nPXMmWIn80TALu0uaFb+iQIUoR7bOC7zBWyzBs5b3tBBJXuyD0cRu1F15GyzjOWA==", + "dev": true, + "requires": { + "@babel/types": "^7.12.5" + } + }, + "@babel/helper-module-transforms": { + "version": "7.12.1", + "resolved": "https://registry.npmjs.org/@babel/helper-module-transforms/-/helper-module-transforms-7.12.1.tgz", + "integrity": "sha512-QQzehgFAZ2bbISiCpmVGfiGux8YVFXQ0abBic2Envhej22DVXV9nCFaS5hIQbkyo1AdGb+gNME2TSh3hYJVV/w==", + "dev": true, + "requires": { + "@babel/helper-module-imports": "^7.12.1", + "@babel/helper-replace-supers": "^7.12.1", + "@babel/helper-simple-access": "^7.12.1", + "@babel/helper-split-export-declaration": "^7.11.0", + "@babel/helper-validator-identifier": "^7.10.4", + "@babel/template": "^7.10.4", + "@babel/traverse": "^7.12.1", + "@babel/types": "^7.12.1", + "lodash": "^4.17.19" + } + }, + "@babel/helper-optimise-call-expression": { + "version": "7.12.7", + "resolved": "https://registry.npmjs.org/@babel/helper-optimise-call-expression/-/helper-optimise-call-expression-7.12.7.tgz", + "integrity": "sha512-I5xc9oSJ2h59OwyUqjv95HRyzxj53DAubUERgQMrpcCEYQyToeHA+NEcUEsVWB4j53RDeskeBJ0SgRAYHDBckw==", + "dev": true, + "requires": { + "@babel/types": "^7.12.7" + } + }, + "@babel/helper-plugin-utils": { + "version": "7.10.4", + "resolved": "https://registry.npmjs.org/@babel/helper-plugin-utils/-/helper-plugin-utils-7.10.4.tgz", + "integrity": "sha512-O4KCvQA6lLiMU9l2eawBPMf1xPP8xPfB3iEQw150hOVTqj/rfXz0ThTb4HEzqQfs2Bmo5Ay8BzxfzVtBrr9dVg==", + "dev": true + }, + "@babel/helper-remap-async-to-generator": { + "version": "7.12.1", + "resolved": "https://registry.npmjs.org/@babel/helper-remap-async-to-generator/-/helper-remap-async-to-generator-7.12.1.tgz", + "integrity": "sha512-9d0KQCRM8clMPcDwo8SevNs+/9a8yWVVmaE80FGJcEP8N1qToREmWEGnBn8BUlJhYRFz6fqxeRL1sl5Ogsed7A==", + "dev": true, + "requires": { + "@babel/helper-annotate-as-pure": "^7.10.4", + "@babel/helper-wrap-function": "^7.10.4", + "@babel/types": "^7.12.1" + } + }, + "@babel/helper-replace-supers": { + "version": "7.12.5", + "resolved": "https://registry.npmjs.org/@babel/helper-replace-supers/-/helper-replace-supers-7.12.5.tgz", + "integrity": "sha512-5YILoed0ZyIpF4gKcpZitEnXEJ9UoDRki1Ey6xz46rxOzfNMAhVIJMoune1hmPVxh40LRv1+oafz7UsWX+vyWA==", + "dev": true, + "requires": { + "@babel/helper-member-expression-to-functions": "^7.12.1", + "@babel/helper-optimise-call-expression": "^7.10.4", + "@babel/traverse": "^7.12.5", + "@babel/types": "^7.12.5" + } + }, + "@babel/helper-simple-access": { + "version": "7.12.1", + "resolved": "https://registry.npmjs.org/@babel/helper-simple-access/-/helper-simple-access-7.12.1.tgz", + "integrity": "sha512-OxBp7pMrjVewSSC8fXDFrHrBcJATOOFssZwv16F3/6Xtc138GHybBfPbm9kfiqQHKhYQrlamWILwlDCeyMFEaA==", + "dev": true, + "requires": { + "@babel/types": "^7.12.1" + } + }, + "@babel/helper-skip-transparent-expression-wrappers": { + "version": "7.12.1", + "resolved": "https://registry.npmjs.org/@babel/helper-skip-transparent-expression-wrappers/-/helper-skip-transparent-expression-wrappers-7.12.1.tgz", + "integrity": "sha512-Mf5AUuhG1/OCChOJ/HcADmvcHM42WJockombn8ATJG3OnyiSxBK/Mm5x78BQWvmtXZKHgbjdGL2kin/HOLlZGA==", + "dev": true, + "requires": { + "@babel/types": "^7.12.1" + } + }, + "@babel/helper-split-export-declaration": { + "version": "7.11.0", + "resolved": "https://registry.npmjs.org/@babel/helper-split-export-declaration/-/helper-split-export-declaration-7.11.0.tgz", + "integrity": "sha512-74Vejvp6mHkGE+m+k5vHY93FX2cAtrw1zXrZXRlG4l410Nm9PxfEiVTn1PjDPV5SnmieiueY4AFg2xqhNFuuZg==", + "requires": { + "@babel/types": "^7.11.0" + } + }, + "@babel/helper-validator-identifier": { + "version": "7.10.4", + "resolved": "https://registry.npmjs.org/@babel/helper-validator-identifier/-/helper-validator-identifier-7.10.4.tgz", + "integrity": "sha512-3U9y+43hz7ZM+rzG24Qe2mufW5KhvFg/NhnNph+i9mgCtdTCtMJuI1TMkrIUiK7Ix4PYlRF9I5dhqaLYA/ADXw==" + }, + "@babel/helper-validator-option": { + "version": "7.12.1", + "resolved": "https://registry.npmjs.org/@babel/helper-validator-option/-/helper-validator-option-7.12.1.tgz", + "integrity": "sha512-YpJabsXlJVWP0USHjnC/AQDTLlZERbON577YUVO/wLpqyj6HAtVYnWaQaN0iUN+1/tWn3c+uKKXjRut5115Y2A==", + "dev": true + }, + "@babel/helper-wrap-function": { + "version": "7.12.3", + "resolved": "https://registry.npmjs.org/@babel/helper-wrap-function/-/helper-wrap-function-7.12.3.tgz", + "integrity": "sha512-Cvb8IuJDln3rs6tzjW3Y8UeelAOdnpB8xtQ4sme2MSZ9wOxrbThporC0y/EtE16VAtoyEfLM404Xr1e0OOp+ow==", + "dev": true, + "requires": { + "@babel/helper-function-name": "^7.10.4", + "@babel/template": "^7.10.4", + "@babel/traverse": "^7.10.4", + "@babel/types": "^7.10.4" + } + }, + "@babel/helpers": { + "version": "7.12.5", + "resolved": "https://registry.npmjs.org/@babel/helpers/-/helpers-7.12.5.tgz", + "integrity": "sha512-lgKGMQlKqA8meJqKsW6rUnc4MdUk35Ln0ATDqdM1a/UpARODdI4j5Y5lVfUScnSNkJcdCRAaWkspykNoFg9sJA==", + "requires": { + "@babel/template": "^7.10.4", + "@babel/traverse": "^7.12.5", + "@babel/types": "^7.12.5" + } + }, + "@babel/highlight": { + "version": "7.10.4", + "resolved": "https://registry.npmjs.org/@babel/highlight/-/highlight-7.10.4.tgz", + "integrity": "sha512-i6rgnR/YgPEQzZZnbTHHuZdlE8qyoBNalD6F+q4vAFlcMEcqmkoG+mPqJYJCo63qPf74+Y1UZsl3l6f7/RIkmA==", + "requires": { + "@babel/helper-validator-identifier": "^7.10.4", + "chalk": "^2.0.0", + "js-tokens": "^4.0.0" + } + }, + "@babel/parser": { + "version": "7.12.7", + "resolved": "https://registry.npmjs.org/@babel/parser/-/parser-7.12.7.tgz", + "integrity": "sha512-oWR02Ubp4xTLCAqPRiNIuMVgNO5Aif/xpXtabhzW2HWUD47XJsAB4Zd/Rg30+XeQA3juXigV7hlquOTmwqLiwg==" + }, + "@babel/plugin-proposal-async-generator-functions": { + "version": "7.12.1", + "resolved": "https://registry.npmjs.org/@babel/plugin-proposal-async-generator-functions/-/plugin-proposal-async-generator-functions-7.12.1.tgz", + "integrity": "sha512-d+/o30tJxFxrA1lhzJqiUcEJdI6jKlNregCv5bASeGf2Q4MXmnwH7viDo7nhx1/ohf09oaH8j1GVYG/e3Yqk6A==", + "dev": true, + "requires": { + "@babel/helper-plugin-utils": "^7.10.4", + "@babel/helper-remap-async-to-generator": "^7.12.1", + "@babel/plugin-syntax-async-generators": "^7.8.0" + } + }, + "@babel/plugin-proposal-class-properties": { + "version": "7.12.1", + "resolved": "https://registry.npmjs.org/@babel/plugin-proposal-class-properties/-/plugin-proposal-class-properties-7.12.1.tgz", + "integrity": "sha512-cKp3dlQsFsEs5CWKnN7BnSHOd0EOW8EKpEjkoz1pO2E5KzIDNV9Ros1b0CnmbVgAGXJubOYVBOGCT1OmJwOI7w==", + "dev": true, + "requires": { + "@babel/helper-create-class-features-plugin": "^7.12.1", + "@babel/helper-plugin-utils": "^7.10.4" + } + }, + "@babel/plugin-proposal-dynamic-import": { + "version": "7.12.1", + "resolved": "https://registry.npmjs.org/@babel/plugin-proposal-dynamic-import/-/plugin-proposal-dynamic-import-7.12.1.tgz", + "integrity": "sha512-a4rhUSZFuq5W8/OO8H7BL5zspjnc1FLd9hlOxIK/f7qG4a0qsqk8uvF/ywgBA8/OmjsapjpvaEOYItfGG1qIvQ==", + "dev": true, + "requires": { + "@babel/helper-plugin-utils": "^7.10.4", + "@babel/plugin-syntax-dynamic-import": "^7.8.0" + } + }, + "@babel/plugin-proposal-export-namespace-from": { + "version": "7.12.1", + "resolved": "https://registry.npmjs.org/@babel/plugin-proposal-export-namespace-from/-/plugin-proposal-export-namespace-from-7.12.1.tgz", + "integrity": "sha512-6CThGf0irEkzujYS5LQcjBx8j/4aQGiVv7J9+2f7pGfxqyKh3WnmVJYW3hdrQjyksErMGBPQrCnHfOtna+WLbw==", + "dev": true, + "requires": { + "@babel/helper-plugin-utils": "^7.10.4", + "@babel/plugin-syntax-export-namespace-from": "^7.8.3" + } + }, + "@babel/plugin-proposal-json-strings": { + "version": "7.12.1", + "resolved": "https://registry.npmjs.org/@babel/plugin-proposal-json-strings/-/plugin-proposal-json-strings-7.12.1.tgz", + "integrity": "sha512-GoLDUi6U9ZLzlSda2Df++VSqDJg3CG+dR0+iWsv6XRw1rEq+zwt4DirM9yrxW6XWaTpmai1cWJLMfM8qQJf+yw==", + "dev": true, + "requires": { + "@babel/helper-plugin-utils": "^7.10.4", + "@babel/plugin-syntax-json-strings": "^7.8.0" + } + }, + "@babel/plugin-proposal-logical-assignment-operators": { + "version": "7.12.1", + "resolved": "https://registry.npmjs.org/@babel/plugin-proposal-logical-assignment-operators/-/plugin-proposal-logical-assignment-operators-7.12.1.tgz", + "integrity": "sha512-k8ZmVv0JU+4gcUGeCDZOGd0lCIamU/sMtIiX3UWnUc5yzgq6YUGyEolNYD+MLYKfSzgECPcqetVcJP9Afe/aCA==", + "dev": true, + "requires": { + "@babel/helper-plugin-utils": "^7.10.4", + "@babel/plugin-syntax-logical-assignment-operators": "^7.10.4" + } + }, + "@babel/plugin-proposal-nullish-coalescing-operator": { + "version": "7.12.1", + "resolved": "https://registry.npmjs.org/@babel/plugin-proposal-nullish-coalescing-operator/-/plugin-proposal-nullish-coalescing-operator-7.12.1.tgz", + "integrity": "sha512-nZY0ESiaQDI1y96+jk6VxMOaL4LPo/QDHBqL+SF3/vl6dHkTwHlOI8L4ZwuRBHgakRBw5zsVylel7QPbbGuYgg==", + "dev": true, + "requires": { + "@babel/helper-plugin-utils": "^7.10.4", + "@babel/plugin-syntax-nullish-coalescing-operator": "^7.8.0" + } + }, + "@babel/plugin-proposal-numeric-separator": { + "version": "7.12.7", + "resolved": "https://registry.npmjs.org/@babel/plugin-proposal-numeric-separator/-/plugin-proposal-numeric-separator-7.12.7.tgz", + "integrity": "sha512-8c+uy0qmnRTeukiGsjLGy6uVs/TFjJchGXUeBqlG4VWYOdJWkhhVPdQ3uHwbmalfJwv2JsV0qffXP4asRfL2SQ==", + "dev": true, + "requires": { + "@babel/helper-plugin-utils": "^7.10.4", + "@babel/plugin-syntax-numeric-separator": "^7.10.4" + } + }, + "@babel/plugin-proposal-object-rest-spread": { + "version": "7.12.1", + "resolved": "https://registry.npmjs.org/@babel/plugin-proposal-object-rest-spread/-/plugin-proposal-object-rest-spread-7.12.1.tgz", + "integrity": "sha512-s6SowJIjzlhx8o7lsFx5zmY4At6CTtDvgNQDdPzkBQucle58A6b/TTeEBYtyDgmcXjUTM+vE8YOGHZzzbc/ioA==", + "dev": true, + "requires": { + "@babel/helper-plugin-utils": "^7.10.4", + "@babel/plugin-syntax-object-rest-spread": "^7.8.0", + "@babel/plugin-transform-parameters": "^7.12.1" + } + }, + "@babel/plugin-proposal-optional-catch-binding": { + "version": "7.12.1", + "resolved": "https://registry.npmjs.org/@babel/plugin-proposal-optional-catch-binding/-/plugin-proposal-optional-catch-binding-7.12.1.tgz", + "integrity": "sha512-hFvIjgprh9mMw5v42sJWLI1lzU5L2sznP805zeT6rySVRA0Y18StRhDqhSxlap0oVgItRsB6WSROp4YnJTJz0g==", + "dev": true, + "requires": { + "@babel/helper-plugin-utils": "^7.10.4", + "@babel/plugin-syntax-optional-catch-binding": "^7.8.0" + } + }, + "@babel/plugin-proposal-optional-chaining": { + "version": "7.12.7", + "resolved": "https://registry.npmjs.org/@babel/plugin-proposal-optional-chaining/-/plugin-proposal-optional-chaining-7.12.7.tgz", + "integrity": "sha512-4ovylXZ0PWmwoOvhU2vhnzVNnm88/Sm9nx7V8BPgMvAzn5zDou3/Awy0EjglyubVHasJj+XCEkr/r1X3P5elCA==", + "dev": true, + "requires": { + "@babel/helper-plugin-utils": "^7.10.4", + "@babel/helper-skip-transparent-expression-wrappers": "^7.12.1", + "@babel/plugin-syntax-optional-chaining": "^7.8.0" + } + }, + "@babel/plugin-proposal-private-methods": { + "version": "7.12.1", + "resolved": "https://registry.npmjs.org/@babel/plugin-proposal-private-methods/-/plugin-proposal-private-methods-7.12.1.tgz", + "integrity": "sha512-mwZ1phvH7/NHK6Kf8LP7MYDogGV+DKB1mryFOEwx5EBNQrosvIczzZFTUmWaeujd5xT6G1ELYWUz3CutMhjE1w==", + "dev": true, + "requires": { + "@babel/helper-create-class-features-plugin": "^7.12.1", + "@babel/helper-plugin-utils": "^7.10.4" + } + }, + "@babel/plugin-proposal-unicode-property-regex": { + "version": "7.12.1", + "resolved": "https://registry.npmjs.org/@babel/plugin-proposal-unicode-property-regex/-/plugin-proposal-unicode-property-regex-7.12.1.tgz", + "integrity": "sha512-MYq+l+PvHuw/rKUz1at/vb6nCnQ2gmJBNaM62z0OgH7B2W1D9pvkpYtlti9bGtizNIU1K3zm4bZF9F91efVY0w==", + "dev": true, + "requires": { + "@babel/helper-create-regexp-features-plugin": "^7.12.1", + "@babel/helper-plugin-utils": "^7.10.4" + } + }, + "@babel/plugin-syntax-async-generators": { + "version": "7.8.4", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-async-generators/-/plugin-syntax-async-generators-7.8.4.tgz", + "integrity": "sha512-tycmZxkGfZaxhMRbXlPXuVFpdWlXpir2W4AMhSJgRKzk/eDlIXOhb2LHWoLpDF7TEHylV5zNhykX6KAgHJmTNw==", + "dev": true, + "requires": { + "@babel/helper-plugin-utils": "^7.8.0" + } + }, + "@babel/plugin-syntax-bigint": { + "version": "7.8.3", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-bigint/-/plugin-syntax-bigint-7.8.3.tgz", + "integrity": "sha512-wnTnFlG+YxQm3vDxpGE57Pj0srRU4sHE/mDkt1qv2YJJSeUAec2ma4WLUnUPeKjyrfntVwe/N6dCXpU+zL3Npg==", + "dev": true, + "requires": { + "@babel/helper-plugin-utils": "^7.8.0" + } + }, + "@babel/plugin-syntax-class-properties": { + "version": "7.12.1", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-class-properties/-/plugin-syntax-class-properties-7.12.1.tgz", + "integrity": "sha512-U40A76x5gTwmESz+qiqssqmeEsKvcSyvtgktrm0uzcARAmM9I1jR221f6Oq+GmHrcD+LvZDag1UTOTe2fL3TeA==", + "dev": true, + "requires": { + "@babel/helper-plugin-utils": "^7.10.4" + } + }, + "@babel/plugin-syntax-dynamic-import": { + "version": "7.8.3", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-dynamic-import/-/plugin-syntax-dynamic-import-7.8.3.tgz", + "integrity": "sha512-5gdGbFon+PszYzqs83S3E5mpi7/y/8M9eC90MRTZfduQOYW76ig6SOSPNe41IG5LoP3FGBn2N0RjVDSQiS94kQ==", + "dev": true, + "requires": { + "@babel/helper-plugin-utils": "^7.8.0" + } + }, + "@babel/plugin-syntax-export-namespace-from": { + "version": "7.8.3", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-export-namespace-from/-/plugin-syntax-export-namespace-from-7.8.3.tgz", + "integrity": "sha512-MXf5laXo6c1IbEbegDmzGPwGNTsHZmEy6QGznu5Sh2UCWvueywb2ee+CCE4zQiZstxU9BMoQO9i6zUFSY0Kj0Q==", + "dev": true, + "requires": { + "@babel/helper-plugin-utils": "^7.8.3" + } + }, + "@babel/plugin-syntax-import-meta": { + "version": "7.10.4", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-import-meta/-/plugin-syntax-import-meta-7.10.4.tgz", + "integrity": "sha512-Yqfm+XDx0+Prh3VSeEQCPU81yC+JWZ2pDPFSS4ZdpfZhp4MkFMaDC1UqseovEKwSUpnIL7+vK+Clp7bfh0iD7g==", + "dev": true, + "requires": { + "@babel/helper-plugin-utils": "^7.10.4" + } + }, + "@babel/plugin-syntax-json-strings": { + "version": "7.8.3", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-json-strings/-/plugin-syntax-json-strings-7.8.3.tgz", + "integrity": "sha512-lY6kdGpWHvjoe2vk4WrAapEuBR69EMxZl+RoGRhrFGNYVK8mOPAW8VfbT/ZgrFbXlDNiiaxQnAtgVCZ6jv30EA==", + "dev": true, + "requires": { + "@babel/helper-plugin-utils": "^7.8.0" + } + }, + "@babel/plugin-syntax-logical-assignment-operators": { + "version": "7.10.4", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-logical-assignment-operators/-/plugin-syntax-logical-assignment-operators-7.10.4.tgz", + "integrity": "sha512-d8waShlpFDinQ5MtvGU9xDAOzKH47+FFoney2baFIoMr952hKOLp1HR7VszoZvOsV/4+RRszNY7D17ba0te0ig==", + "dev": true, + "requires": { + "@babel/helper-plugin-utils": "^7.10.4" + } + }, + "@babel/plugin-syntax-nullish-coalescing-operator": { + "version": "7.8.3", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-nullish-coalescing-operator/-/plugin-syntax-nullish-coalescing-operator-7.8.3.tgz", + "integrity": "sha512-aSff4zPII1u2QD7y+F8oDsz19ew4IGEJg9SVW+bqwpwtfFleiQDMdzA/R+UlWDzfnHFCxxleFT0PMIrR36XLNQ==", + "dev": true, + "requires": { + "@babel/helper-plugin-utils": "^7.8.0" + } + }, + "@babel/plugin-syntax-numeric-separator": { + "version": "7.10.4", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-numeric-separator/-/plugin-syntax-numeric-separator-7.10.4.tgz", + "integrity": "sha512-9H6YdfkcK/uOnY/K7/aA2xpzaAgkQn37yzWUMRK7OaPOqOpGS1+n0H5hxT9AUw9EsSjPW8SVyMJwYRtWs3X3ug==", + "dev": true, + "requires": { + "@babel/helper-plugin-utils": "^7.10.4" + } + }, + "@babel/plugin-syntax-object-rest-spread": { + "version": "7.8.3", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-object-rest-spread/-/plugin-syntax-object-rest-spread-7.8.3.tgz", + "integrity": "sha512-XoqMijGZb9y3y2XskN+P1wUGiVwWZ5JmoDRwx5+3GmEplNyVM2s2Dg8ILFQm8rWM48orGy5YpI5Bl8U1y7ydlA==", + "dev": true, + "requires": { + "@babel/helper-plugin-utils": "^7.8.0" + } + }, + "@babel/plugin-syntax-optional-catch-binding": { + "version": "7.8.3", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-optional-catch-binding/-/plugin-syntax-optional-catch-binding-7.8.3.tgz", + "integrity": "sha512-6VPD0Pc1lpTqw0aKoeRTMiB+kWhAoT24PA+ksWSBrFtl5SIRVpZlwN3NNPQjehA2E/91FV3RjLWoVTglWcSV3Q==", + "dev": true, + "requires": { + "@babel/helper-plugin-utils": "^7.8.0" + } + }, + "@babel/plugin-syntax-optional-chaining": { + "version": "7.8.3", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-optional-chaining/-/plugin-syntax-optional-chaining-7.8.3.tgz", + "integrity": "sha512-KoK9ErH1MBlCPxV0VANkXW2/dw4vlbGDrFgz8bmUsBGYkFRcbRwMh6cIJubdPrkxRwuGdtCk0v/wPTKbQgBjkg==", + "dev": true, + "requires": { + "@babel/helper-plugin-utils": "^7.8.0" + } + }, + "@babel/plugin-syntax-top-level-await": { + "version": "7.12.1", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-top-level-await/-/plugin-syntax-top-level-await-7.12.1.tgz", + "integrity": "sha512-i7ooMZFS+a/Om0crxZodrTzNEPJHZrlMVGMTEpFAj6rYY/bKCddB0Dk/YxfPuYXOopuhKk/e1jV6h+WUU9XN3A==", + "dev": true, + "requires": { + "@babel/helper-plugin-utils": "^7.10.4" + } + }, + "@babel/plugin-transform-arrow-functions": { + "version": "7.12.1", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-arrow-functions/-/plugin-transform-arrow-functions-7.12.1.tgz", + "integrity": "sha512-5QB50qyN44fzzz4/qxDPQMBCTHgxg3n0xRBLJUmBlLoU/sFvxVWGZF/ZUfMVDQuJUKXaBhbupxIzIfZ6Fwk/0A==", + "dev": true, + "requires": { + "@babel/helper-plugin-utils": "^7.10.4" + } + }, + "@babel/plugin-transform-async-to-generator": { + "version": "7.12.1", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-async-to-generator/-/plugin-transform-async-to-generator-7.12.1.tgz", + "integrity": "sha512-SDtqoEcarK1DFlRJ1hHRY5HvJUj5kX4qmtpMAm2QnhOlyuMC4TMdCRgW6WXpv93rZeYNeLP22y8Aq2dbcDRM1A==", + "dev": true, + "requires": { + "@babel/helper-module-imports": "^7.12.1", + "@babel/helper-plugin-utils": "^7.10.4", + "@babel/helper-remap-async-to-generator": "^7.12.1" + } + }, + "@babel/plugin-transform-block-scoped-functions": { + "version": "7.12.1", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-block-scoped-functions/-/plugin-transform-block-scoped-functions-7.12.1.tgz", + "integrity": "sha512-5OpxfuYnSgPalRpo8EWGPzIYf0lHBWORCkj5M0oLBwHdlux9Ri36QqGW3/LR13RSVOAoUUMzoPI/jpE4ABcHoA==", + "dev": true, + "requires": { + "@babel/helper-plugin-utils": "^7.10.4" + } + }, + "@babel/plugin-transform-block-scoping": { + "version": "7.12.1", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-block-scoping/-/plugin-transform-block-scoping-7.12.1.tgz", + "integrity": "sha512-zJyAC9sZdE60r1nVQHblcfCj29Dh2Y0DOvlMkcqSo0ckqjiCwNiUezUKw+RjOCwGfpLRwnAeQ2XlLpsnGkvv9w==", + "dev": true, + "requires": { + "@babel/helper-plugin-utils": "^7.10.4" + } + }, + "@babel/plugin-transform-classes": { + "version": "7.12.1", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-classes/-/plugin-transform-classes-7.12.1.tgz", + "integrity": "sha512-/74xkA7bVdzQTBeSUhLLJgYIcxw/dpEpCdRDiHgPJ3Mv6uC11UhjpOhl72CgqbBCmt1qtssCyB2xnJm1+PFjog==", + "dev": true, + "requires": { + "@babel/helper-annotate-as-pure": "^7.10.4", + "@babel/helper-define-map": "^7.10.4", + "@babel/helper-function-name": "^7.10.4", + "@babel/helper-optimise-call-expression": "^7.10.4", + "@babel/helper-plugin-utils": "^7.10.4", + "@babel/helper-replace-supers": "^7.12.1", + "@babel/helper-split-export-declaration": "^7.10.4", + "globals": "^11.1.0" + } + }, + "@babel/plugin-transform-computed-properties": { + "version": "7.12.1", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-computed-properties/-/plugin-transform-computed-properties-7.12.1.tgz", + "integrity": "sha512-vVUOYpPWB7BkgUWPo4C44mUQHpTZXakEqFjbv8rQMg7TC6S6ZhGZ3otQcRH6u7+adSlE5i0sp63eMC/XGffrzg==", + "dev": true, + "requires": { + "@babel/helper-plugin-utils": "^7.10.4" + } + }, + "@babel/plugin-transform-destructuring": { + "version": "7.12.1", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-destructuring/-/plugin-transform-destructuring-7.12.1.tgz", + "integrity": "sha512-fRMYFKuzi/rSiYb2uRLiUENJOKq4Gnl+6qOv5f8z0TZXg3llUwUhsNNwrwaT/6dUhJTzNpBr+CUvEWBtfNY1cw==", + "dev": true, + "requires": { + "@babel/helper-plugin-utils": "^7.10.4" + } + }, + "@babel/plugin-transform-dotall-regex": { + "version": "7.12.1", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-dotall-regex/-/plugin-transform-dotall-regex-7.12.1.tgz", + "integrity": "sha512-B2pXeRKoLszfEW7J4Hg9LoFaWEbr/kzo3teWHmtFCszjRNa/b40f9mfeqZsIDLLt/FjwQ6pz/Gdlwy85xNckBA==", + "dev": true, + "requires": { + "@babel/helper-create-regexp-features-plugin": "^7.12.1", + "@babel/helper-plugin-utils": "^7.10.4" + } + }, + "@babel/plugin-transform-duplicate-keys": { + "version": "7.12.1", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-duplicate-keys/-/plugin-transform-duplicate-keys-7.12.1.tgz", + "integrity": "sha512-iRght0T0HztAb/CazveUpUQrZY+aGKKaWXMJ4uf9YJtqxSUe09j3wteztCUDRHs+SRAL7yMuFqUsLoAKKzgXjw==", + "dev": true, + "requires": { + "@babel/helper-plugin-utils": "^7.10.4" + } + }, + "@babel/plugin-transform-exponentiation-operator": { + "version": "7.12.1", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-exponentiation-operator/-/plugin-transform-exponentiation-operator-7.12.1.tgz", + "integrity": "sha512-7tqwy2bv48q+c1EHbXK0Zx3KXd2RVQp6OC7PbwFNt/dPTAV3Lu5sWtWuAj8owr5wqtWnqHfl2/mJlUmqkChKug==", + "dev": true, + "requires": { + "@babel/helper-builder-binary-assignment-operator-visitor": "^7.10.4", + "@babel/helper-plugin-utils": "^7.10.4" + } + }, + "@babel/plugin-transform-for-of": { + "version": "7.12.1", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-for-of/-/plugin-transform-for-of-7.12.1.tgz", + "integrity": "sha512-Zaeq10naAsuHo7heQvyV0ptj4dlZJwZgNAtBYBnu5nNKJoW62m0zKcIEyVECrUKErkUkg6ajMy4ZfnVZciSBhg==", + "dev": true, + "requires": { + "@babel/helper-plugin-utils": "^7.10.4" + } + }, + "@babel/plugin-transform-function-name": { + "version": "7.12.1", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-function-name/-/plugin-transform-function-name-7.12.1.tgz", + "integrity": "sha512-JF3UgJUILoFrFMEnOJLJkRHSk6LUSXLmEFsA23aR2O5CSLUxbeUX1IZ1YQ7Sn0aXb601Ncwjx73a+FVqgcljVw==", + "dev": true, + "requires": { + "@babel/helper-function-name": "^7.10.4", + "@babel/helper-plugin-utils": "^7.10.4" + } + }, + "@babel/plugin-transform-literals": { + "version": "7.12.1", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-literals/-/plugin-transform-literals-7.12.1.tgz", + "integrity": "sha512-+PxVGA+2Ag6uGgL0A5f+9rklOnnMccwEBzwYFL3EUaKuiyVnUipyXncFcfjSkbimLrODoqki1U9XxZzTvfN7IQ==", + "dev": true, + "requires": { + "@babel/helper-plugin-utils": "^7.10.4" + } + }, + "@babel/plugin-transform-member-expression-literals": { + "version": "7.12.1", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-member-expression-literals/-/plugin-transform-member-expression-literals-7.12.1.tgz", + "integrity": "sha512-1sxePl6z9ad0gFMB9KqmYofk34flq62aqMt9NqliS/7hPEpURUCMbyHXrMPlo282iY7nAvUB1aQd5mg79UD9Jg==", + "dev": true, + "requires": { + "@babel/helper-plugin-utils": "^7.10.4" + } + }, + "@babel/plugin-transform-modules-amd": { + "version": "7.12.1", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-modules-amd/-/plugin-transform-modules-amd-7.12.1.tgz", + "integrity": "sha512-tDW8hMkzad5oDtzsB70HIQQRBiTKrhfgwC/KkJeGsaNFTdWhKNt/BiE8c5yj19XiGyrxpbkOfH87qkNg1YGlOQ==", + "dev": true, + "requires": { + "@babel/helper-module-transforms": "^7.12.1", + "@babel/helper-plugin-utils": "^7.10.4", + "babel-plugin-dynamic-import-node": "^2.3.3" + } + }, + "@babel/plugin-transform-modules-commonjs": { + "version": "7.12.1", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-modules-commonjs/-/plugin-transform-modules-commonjs-7.12.1.tgz", + "integrity": "sha512-dY789wq6l0uLY8py9c1B48V8mVL5gZh/+PQ5ZPrylPYsnAvnEMjqsUXkuoDVPeVK+0VyGar+D08107LzDQ6pag==", + "dev": true, + "requires": { + "@babel/helper-module-transforms": "^7.12.1", + "@babel/helper-plugin-utils": "^7.10.4", + "@babel/helper-simple-access": "^7.12.1", + "babel-plugin-dynamic-import-node": "^2.3.3" + } + }, + "@babel/plugin-transform-modules-systemjs": { + "version": "7.12.1", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-modules-systemjs/-/plugin-transform-modules-systemjs-7.12.1.tgz", + "integrity": "sha512-Hn7cVvOavVh8yvW6fLwveFqSnd7rbQN3zJvoPNyNaQSvgfKmDBO9U1YL9+PCXGRlZD9tNdWTy5ACKqMuzyn32Q==", + "dev": true, + "requires": { + "@babel/helper-hoist-variables": "^7.10.4", + "@babel/helper-module-transforms": "^7.12.1", + "@babel/helper-plugin-utils": "^7.10.4", + "@babel/helper-validator-identifier": "^7.10.4", + "babel-plugin-dynamic-import-node": "^2.3.3" + } + }, + "@babel/plugin-transform-modules-umd": { + "version": "7.12.1", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-modules-umd/-/plugin-transform-modules-umd-7.12.1.tgz", + "integrity": "sha512-aEIubCS0KHKM0zUos5fIoQm+AZUMt1ZvMpqz0/H5qAQ7vWylr9+PLYurT+Ic7ID/bKLd4q8hDovaG3Zch2uz5Q==", + "dev": true, + "requires": { + "@babel/helper-module-transforms": "^7.12.1", + "@babel/helper-plugin-utils": "^7.10.4" + } + }, + "@babel/plugin-transform-named-capturing-groups-regex": { + "version": "7.12.1", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-named-capturing-groups-regex/-/plugin-transform-named-capturing-groups-regex-7.12.1.tgz", + "integrity": "sha512-tB43uQ62RHcoDp9v2Nsf+dSM8sbNodbEicbQNA53zHz8pWUhsgHSJCGpt7daXxRydjb0KnfmB+ChXOv3oADp1Q==", + "dev": true, + "requires": { + "@babel/helper-create-regexp-features-plugin": "^7.12.1" + } + }, + "@babel/plugin-transform-new-target": { + "version": "7.12.1", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-new-target/-/plugin-transform-new-target-7.12.1.tgz", + "integrity": "sha512-+eW/VLcUL5L9IvJH7rT1sT0CzkdUTvPrXC2PXTn/7z7tXLBuKvezYbGdxD5WMRoyvyaujOq2fWoKl869heKjhw==", + "dev": true, + "requires": { + "@babel/helper-plugin-utils": "^7.10.4" + } + }, + "@babel/plugin-transform-object-super": { + "version": "7.12.1", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-object-super/-/plugin-transform-object-super-7.12.1.tgz", + "integrity": "sha512-AvypiGJH9hsquNUn+RXVcBdeE3KHPZexWRdimhuV59cSoOt5kFBmqlByorAeUlGG2CJWd0U+4ZtNKga/TB0cAw==", + "dev": true, + "requires": { + "@babel/helper-plugin-utils": "^7.10.4", + "@babel/helper-replace-supers": "^7.12.1" + } + }, + "@babel/plugin-transform-parameters": { + "version": "7.12.1", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-parameters/-/plugin-transform-parameters-7.12.1.tgz", + "integrity": "sha512-xq9C5EQhdPK23ZeCdMxl8bbRnAgHFrw5EOC3KJUsSylZqdkCaFEXxGSBuTSObOpiiHHNyb82es8M1QYgfQGfNg==", + "dev": true, + "requires": { + "@babel/helper-plugin-utils": "^7.10.4" + } + }, + "@babel/plugin-transform-property-literals": { + "version": "7.12.1", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-property-literals/-/plugin-transform-property-literals-7.12.1.tgz", + "integrity": "sha512-6MTCR/mZ1MQS+AwZLplX4cEySjCpnIF26ToWo942nqn8hXSm7McaHQNeGx/pt7suI1TWOWMfa/NgBhiqSnX0cQ==", + "dev": true, + "requires": { + "@babel/helper-plugin-utils": "^7.10.4" + } + }, + "@babel/plugin-transform-regenerator": { + "version": "7.12.1", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-regenerator/-/plugin-transform-regenerator-7.12.1.tgz", + "integrity": "sha512-gYrHqs5itw6i4PflFX3OdBPMQdPbF4bj2REIUxlMRUFk0/ZOAIpDFuViuxPjUL7YC8UPnf+XG7/utJvqXdPKng==", + "dev": true, + "requires": { + "regenerator-transform": "^0.14.2" + } + }, + "@babel/plugin-transform-reserved-words": { + "version": "7.12.1", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-reserved-words/-/plugin-transform-reserved-words-7.12.1.tgz", + "integrity": "sha512-pOnUfhyPKvZpVyBHhSBoX8vfA09b7r00Pmm1sH+29ae2hMTKVmSp4Ztsr8KBKjLjx17H0eJqaRC3bR2iThM54A==", + "dev": true, + "requires": { + "@babel/helper-plugin-utils": "^7.10.4" + } + }, + "@babel/plugin-transform-shorthand-properties": { + "version": "7.12.1", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-shorthand-properties/-/plugin-transform-shorthand-properties-7.12.1.tgz", + "integrity": "sha512-GFZS3c/MhX1OusqB1MZ1ct2xRzX5ppQh2JU1h2Pnfk88HtFTM+TWQqJNfwkmxtPQtb/s1tk87oENfXJlx7rSDw==", + "dev": true, + "requires": { + "@babel/helper-plugin-utils": "^7.10.4" + } + }, + "@babel/plugin-transform-spread": { + "version": "7.12.1", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-spread/-/plugin-transform-spread-7.12.1.tgz", + "integrity": "sha512-vuLp8CP0BE18zVYjsEBZ5xoCecMK6LBMMxYzJnh01rxQRvhNhH1csMMmBfNo5tGpGO+NhdSNW2mzIvBu3K1fng==", + "dev": true, + "requires": { + "@babel/helper-plugin-utils": "^7.10.4", + "@babel/helper-skip-transparent-expression-wrappers": "^7.12.1" + } + }, + "@babel/plugin-transform-sticky-regex": { + "version": "7.12.7", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-sticky-regex/-/plugin-transform-sticky-regex-7.12.7.tgz", + "integrity": "sha512-VEiqZL5N/QvDbdjfYQBhruN0HYjSPjC4XkeqW4ny/jNtH9gcbgaqBIXYEZCNnESMAGs0/K/R7oFGMhOyu/eIxg==", + "dev": true, + "requires": { + "@babel/helper-plugin-utils": "^7.10.4" + } + }, + "@babel/plugin-transform-template-literals": { + "version": "7.12.1", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-template-literals/-/plugin-transform-template-literals-7.12.1.tgz", + "integrity": "sha512-b4Zx3KHi+taXB1dVRBhVJtEPi9h1THCeKmae2qP0YdUHIFhVjtpqqNfxeVAa1xeHVhAy4SbHxEwx5cltAu5apw==", + "dev": true, + "requires": { + "@babel/helper-plugin-utils": "^7.10.4" + } + }, + "@babel/plugin-transform-typeof-symbol": { + "version": "7.12.1", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-typeof-symbol/-/plugin-transform-typeof-symbol-7.12.1.tgz", + "integrity": "sha512-EPGgpGy+O5Kg5pJFNDKuxt9RdmTgj5sgrus2XVeMp/ZIbOESadgILUbm50SNpghOh3/6yrbsH+NB5+WJTmsA7Q==", + "dev": true, + "requires": { + "@babel/helper-plugin-utils": "^7.10.4" + } + }, + "@babel/plugin-transform-unicode-escapes": { + "version": "7.12.1", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-unicode-escapes/-/plugin-transform-unicode-escapes-7.12.1.tgz", + "integrity": "sha512-I8gNHJLIc7GdApm7wkVnStWssPNbSRMPtgHdmH3sRM1zopz09UWPS4x5V4n1yz/MIWTVnJ9sp6IkuXdWM4w+2Q==", + "dev": true, + "requires": { + "@babel/helper-plugin-utils": "^7.10.4" + } + }, + "@babel/plugin-transform-unicode-regex": { + "version": "7.12.1", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-unicode-regex/-/plugin-transform-unicode-regex-7.12.1.tgz", + "integrity": "sha512-SqH4ClNngh/zGwHZOOQMTD+e8FGWexILV+ePMyiDJttAWRh5dhDL8rcl5lSgU3Huiq6Zn6pWTMvdPAb21Dwdyg==", + "dev": true, + "requires": { + "@babel/helper-create-regexp-features-plugin": "^7.12.1", + "@babel/helper-plugin-utils": "^7.10.4" + } + }, + "@babel/preset-env": { + "version": "7.12.7", + "resolved": "https://registry.npmjs.org/@babel/preset-env/-/preset-env-7.12.7.tgz", + "integrity": "sha512-OnNdfAr1FUQg7ksb7bmbKoby4qFOHw6DKWWUNB9KqnnCldxhxJlP+21dpyaWFmf2h0rTbOkXJtAGevY3XW1eew==", + "dev": true, + "requires": { + "@babel/compat-data": "^7.12.7", + "@babel/helper-compilation-targets": "^7.12.5", + "@babel/helper-module-imports": "^7.12.5", + "@babel/helper-plugin-utils": "^7.10.4", + "@babel/helper-validator-option": "^7.12.1", + "@babel/plugin-proposal-async-generator-functions": "^7.12.1", + "@babel/plugin-proposal-class-properties": "^7.12.1", + "@babel/plugin-proposal-dynamic-import": "^7.12.1", + "@babel/plugin-proposal-export-namespace-from": "^7.12.1", + "@babel/plugin-proposal-json-strings": "^7.12.1", + "@babel/plugin-proposal-logical-assignment-operators": "^7.12.1", + "@babel/plugin-proposal-nullish-coalescing-operator": "^7.12.1", + "@babel/plugin-proposal-numeric-separator": "^7.12.7", + "@babel/plugin-proposal-object-rest-spread": "^7.12.1", + "@babel/plugin-proposal-optional-catch-binding": "^7.12.1", + "@babel/plugin-proposal-optional-chaining": "^7.12.7", + "@babel/plugin-proposal-private-methods": "^7.12.1", + "@babel/plugin-proposal-unicode-property-regex": "^7.12.1", + "@babel/plugin-syntax-async-generators": "^7.8.0", + "@babel/plugin-syntax-class-properties": "^7.12.1", + "@babel/plugin-syntax-dynamic-import": "^7.8.0", + "@babel/plugin-syntax-export-namespace-from": "^7.8.3", + "@babel/plugin-syntax-json-strings": "^7.8.0", + "@babel/plugin-syntax-logical-assignment-operators": "^7.10.4", + "@babel/plugin-syntax-nullish-coalescing-operator": "^7.8.0", + "@babel/plugin-syntax-numeric-separator": "^7.10.4", + "@babel/plugin-syntax-object-rest-spread": "^7.8.0", + "@babel/plugin-syntax-optional-catch-binding": "^7.8.0", + "@babel/plugin-syntax-optional-chaining": "^7.8.0", + "@babel/plugin-syntax-top-level-await": "^7.12.1", + "@babel/plugin-transform-arrow-functions": "^7.12.1", + "@babel/plugin-transform-async-to-generator": "^7.12.1", + "@babel/plugin-transform-block-scoped-functions": "^7.12.1", + "@babel/plugin-transform-block-scoping": "^7.12.1", + "@babel/plugin-transform-classes": "^7.12.1", + "@babel/plugin-transform-computed-properties": "^7.12.1", + "@babel/plugin-transform-destructuring": "^7.12.1", + "@babel/plugin-transform-dotall-regex": "^7.12.1", + "@babel/plugin-transform-duplicate-keys": "^7.12.1", + "@babel/plugin-transform-exponentiation-operator": "^7.12.1", + "@babel/plugin-transform-for-of": "^7.12.1", + "@babel/plugin-transform-function-name": "^7.12.1", + "@babel/plugin-transform-literals": "^7.12.1", + "@babel/plugin-transform-member-expression-literals": "^7.12.1", + "@babel/plugin-transform-modules-amd": "^7.12.1", + "@babel/plugin-transform-modules-commonjs": "^7.12.1", + "@babel/plugin-transform-modules-systemjs": "^7.12.1", + "@babel/plugin-transform-modules-umd": "^7.12.1", + "@babel/plugin-transform-named-capturing-groups-regex": "^7.12.1", + "@babel/plugin-transform-new-target": "^7.12.1", + "@babel/plugin-transform-object-super": "^7.12.1", + "@babel/plugin-transform-parameters": "^7.12.1", + "@babel/plugin-transform-property-literals": "^7.12.1", + "@babel/plugin-transform-regenerator": "^7.12.1", + "@babel/plugin-transform-reserved-words": "^7.12.1", + "@babel/plugin-transform-shorthand-properties": "^7.12.1", + "@babel/plugin-transform-spread": "^7.12.1", + "@babel/plugin-transform-sticky-regex": "^7.12.7", + "@babel/plugin-transform-template-literals": "^7.12.1", + "@babel/plugin-transform-typeof-symbol": "^7.12.1", + "@babel/plugin-transform-unicode-escapes": "^7.12.1", + "@babel/plugin-transform-unicode-regex": "^7.12.1", + "@babel/preset-modules": "^0.1.3", + "@babel/types": "^7.12.7", + "core-js-compat": "^3.7.0", + "semver": "^5.5.0" + } + }, + "@babel/preset-modules": { + "version": "0.1.4", + "resolved": "https://registry.npmjs.org/@babel/preset-modules/-/preset-modules-0.1.4.tgz", + "integrity": "sha512-J36NhwnfdzpmH41M1DrnkkgAqhZaqr/NBdPfQ677mLzlaXo+oDiv1deyCDtgAhz8p328otdob0Du7+xgHGZbKg==", + "dev": true, + "requires": { + "@babel/helper-plugin-utils": "^7.0.0", + "@babel/plugin-proposal-unicode-property-regex": "^7.4.4", + "@babel/plugin-transform-dotall-regex": "^7.4.4", + "@babel/types": "^7.4.4", + "esutils": "^2.0.2" + } + }, + "@babel/runtime": { + "version": "7.12.5", + "resolved": "https://registry.npmjs.org/@babel/runtime/-/runtime-7.12.5.tgz", + "integrity": "sha512-plcc+hbExy3McchJCEQG3knOsuh3HH+Prx1P6cLIkET/0dLuQDEnrT+s27Axgc9bqfsmNUNHfscgMUdBpC9xfg==", + "dev": true, + "requires": { + "regenerator-runtime": "^0.13.4" + } + }, + "@babel/template": { + "version": "7.12.7", + "resolved": "https://registry.npmjs.org/@babel/template/-/template-7.12.7.tgz", + "integrity": "sha512-GkDzmHS6GV7ZeXfJZ0tLRBhZcMcY0/Lnb+eEbXDBfCAcZCjrZKe6p3J4we/D24O9Y8enxWAg1cWwof59yLh2ow==", + "requires": { + "@babel/code-frame": "^7.10.4", + "@babel/parser": "^7.12.7", + "@babel/types": "^7.12.7" + } + }, + "@babel/traverse": { + "version": "7.12.8", + "resolved": "https://registry.npmjs.org/@babel/traverse/-/traverse-7.12.8.tgz", + "integrity": "sha512-EIRQXPTwFEGRZyu6gXbjfpNORN1oZvwuzJbxcXjAgWV0iqXYDszN1Hx3FVm6YgZfu1ZQbCVAk3l+nIw95Xll9Q==", + "requires": { + "@babel/code-frame": "^7.10.4", + "@babel/generator": "^7.12.5", + "@babel/helper-function-name": "^7.10.4", + "@babel/helper-split-export-declaration": "^7.11.0", + "@babel/parser": "^7.12.7", + "@babel/types": "^7.12.7", + "debug": "^4.1.0", + "globals": "^11.1.0", + "lodash": "^4.17.19" + } + }, + "@babel/types": { + "version": "7.12.7", + "resolved": "https://registry.npmjs.org/@babel/types/-/types-7.12.7.tgz", + "integrity": "sha512-MNyI92qZq6jrQkXvtIiykvl4WtoRrVV9MPn+ZfsoEENjiWcBQ3ZSHrkxnJWgWtLX3XXqX5hrSQ+X69wkmesXuQ==", + "requires": { + "@babel/helper-validator-identifier": "^7.10.4", + "lodash": "^4.17.19", + "to-fast-properties": "^2.0.0" + } + }, + "@bcoe/v8-coverage": { + "version": "0.2.3", + "resolved": "https://registry.npmjs.org/@bcoe/v8-coverage/-/v8-coverage-0.2.3.tgz", + "integrity": "sha512-0hYQ8SB4Db5zvZB4axdMHGwEaQjkZzFjQiN9LVYvIFB2nSUHW9tYpxWriPrWDASIxiaXax83REcLxuSdnGPZtw==", + "dev": true + }, + "@cnakazawa/watch": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/@cnakazawa/watch/-/watch-1.0.4.tgz", + "integrity": "sha512-v9kIhKwjeZThiWrLmj0y17CWoyddASLj9O2yvbZkbvw/N3rWOYy9zkV66ursAoVr0mV15bL8g0c4QZUE6cdDoQ==", + "dev": true, + "requires": { + "exec-sh": "^0.3.2", + "minimist": "^1.2.0" + } + }, + "@fortawesome/angular-fontawesome": { + "version": "0.6.1", + "resolved": "https://registry.npmjs.org/@fortawesome/angular-fontawesome/-/angular-fontawesome-0.6.1.tgz", + "integrity": "sha512-ARQjtRuT+ZskzJDJKPwuiGO3+7nS0iyNLU/uHVJHfG4LwGJxwVIGldwg1SU957sra0Z0OtWEajHMhiS4vB9LwQ==" + }, + "@fortawesome/fontawesome-common-types": { + "version": "0.2.32", + "resolved": "https://registry.npmjs.org/@fortawesome/fontawesome-common-types/-/fontawesome-common-types-0.2.32.tgz", + "integrity": "sha512-ux2EDjKMpcdHBVLi/eWZynnPxs0BtFVXJkgHIxXRl+9ZFaHPvYamAfCzeeQFqHRjuJtX90wVnMRaMQAAlctz3w==" + }, + "@fortawesome/fontawesome-svg-core": { + "version": "1.2.29", + "resolved": "https://registry.npmjs.org/@fortawesome/fontawesome-svg-core/-/fontawesome-svg-core-1.2.29.tgz", + "integrity": "sha512-xmPmP2t8qrdo8RyKihTkGb09RnZoc+7HFBCnr0/6ZhStdGDSLeEd7ajV181+2W29NWIFfylO13rU+s3fpy3cnA==", + "requires": { + "@fortawesome/fontawesome-common-types": "^0.2.29" + } + }, + "@fortawesome/free-solid-svg-icons": { + "version": "5.13.1", + "resolved": "https://registry.npmjs.org/@fortawesome/free-solid-svg-icons/-/free-solid-svg-icons-5.13.1.tgz", + "integrity": "sha512-LQH/0L1p4+rqtoSHa9qFYR84hpuRZKqaQ41cfBQx8b68p21zoWSekTAeA54I/2x9VlCHDLFlG74Nmdg4iTPQOg==", + "requires": { + "@fortawesome/fontawesome-common-types": "^0.2.29" + } + }, + "@hapi/address": { + "version": "2.1.4", + "resolved": "https://registry.npmjs.org/@hapi/address/-/address-2.1.4.tgz", + "integrity": "sha512-QD1PhQk+s31P1ixsX0H0Suoupp3VMXzIVMSwobR3F3MSUO2YCV0B7xqLcUw/Bh8yuvd3LhpyqLQWTNcRmp6IdQ==", + "dev": true + }, + "@hapi/bourne": { + "version": "1.3.2", + "resolved": "https://registry.npmjs.org/@hapi/bourne/-/bourne-1.3.2.tgz", + "integrity": "sha512-1dVNHT76Uu5N3eJNTYcvxee+jzX4Z9lfciqRRHCU27ihbUcYi+iSc2iml5Ke1LXe1SyJCLA0+14Jh4tXJgOppA==", + "dev": true + }, + "@hapi/hoek": { + "version": "8.5.1", + "resolved": "https://registry.npmjs.org/@hapi/hoek/-/hoek-8.5.1.tgz", + "integrity": "sha512-yN7kbciD87WzLGc5539Tn0sApjyiGHAJgKvG9W8C7O+6c7qmoQMfVs0W4bX17eqz6C78QJqqFrtgdK5EWf6Qow==", + "dev": true + }, + "@hapi/joi": { + "version": "15.1.1", + "resolved": "https://registry.npmjs.org/@hapi/joi/-/joi-15.1.1.tgz", + "integrity": "sha512-entf8ZMOK8sc+8YfeOlM8pCfg3b5+WZIKBfUaaJT8UsjAAPjartzxIYm3TIbjvA4u+u++KbcXD38k682nVHDAQ==", + "dev": true, + "requires": { + "@hapi/address": "2.x.x", + "@hapi/bourne": "1.x.x", + "@hapi/hoek": "8.x.x", + "@hapi/topo": "3.x.x" + } + }, + "@hapi/topo": { + "version": "3.1.6", + "resolved": "https://registry.npmjs.org/@hapi/topo/-/topo-3.1.6.tgz", + "integrity": "sha512-tAag0jEcjwH+P2quUfipd7liWCNX2F8NvYjQp2wtInsZxnMlypdw0FtAOLxtvvkO+GSRRbmNi8m/5y42PQJYCQ==", + "dev": true, + "requires": { + "@hapi/hoek": "^8.3.0" + } + }, + "@istanbuljs/load-nyc-config": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/@istanbuljs/load-nyc-config/-/load-nyc-config-1.1.0.tgz", + "integrity": "sha512-VjeHSlIzpv/NyD3N0YuHfXOPDIixcA1q2ZV98wsMqcYlPmv2n3Yb2lYP9XMElnaFVXg5A7YLTeLu6V84uQDjmQ==", + "dev": true, + "requires": { + "camelcase": "^5.3.1", + "find-up": "^4.1.0", + "get-package-type": "^0.1.0", + "js-yaml": "^3.13.1", + "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 + } + } + }, + "@istanbuljs/schema": { + "version": "0.1.2", + "resolved": "https://registry.npmjs.org/@istanbuljs/schema/-/schema-0.1.2.tgz", + "integrity": "sha512-tsAQNx32a8CoFhjhijUIhI4kccIAgmGhy8LZMZgGfmXcpMbPRUqn5LWmgRttILi6yeGmBJd2xsPkFMs0PzgPCw==", + "dev": true + }, + "@jest/console": { + "version": "26.6.2", + "resolved": "https://registry.npmjs.org/@jest/console/-/console-26.6.2.tgz", + "integrity": "sha512-IY1R2i2aLsLr7Id3S6p2BA82GNWryt4oSvEXLAKc+L2zdi89dSkE8xC1C+0kpATG4JhBJREnQOH7/zmccM2B0g==", + "dev": true, + "requires": { + "@jest/types": "^26.6.2", + "@types/node": "*", + "chalk": "^4.0.0", + "jest-message-util": "^26.6.2", + "jest-util": "^26.6.2", + "slash": "^3.0.0" + }, + "dependencies": { + "@jest/types": { + "version": "26.6.2", + "resolved": "https://registry.npmjs.org/@jest/types/-/types-26.6.2.tgz", + "integrity": "sha512-fC6QCp7Sc5sX6g8Tvbmj4XUTbyrik0akgRy03yjXbQaBWWNWGE7SGtJk98m0N8nzegD/7SggrUlivxo5ax4KWQ==", + "dev": true, + "requires": { + "@types/istanbul-lib-coverage": "^2.0.0", + "@types/istanbul-reports": "^3.0.0", + "@types/node": "*", + "@types/yargs": "^15.0.0", + "chalk": "^4.0.0" + } + }, + "@types/istanbul-reports": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/@types/istanbul-reports/-/istanbul-reports-3.0.0.tgz", + "integrity": "sha512-nwKNbvnwJ2/mndE9ItP/zc2TCzw6uuodnF4EHYWD+gCQDVBuRQL5UzbZD0/ezy1iKsFU2ZQiDqg4M9dN4+wZgA==", + "dev": true, + "requires": { + "@types/istanbul-lib-report": "*" + } + }, + "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.0", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.0.tgz", + "integrity": "sha512-qwx12AxXe2Q5xQ43Ac//I6v5aXTipYrSESdOgzrN+9XjgEpyjpKuvSGaN4qE93f7TQTlerQQ8S+EQ0EyDoVL1A==", + "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 + }, + "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" + } + } + } + }, + "@jest/core": { + "version": "26.6.3", + "resolved": "https://registry.npmjs.org/@jest/core/-/core-26.6.3.tgz", + "integrity": "sha512-xvV1kKbhfUqFVuZ8Cyo+JPpipAHHAV3kcDBftiduK8EICXmTFddryy3P7NfZt8Pv37rA9nEJBKCCkglCPt/Xjw==", + "dev": true, + "requires": { + "@jest/console": "^26.6.2", + "@jest/reporters": "^26.6.2", + "@jest/test-result": "^26.6.2", + "@jest/transform": "^26.6.2", + "@jest/types": "^26.6.2", + "@types/node": "*", + "ansi-escapes": "^4.2.1", + "chalk": "^4.0.0", + "exit": "^0.1.2", + "graceful-fs": "^4.2.4", + "jest-changed-files": "^26.6.2", + "jest-config": "^26.6.3", + "jest-haste-map": "^26.6.2", + "jest-message-util": "^26.6.2", + "jest-regex-util": "^26.0.0", + "jest-resolve": "^26.6.2", + "jest-resolve-dependencies": "^26.6.3", + "jest-runner": "^26.6.3", + "jest-runtime": "^26.6.3", + "jest-snapshot": "^26.6.2", + "jest-util": "^26.6.2", + "jest-validate": "^26.6.2", + "jest-watcher": "^26.6.2", + "micromatch": "^4.0.2", + "p-each-series": "^2.1.0", + "rimraf": "^3.0.0", + "slash": "^3.0.0", + "strip-ansi": "^6.0.0" + }, + "dependencies": { + "@jest/types": { + "version": "26.6.2", + "resolved": "https://registry.npmjs.org/@jest/types/-/types-26.6.2.tgz", + "integrity": "sha512-fC6QCp7Sc5sX6g8Tvbmj4XUTbyrik0akgRy03yjXbQaBWWNWGE7SGtJk98m0N8nzegD/7SggrUlivxo5ax4KWQ==", + "dev": true, + "requires": { + "@types/istanbul-lib-coverage": "^2.0.0", + "@types/istanbul-reports": "^3.0.0", + "@types/node": "*", + "@types/yargs": "^15.0.0", + "chalk": "^4.0.0" + } + }, + "@types/istanbul-reports": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/@types/istanbul-reports/-/istanbul-reports-3.0.0.tgz", + "integrity": "sha512-nwKNbvnwJ2/mndE9ItP/zc2TCzw6uuodnF4EHYWD+gCQDVBuRQL5UzbZD0/ezy1iKsFU2ZQiDqg4M9dN4+wZgA==", + "dev": true, + "requires": { + "@types/istanbul-lib-report": "*" + } + }, + "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.0", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.0.tgz", + "integrity": "sha512-qwx12AxXe2Q5xQ43Ac//I6v5aXTipYrSESdOgzrN+9XjgEpyjpKuvSGaN4qE93f7TQTlerQQ8S+EQ0EyDoVL1A==", + "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 + }, + "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 + }, + "micromatch": { + "version": "4.0.2", + "resolved": "https://registry.npmjs.org/micromatch/-/micromatch-4.0.2.tgz", + "integrity": "sha512-y7FpHSbMUMoyPbYUSzO6PaZ6FyRnQOpHuKwbo1G+Knck95XVU4QAiKdGEnj5wwoS7PlOgthX/09u5iFJ+aYf5Q==", + "dev": true, + "requires": { + "braces": "^3.0.1", + "picomatch": "^2.0.5" + } + }, + "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" + } + } + } + }, + "@jest/environment": { + "version": "26.6.2", + "resolved": "https://registry.npmjs.org/@jest/environment/-/environment-26.6.2.tgz", + "integrity": "sha512-nFy+fHl28zUrRsCeMB61VDThV1pVTtlEokBRgqPrcT1JNq4yRNIyTHfyht6PqtUvY9IsuLGTrbG8kPXjSZIZwA==", + "dev": true, + "requires": { + "@jest/fake-timers": "^26.6.2", + "@jest/types": "^26.6.2", + "@types/node": "*", + "jest-mock": "^26.6.2" + }, + "dependencies": { + "@jest/types": { + "version": "26.6.2", + "resolved": "https://registry.npmjs.org/@jest/types/-/types-26.6.2.tgz", + "integrity": "sha512-fC6QCp7Sc5sX6g8Tvbmj4XUTbyrik0akgRy03yjXbQaBWWNWGE7SGtJk98m0N8nzegD/7SggrUlivxo5ax4KWQ==", + "dev": true, + "requires": { + "@types/istanbul-lib-coverage": "^2.0.0", + "@types/istanbul-reports": "^3.0.0", + "@types/node": "*", + "@types/yargs": "^15.0.0", + "chalk": "^4.0.0" + } + }, + "@types/istanbul-reports": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/@types/istanbul-reports/-/istanbul-reports-3.0.0.tgz", + "integrity": "sha512-nwKNbvnwJ2/mndE9ItP/zc2TCzw6uuodnF4EHYWD+gCQDVBuRQL5UzbZD0/ezy1iKsFU2ZQiDqg4M9dN4+wZgA==", + "dev": true, + "requires": { + "@types/istanbul-lib-report": "*" + } + }, + "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.0", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.0.tgz", + "integrity": "sha512-qwx12AxXe2Q5xQ43Ac//I6v5aXTipYrSESdOgzrN+9XjgEpyjpKuvSGaN4qE93f7TQTlerQQ8S+EQ0EyDoVL1A==", + "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 + }, + "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" + } + } + } + }, + "@jest/fake-timers": { + "version": "26.6.2", + "resolved": "https://registry.npmjs.org/@jest/fake-timers/-/fake-timers-26.6.2.tgz", + "integrity": "sha512-14Uleatt7jdzefLPYM3KLcnUl1ZNikaKq34enpb5XG9i81JpppDb5muZvonvKyrl7ftEHkKS5L5/eB/kxJ+bvA==", + "dev": true, + "requires": { + "@jest/types": "^26.6.2", + "@sinonjs/fake-timers": "^6.0.1", + "@types/node": "*", + "jest-message-util": "^26.6.2", + "jest-mock": "^26.6.2", + "jest-util": "^26.6.2" + }, + "dependencies": { + "@jest/types": { + "version": "26.6.2", + "resolved": "https://registry.npmjs.org/@jest/types/-/types-26.6.2.tgz", + "integrity": "sha512-fC6QCp7Sc5sX6g8Tvbmj4XUTbyrik0akgRy03yjXbQaBWWNWGE7SGtJk98m0N8nzegD/7SggrUlivxo5ax4KWQ==", + "dev": true, + "requires": { + "@types/istanbul-lib-coverage": "^2.0.0", + "@types/istanbul-reports": "^3.0.0", + "@types/node": "*", + "@types/yargs": "^15.0.0", + "chalk": "^4.0.0" + } + }, + "@types/istanbul-reports": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/@types/istanbul-reports/-/istanbul-reports-3.0.0.tgz", + "integrity": "sha512-nwKNbvnwJ2/mndE9ItP/zc2TCzw6uuodnF4EHYWD+gCQDVBuRQL5UzbZD0/ezy1iKsFU2ZQiDqg4M9dN4+wZgA==", + "dev": true, + "requires": { + "@types/istanbul-lib-report": "*" + } + }, + "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.0", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.0.tgz", + "integrity": "sha512-qwx12AxXe2Q5xQ43Ac//I6v5aXTipYrSESdOgzrN+9XjgEpyjpKuvSGaN4qE93f7TQTlerQQ8S+EQ0EyDoVL1A==", + "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 + }, + "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" + } + } + } + }, + "@jest/globals": { + "version": "26.6.2", + "resolved": "https://registry.npmjs.org/@jest/globals/-/globals-26.6.2.tgz", + "integrity": "sha512-85Ltnm7HlB/KesBUuALwQ68YTU72w9H2xW9FjZ1eL1U3lhtefjjl5c2MiUbpXt/i6LaPRvoOFJ22yCBSfQ0JIA==", + "dev": true, + "requires": { + "@jest/environment": "^26.6.2", + "@jest/types": "^26.6.2", + "expect": "^26.6.2" + }, + "dependencies": { + "@jest/types": { + "version": "26.6.2", + "resolved": "https://registry.npmjs.org/@jest/types/-/types-26.6.2.tgz", + "integrity": "sha512-fC6QCp7Sc5sX6g8Tvbmj4XUTbyrik0akgRy03yjXbQaBWWNWGE7SGtJk98m0N8nzegD/7SggrUlivxo5ax4KWQ==", + "dev": true, + "requires": { + "@types/istanbul-lib-coverage": "^2.0.0", + "@types/istanbul-reports": "^3.0.0", + "@types/node": "*", + "@types/yargs": "^15.0.0", + "chalk": "^4.0.0" + } + }, + "@types/istanbul-reports": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/@types/istanbul-reports/-/istanbul-reports-3.0.0.tgz", + "integrity": "sha512-nwKNbvnwJ2/mndE9ItP/zc2TCzw6uuodnF4EHYWD+gCQDVBuRQL5UzbZD0/ezy1iKsFU2ZQiDqg4M9dN4+wZgA==", + "dev": true, + "requires": { + "@types/istanbul-lib-report": "*" + } + }, + "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.0", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.0.tgz", + "integrity": "sha512-qwx12AxXe2Q5xQ43Ac//I6v5aXTipYrSESdOgzrN+9XjgEpyjpKuvSGaN4qE93f7TQTlerQQ8S+EQ0EyDoVL1A==", + "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 + }, + "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" + } + } + } + }, + "@jest/reporters": { + "version": "26.6.2", + "resolved": "https://registry.npmjs.org/@jest/reporters/-/reporters-26.6.2.tgz", + "integrity": "sha512-h2bW53APG4HvkOnVMo8q3QXa6pcaNt1HkwVsOPMBV6LD/q9oSpxNSYZQYkAnjdMjrJ86UuYeLo+aEZClV6opnw==", + "dev": true, + "requires": { + "@bcoe/v8-coverage": "^0.2.3", + "@jest/console": "^26.6.2", + "@jest/test-result": "^26.6.2", + "@jest/transform": "^26.6.2", + "@jest/types": "^26.6.2", + "chalk": "^4.0.0", + "collect-v8-coverage": "^1.0.0", + "exit": "^0.1.2", + "glob": "^7.1.2", + "graceful-fs": "^4.2.4", + "istanbul-lib-coverage": "^3.0.0", + "istanbul-lib-instrument": "^4.0.3", + "istanbul-lib-report": "^3.0.0", + "istanbul-lib-source-maps": "^4.0.0", + "istanbul-reports": "^3.0.2", + "jest-haste-map": "^26.6.2", + "jest-resolve": "^26.6.2", + "jest-util": "^26.6.2", + "jest-worker": "^26.6.2", + "node-notifier": "^8.0.0", + "slash": "^3.0.0", + "source-map": "^0.6.0", + "string-length": "^4.0.1", + "terminal-link": "^2.0.0", + "v8-to-istanbul": "^7.0.0" + }, + "dependencies": { + "@jest/types": { + "version": "26.6.2", + "resolved": "https://registry.npmjs.org/@jest/types/-/types-26.6.2.tgz", + "integrity": "sha512-fC6QCp7Sc5sX6g8Tvbmj4XUTbyrik0akgRy03yjXbQaBWWNWGE7SGtJk98m0N8nzegD/7SggrUlivxo5ax4KWQ==", + "dev": true, + "requires": { + "@types/istanbul-lib-coverage": "^2.0.0", + "@types/istanbul-reports": "^3.0.0", + "@types/node": "*", + "@types/yargs": "^15.0.0", + "chalk": "^4.0.0" + } + }, + "@types/istanbul-reports": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/@types/istanbul-reports/-/istanbul-reports-3.0.0.tgz", + "integrity": "sha512-nwKNbvnwJ2/mndE9ItP/zc2TCzw6uuodnF4EHYWD+gCQDVBuRQL5UzbZD0/ezy1iKsFU2ZQiDqg4M9dN4+wZgA==", + "dev": true, + "requires": { + "@types/istanbul-lib-report": "*" + } + }, + "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.0", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.0.tgz", + "integrity": "sha512-qwx12AxXe2Q5xQ43Ac//I6v5aXTipYrSESdOgzrN+9XjgEpyjpKuvSGaN4qE93f7TQTlerQQ8S+EQ0EyDoVL1A==", + "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 + }, + "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 + }, + "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 + }, + "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" + } + } + } + }, + "@jest/source-map": { + "version": "26.6.2", + "resolved": "https://registry.npmjs.org/@jest/source-map/-/source-map-26.6.2.tgz", + "integrity": "sha512-YwYcCwAnNmOVsZ8mr3GfnzdXDAl4LaenZP5z+G0c8bzC9/dugL8zRmxZzdoTl4IaS3CryS1uWnROLPFmb6lVvA==", + "dev": true, + "requires": { + "callsites": "^3.0.0", + "graceful-fs": "^4.2.4", + "source-map": "^0.6.0" + }, + "dependencies": { + "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 + } + } + }, + "@jest/test-result": { + "version": "26.6.2", + "resolved": "https://registry.npmjs.org/@jest/test-result/-/test-result-26.6.2.tgz", + "integrity": "sha512-5O7H5c/7YlojphYNrK02LlDIV2GNPYisKwHm2QTKjNZeEzezCbwYs9swJySv2UfPMyZ0VdsmMv7jIlD/IKYQpQ==", + "dev": true, + "requires": { + "@jest/console": "^26.6.2", + "@jest/types": "^26.6.2", + "@types/istanbul-lib-coverage": "^2.0.0", + "collect-v8-coverage": "^1.0.0" + }, + "dependencies": { + "@jest/types": { + "version": "26.6.2", + "resolved": "https://registry.npmjs.org/@jest/types/-/types-26.6.2.tgz", + "integrity": "sha512-fC6QCp7Sc5sX6g8Tvbmj4XUTbyrik0akgRy03yjXbQaBWWNWGE7SGtJk98m0N8nzegD/7SggrUlivxo5ax4KWQ==", + "dev": true, + "requires": { + "@types/istanbul-lib-coverage": "^2.0.0", + "@types/istanbul-reports": "^3.0.0", + "@types/node": "*", + "@types/yargs": "^15.0.0", + "chalk": "^4.0.0" + } + }, + "@types/istanbul-reports": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/@types/istanbul-reports/-/istanbul-reports-3.0.0.tgz", + "integrity": "sha512-nwKNbvnwJ2/mndE9ItP/zc2TCzw6uuodnF4EHYWD+gCQDVBuRQL5UzbZD0/ezy1iKsFU2ZQiDqg4M9dN4+wZgA==", + "dev": true, + "requires": { + "@types/istanbul-lib-report": "*" + } + }, + "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.0", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.0.tgz", + "integrity": "sha512-qwx12AxXe2Q5xQ43Ac//I6v5aXTipYrSESdOgzrN+9XjgEpyjpKuvSGaN4qE93f7TQTlerQQ8S+EQ0EyDoVL1A==", + "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 + }, + "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" + } + } + } + }, + "@jest/test-sequencer": { + "version": "26.6.3", + "resolved": "https://registry.npmjs.org/@jest/test-sequencer/-/test-sequencer-26.6.3.tgz", + "integrity": "sha512-YHlVIjP5nfEyjlrSr8t/YdNfU/1XEt7c5b4OxcXCjyRhjzLYu/rO69/WHPuYcbCWkz8kAeZVZp2N2+IOLLEPGw==", + "dev": true, + "requires": { + "@jest/test-result": "^26.6.2", + "graceful-fs": "^4.2.4", + "jest-haste-map": "^26.6.2", + "jest-runner": "^26.6.3", + "jest-runtime": "^26.6.3" + } + }, + "@jest/transform": { + "version": "26.6.2", + "resolved": "https://registry.npmjs.org/@jest/transform/-/transform-26.6.2.tgz", + "integrity": "sha512-E9JjhUgNzvuQ+vVAL21vlyfy12gP0GhazGgJC4h6qUt1jSdUXGWJ1wfu/X7Sd8etSgxV4ovT1pb9v5D6QW4XgA==", + "dev": true, + "requires": { + "@babel/core": "^7.1.0", + "@jest/types": "^26.6.2", + "babel-plugin-istanbul": "^6.0.0", + "chalk": "^4.0.0", + "convert-source-map": "^1.4.0", + "fast-json-stable-stringify": "^2.0.0", + "graceful-fs": "^4.2.4", + "jest-haste-map": "^26.6.2", + "jest-regex-util": "^26.0.0", + "jest-util": "^26.6.2", + "micromatch": "^4.0.2", + "pirates": "^4.0.1", + "slash": "^3.0.0", + "source-map": "^0.6.1", + "write-file-atomic": "^3.0.0" + }, + "dependencies": { + "@jest/types": { + "version": "26.6.2", + "resolved": "https://registry.npmjs.org/@jest/types/-/types-26.6.2.tgz", + "integrity": "sha512-fC6QCp7Sc5sX6g8Tvbmj4XUTbyrik0akgRy03yjXbQaBWWNWGE7SGtJk98m0N8nzegD/7SggrUlivxo5ax4KWQ==", + "dev": true, + "requires": { + "@types/istanbul-lib-coverage": "^2.0.0", + "@types/istanbul-reports": "^3.0.0", + "@types/node": "*", + "@types/yargs": "^15.0.0", + "chalk": "^4.0.0" + } + }, + "@types/istanbul-reports": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/@types/istanbul-reports/-/istanbul-reports-3.0.0.tgz", + "integrity": "sha512-nwKNbvnwJ2/mndE9ItP/zc2TCzw6uuodnF4EHYWD+gCQDVBuRQL5UzbZD0/ezy1iKsFU2ZQiDqg4M9dN4+wZgA==", + "dev": true, + "requires": { + "@types/istanbul-lib-report": "*" + } + }, + "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.0", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.0.tgz", + "integrity": "sha512-qwx12AxXe2Q5xQ43Ac//I6v5aXTipYrSESdOgzrN+9XjgEpyjpKuvSGaN4qE93f7TQTlerQQ8S+EQ0EyDoVL1A==", + "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 + }, + "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 + }, + "micromatch": { + "version": "4.0.2", + "resolved": "https://registry.npmjs.org/micromatch/-/micromatch-4.0.2.tgz", + "integrity": "sha512-y7FpHSbMUMoyPbYUSzO6PaZ6FyRnQOpHuKwbo1G+Knck95XVU4QAiKdGEnj5wwoS7PlOgthX/09u5iFJ+aYf5Q==", + "dev": true, + "requires": { + "braces": "^3.0.1", + "picomatch": "^2.0.5" + } + }, + "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 + }, + "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" + } + } + } + }, + "@jest/types": { + "version": "25.5.0", + "resolved": "https://registry.npmjs.org/@jest/types/-/types-25.5.0.tgz", + "integrity": "sha512-OXD0RgQ86Tu3MazKo8bnrkDRaDXXMGUqd+kTtLtK1Zb7CRzQcaSRPPPV37SvYTdevXEBVxe0HXylEjs8ibkmCw==", + "dev": true, + "requires": { + "@types/istanbul-lib-coverage": "^2.0.0", + "@types/istanbul-reports": "^1.1.1", + "@types/yargs": "^15.0.0", + "chalk": "^3.0.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": "3.0.0", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-3.0.0.tgz", + "integrity": "sha512-4D3B6Wf41KOYRFdszmDqMCGq5VV/uMAB273JILmO+3jAlh8X4qDtdtgCR3fxtbLEMzSx22QdhnDcJvu2u1fVwg==", + "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 + }, + "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" + } + } + } + }, + "@mrmlnc/readdir-enhanced": { + "version": "2.2.1", + "resolved": "https://registry.npmjs.org/@mrmlnc/readdir-enhanced/-/readdir-enhanced-2.2.1.tgz", + "integrity": "sha512-bPHp6Ji8b41szTOcaP63VlnbbO5Ny6dwAATtY6JTjh5N2OLrb5Qk/Th5cRkRQhkWCt+EJsYrNB0MiL+Gpn6e3g==", + "dev": true, + "requires": { + "call-me-maybe": "^1.0.1", + "glob-to-regexp": "^0.3.0" + } + }, + "@ng-bootstrap/ng-bootstrap": { + "version": "6.1.0", + "resolved": "https://registry.npmjs.org/@ng-bootstrap/ng-bootstrap/-/ng-bootstrap-6.1.0.tgz", + "integrity": "sha512-2GzkNJBKdeHkaUqaCAqSILPft0IzzHjMfAlAuGY6/ZLlVQ0glt5MTbIXtIhSbjR+OvlrljoXFLrvzs1LGdmE+A==" + }, + "@ngtools/webpack": { + "version": "10.0.0", + "resolved": "https://registry.npmjs.org/@ngtools/webpack/-/webpack-10.0.0.tgz", + "integrity": "sha512-2mnuzkCokn84PrBIiSwUqZImd5uKdsdZcpnuvgABmbxe3bAPxbOgzud/bLxG1/ynLvk/vd/FCNeDcY6HylB8OQ==", + "dev": true, + "requires": { + "@angular-devkit/core": "10.0.0", + "enhanced-resolve": "4.1.1", + "rxjs": "6.5.5", + "webpack-sources": "1.4.3" + } + }, + "@ngx-translate/core": { + "version": "12.1.2", + "resolved": "https://registry.npmjs.org/@ngx-translate/core/-/core-12.1.2.tgz", + "integrity": "sha512-ZudJsqIxTKlLmPoqK8gJY3UpMGujR0Xm7HfXL6AR79yGRS23QqpjAhMfx4v5qUCcHMmQ9/78bW8QJLfp31c7vQ==" + }, + "@ngx-translate/http-loader": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/@ngx-translate/http-loader/-/http-loader-5.0.0.tgz", + "integrity": "sha512-8+aV7N52qed+6t4LIu4Yru/PkeBX4TR2ioXGwXzQE5syqSLTj/8TgKQIi3i2Z61ZhPxQG1qrGbapUoGQzUDVeg==" + }, + "@nodelib/fs.scandir": { + "version": "2.1.3", + "resolved": "https://registry.npmjs.org/@nodelib/fs.scandir/-/fs.scandir-2.1.3.tgz", + "integrity": "sha512-eGmwYQn3gxo4r7jdQnkrrN6bY478C3P+a/y72IJukF8LjB6ZHeB3c+Ehacj3sYeSmUXGlnA67/PmbM9CVwL7Dw==", + "dev": true, + "requires": { + "@nodelib/fs.stat": "2.0.3", + "run-parallel": "^1.1.9" + } + }, + "@nodelib/fs.stat": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/@nodelib/fs.stat/-/fs.stat-2.0.3.tgz", + "integrity": "sha512-bQBFruR2TAwoevBEd/NWMoAAtNGzTRgdrqnYCc7dhzfoNvqPzLyqlEQnzZ3kVnNrSp25iyxE00/3h2fqGAGArA==", + "dev": true + }, + "@nodelib/fs.walk": { + "version": "1.2.4", + "resolved": "https://registry.npmjs.org/@nodelib/fs.walk/-/fs.walk-1.2.4.tgz", + "integrity": "sha512-1V9XOY4rDW0rehzbrcqAmHnz8e7SKvX27gh8Gt2WgB0+pdzdiLV83p72kZPU+jvMbS1qU5mauP2iOvO8rhmurQ==", + "dev": true, + "requires": { + "@nodelib/fs.scandir": "2.1.3", + "fastq": "^1.6.0" + } + }, + "@npmcli/move-file": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/@npmcli/move-file/-/move-file-1.0.1.tgz", + "integrity": "sha512-Uv6h1sT+0DrblvIrolFtbvM1FgWm+/sy4B3pvLp67Zys+thcukzS5ekn7HsZFGpWP4Q3fYJCljbWQE/XivMRLw==", + "dev": true, + "requires": { + "mkdirp": "^1.0.4" + }, + "dependencies": { + "mkdirp": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/mkdirp/-/mkdirp-1.0.4.tgz", + "integrity": "sha512-vVqVZQyf3WLx2Shd0qJ9xuvqgAyKPLAiqITEtqW0oIUjzo3PePDd6fW9iFz30ef7Ysp/oiWqbhszeGWW2T6Gzw==", + "dev": true + } + } + }, + "@openapitools/openapi-generator-cli": { + "version": "1.0.13-4.3.1", + "resolved": "https://registry.npmjs.org/@openapitools/openapi-generator-cli/-/openapi-generator-cli-1.0.13-4.3.1.tgz", + "integrity": "sha512-hUbSHCXak8vujMpTgH5dDACq0mynya37xgPFD3JbmnwRrPrgrZ4mr3+w7QZJ5aLMCIfb2j3s96d1sWaOZYZzxA==", + "dev": true + }, + "@rollup/plugin-node-resolve": { + "version": "7.1.3", + "resolved": "https://registry.npmjs.org/@rollup/plugin-node-resolve/-/plugin-node-resolve-7.1.3.tgz", + "integrity": "sha512-RxtSL3XmdTAE2byxekYLnx+98kEUOrPHF/KRVjLH+DEIHy6kjIw7YINQzn+NXiH/NTrQLAwYs0GWB+csWygA9Q==", + "dev": true, + "requires": { + "@rollup/pluginutils": "^3.0.8", + "@types/resolve": "0.0.8", + "builtin-modules": "^3.1.0", + "is-module": "^1.0.0", + "resolve": "^1.14.2" + }, + "dependencies": { + "builtin-modules": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/builtin-modules/-/builtin-modules-3.1.0.tgz", + "integrity": "sha512-k0KL0aWZuBt2lrxrcASWDfwOLMnodeQjodT/1SxEQAXsHANgo6ZC/VEaSEHCXt7aSTZ4/4H5LKa+tBXmW7Vtvw==", + "dev": true + } + } + }, + "@rollup/plugin-replace": { + "version": "2.3.4", + "resolved": "https://registry.npmjs.org/@rollup/plugin-replace/-/plugin-replace-2.3.4.tgz", + "integrity": "sha512-waBhMzyAtjCL1GwZes2jaE9MjuQ/DQF2BatH3fRivUF3z0JBFrU0U6iBNC/4WR+2rLKhaAhPWDNPYp4mI6RqdQ==", + "dev": true, + "requires": { + "@rollup/pluginutils": "^3.1.0", + "magic-string": "^0.25.7" + } + }, + "@rollup/pluginutils": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/@rollup/pluginutils/-/pluginutils-3.1.0.tgz", + "integrity": "sha512-GksZ6pr6TpIjHm8h9lSQ8pi8BE9VeubNT0OMJ3B5uZJ8pz73NPiqOtCog/x2/QzM1ENChPKxMDhiQuRHsqc+lg==", + "dev": true, + "requires": { + "@types/estree": "0.0.39", + "estree-walker": "^1.0.1", + "picomatch": "^2.2.2" + } + }, + "@samverschueren/stream-to-observable": { + "version": "0.3.1", + "resolved": "https://registry.npmjs.org/@samverschueren/stream-to-observable/-/stream-to-observable-0.3.1.tgz", + "integrity": "sha512-c/qwwcHyafOQuVQJj0IlBjf5yYgBI7YPJ77k4fOJYesb41jio65eaJODRUmfYKhTOFBrIZ66kgvGPlNbjuoRdQ==", + "dev": true, + "requires": { + "any-observable": "^0.3.0" + } + }, + "@scarf/scarf": { + "version": "0.1.5", + "resolved": "https://registry.npmjs.org/@scarf/scarf/-/scarf-0.1.5.tgz", + "integrity": "sha512-Fx6atDc7JM1r0WkPCDhNetVZNp+DO21q/HGlomAKBG+k8vb1B8fg8Yige4oCf1P9OWTZWm5tM5i3jlXhrSbNOg==" + }, + "@schematics/angular": { + "version": "10.0.0", + "resolved": "https://registry.npmjs.org/@schematics/angular/-/angular-10.0.0.tgz", + "integrity": "sha512-m7Pxz4guAMbe7NASKCPUNxvUX/LeieDjGsXwIt09tVE4dEi9yqJP5zq8kOnZEiLKKflP7GoB65RNex4dTxsydw==", + "dev": true, + "requires": { + "@angular-devkit/core": "10.0.0", + "@angular-devkit/schematics": "10.0.0" + } + }, + "@schematics/update": { + "version": "0.1000.0", + "resolved": "https://registry.npmjs.org/@schematics/update/-/update-0.1000.0.tgz", + "integrity": "sha512-snjz7sQCOn4Xi66XQREXZx9K6R/vAnUfdyO5nXekls8+E+MIowlP+gqHM0whi8qJDwCLd9maYmeVsD6XZaGImQ==", + "dev": true, + "requires": { + "@angular-devkit/core": "10.0.0", + "@angular-devkit/schematics": "10.0.0", + "@yarnpkg/lockfile": "1.1.0", + "ini": "1.3.5", + "npm-package-arg": "^8.0.0", + "pacote": "9.5.12", + "rxjs": "6.5.5", + "semver": "7.3.2", + "semver-intersect": "1.4.0" + }, + "dependencies": { + "semver": { + "version": "7.3.2", + "resolved": "https://registry.npmjs.org/semver/-/semver-7.3.2.tgz", + "integrity": "sha512-OrOb32TeeambH6UrhtShmF7CRDqhL6/5XpPNp2DuRH6+9QLw/orhp72j87v8Qa1ScDkvrrBNpZcDejAirJmfXQ==", + "dev": true + } + } + }, + "@sinonjs/commons": { + "version": "1.8.1", + "resolved": "https://registry.npmjs.org/@sinonjs/commons/-/commons-1.8.1.tgz", + "integrity": "sha512-892K+kWUUi3cl+LlqEWIDrhvLgdL79tECi8JZUyq6IviKy/DNhuzCRlbHUjxK89f4ypPMMaFnFuR9Ie6DoIMsw==", + "dev": true, + "requires": { + "type-detect": "4.0.8" + } + }, + "@sinonjs/fake-timers": { + "version": "6.0.1", + "resolved": "https://registry.npmjs.org/@sinonjs/fake-timers/-/fake-timers-6.0.1.tgz", + "integrity": "sha512-MZPUxrmFubI36XS1DI3qmI0YdN1gks62JtFZvxR67ljjSNCeK6U08Zx4msEWOXuofgqUt6zPHSi1H9fbjR/NRA==", + "dev": true, + "requires": { + "@sinonjs/commons": "^1.7.0" + } + }, + "@surma/rollup-plugin-off-main-thread": { + "version": "1.4.2", + "resolved": "https://registry.npmjs.org/@surma/rollup-plugin-off-main-thread/-/rollup-plugin-off-main-thread-1.4.2.tgz", + "integrity": "sha512-yBMPqmd1yEJo/280PAMkychuaALyQ9Lkb5q1ck3mjJrFuEobIfhnQ4J3mbvBoISmR3SWMWV+cGB/I0lCQee79A==", + "dev": true, + "requires": { + "ejs": "^2.6.1", + "magic-string": "^0.25.0" + }, + "dependencies": { + "ejs": { + "version": "2.7.4", + "resolved": "https://registry.npmjs.org/ejs/-/ejs-2.7.4.tgz", + "integrity": "sha512-7vmuyh5+kuUyJKePhQfRQBhXV5Ce+RnaeeQArKu1EAMpL3WbgMt5WG6uQZpEVvYSSsxMXRKOewtDk9RaTKXRlA==", + "dev": true + } + } + }, + "@types/anymatch": { + "version": "1.3.1", + "resolved": "https://registry.npmjs.org/@types/anymatch/-/anymatch-1.3.1.tgz", + "integrity": "sha512-/+CRPXpBDpo2RK9C68N3b2cOvO0Cf5B9aPijHsoDQTHivnGSObdOF2BRQOYjojWTDy6nQvMjmqRXIxH55VjxxA==", + "dev": true + }, + "@types/babel__core": { + "version": "7.1.12", + "resolved": "https://registry.npmjs.org/@types/babel__core/-/babel__core-7.1.12.tgz", + "integrity": "sha512-wMTHiiTiBAAPebqaPiPDLFA4LYPKr6Ph0Xq/6rq1Ur3v66HXyG+clfR9CNETkD7MQS8ZHvpQOtA53DLws5WAEQ==", + "dev": true, + "requires": { + "@babel/parser": "^7.1.0", + "@babel/types": "^7.0.0", + "@types/babel__generator": "*", + "@types/babel__template": "*", + "@types/babel__traverse": "*" + } + }, + "@types/babel__generator": { + "version": "7.6.2", + "resolved": "https://registry.npmjs.org/@types/babel__generator/-/babel__generator-7.6.2.tgz", + "integrity": "sha512-MdSJnBjl+bdwkLskZ3NGFp9YcXGx5ggLpQQPqtgakVhsWK0hTtNYhjpZLlWQTviGTvF8at+Bvli3jV7faPdgeQ==", + "dev": true, + "requires": { + "@babel/types": "^7.0.0" + } + }, + "@types/babel__template": { + "version": "7.4.0", + "resolved": "https://registry.npmjs.org/@types/babel__template/-/babel__template-7.4.0.tgz", + "integrity": "sha512-NTPErx4/FiPCGScH7foPyr+/1Dkzkni+rHiYHHoTjvwou7AQzJkNeD60A9CXRy+ZEN2B1bggmkTMCDb+Mv5k+A==", + "dev": true, + "requires": { + "@babel/parser": "^7.1.0", + "@babel/types": "^7.0.0" + } + }, + "@types/babel__traverse": { + "version": "7.0.15", + "resolved": "https://registry.npmjs.org/@types/babel__traverse/-/babel__traverse-7.0.15.tgz", + "integrity": "sha512-Pzh9O3sTK8V6I1olsXpCfj2k/ygO2q1X0vhhnDrEQyYLHZesWz+zMZMVcwXLCYf0U36EtmyYaFGPfXlTtDHe3A==", + "dev": true, + "requires": { + "@babel/types": "^7.3.0" + } + }, + "@types/concat-stream": { + "version": "1.6.0", + "resolved": "https://registry.npmjs.org/@types/concat-stream/-/concat-stream-1.6.0.tgz", + "integrity": "sha1-OU2+C7X+5Gs42JZzXoto7yOQ0A0=", + "dev": true, + "requires": { + "@types/node": "*" + } + }, + "@types/eslint-visitor-keys": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/@types/eslint-visitor-keys/-/eslint-visitor-keys-1.0.0.tgz", + "integrity": "sha512-OCutwjDZ4aFS6PB1UZ988C4YgwlBHJd6wCeQqaLdmadZ/7e+w79+hbMUFC1QXDNCmdyoRfAFdm0RypzwR+Qpag==", + "dev": true + }, + "@types/estree": { + "version": "0.0.39", + "resolved": "https://registry.npmjs.org/@types/estree/-/estree-0.0.39.tgz", + "integrity": "sha512-EYNwp3bU+98cpU4lAWYYL7Zz+2gryWH1qbdDTidVd6hkiR6weksdbMadyXKXNPEkQFhXM+hVO9ZygomHXp+AIw==", + "dev": true + }, + "@types/form-data": { + "version": "0.0.33", + "resolved": "https://registry.npmjs.org/@types/form-data/-/form-data-0.0.33.tgz", + "integrity": "sha1-yayFsqX9GENbjIXZ7LUObWyJP/g=", + "dev": true, + "requires": { + "@types/node": "*" + } + }, + "@types/glob": { + "version": "7.1.3", + "resolved": "https://registry.npmjs.org/@types/glob/-/glob-7.1.3.tgz", + "integrity": "sha512-SEYeGAIQIQX8NN6LDKprLjbrd5dARM5EXsd8GI/A5l0apYI1fGMWgPHSe4ZKL4eozlAyI+doUE9XbYS4xCkQ1w==", + "dev": true, + "requires": { + "@types/minimatch": "*", + "@types/node": "*" + } + }, + "@types/graceful-fs": { + "version": "4.1.4", + "resolved": "https://registry.npmjs.org/@types/graceful-fs/-/graceful-fs-4.1.4.tgz", + "integrity": "sha512-mWA/4zFQhfvOA8zWkXobwJvBD7vzcxgrOQ0J5CH1votGqdq9m7+FwtGaqyCZqC3NyyBkc9z4m+iry4LlqcMWJg==", + "dev": true, + "requires": { + "@types/node": "*" + } + }, + "@types/html-minifier-terser": { + "version": "5.1.1", + "resolved": "https://registry.npmjs.org/@types/html-minifier-terser/-/html-minifier-terser-5.1.1.tgz", + "integrity": "sha512-giAlZwstKbmvMk1OO7WXSj4OZ0keXAcl2TQq4LWHiiPH2ByaH7WeUzng+Qej8UPxxv+8lRTuouo0iaNDBuzIBA==", + "dev": true + }, + "@types/istanbul-lib-coverage": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/@types/istanbul-lib-coverage/-/istanbul-lib-coverage-2.0.3.tgz", + "integrity": "sha512-sz7iLqvVUg1gIedBOvlkxPlc8/uVzyS5OwGz1cKjXzkl3FpL3al0crU8YGU1WoHkxn0Wxbw5tyi6hvzJKNzFsw==", + "dev": true + }, + "@types/istanbul-lib-report": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/@types/istanbul-lib-report/-/istanbul-lib-report-3.0.0.tgz", + "integrity": "sha512-plGgXAPfVKFoYfa9NpYDAkseG+g6Jr294RqeqcqDixSbU34MZVJRi/P+7Y8GDpzkEwLaGZZOpKIEmeVZNtKsrg==", + "dev": true, + "requires": { + "@types/istanbul-lib-coverage": "*" + } + }, + "@types/istanbul-reports": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/@types/istanbul-reports/-/istanbul-reports-1.1.2.tgz", + "integrity": "sha512-P/W9yOX/3oPZSpaYOCQzGqgCQRXn0FFO/V8bWrCQs+wLmvVVxk6CRBXALEvNs9OHIatlnlFokfhuDo2ug01ciw==", + "dev": true, + "requires": { + "@types/istanbul-lib-coverage": "*", + "@types/istanbul-lib-report": "*" + } + }, + "@types/jest": { + "version": "26.0.3", + "resolved": "https://registry.npmjs.org/@types/jest/-/jest-26.0.3.tgz", + "integrity": "sha512-v89ga1clpVL/Y1+YI0eIu1VMW+KU7Xl8PhylVtDKVWaSUHBHYPLXMQGBdrpHewaKoTvlXkksbYqPgz8b4cmRZg==", + "dev": true, + "requires": { + "jest-diff": "^25.2.1", + "pretty-format": "^25.2.1" + } + }, + "@types/json-schema": { + "version": "7.0.6", + "resolved": "https://registry.npmjs.org/@types/json-schema/-/json-schema-7.0.6.tgz", + "integrity": "sha512-3c+yGKvVP5Y9TYBEibGNR+kLtijnj7mYrXRg+WpFb2X9xm04g/DXYkfg4hmzJQosc9snFNUPkbYIhu+KAm6jJw==", + "dev": true + }, + "@types/minimatch": { + "version": "3.0.3", + "resolved": "https://registry.npmjs.org/@types/minimatch/-/minimatch-3.0.3.tgz", + "integrity": "sha512-tHq6qdbT9U1IRSGf14CL0pUlULksvY9OZ+5eEgl1N7t+OA3tGvNpxJCzuKQlsNgCVwbAs670L1vcVQi8j9HjnA==", + "dev": true + }, + "@types/minimist": { + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/@types/minimist/-/minimist-1.2.1.tgz", + "integrity": "sha512-fZQQafSREFyuZcdWFAExYjBiCL7AUCdgsk80iO0q4yihYYdcIiH28CcuPTGFgLOCC8RlW49GSQxdHwZP+I7CNg==", + "dev": true + }, + "@types/node": { + "version": "13.13.4", + "resolved": "https://registry.npmjs.org/@types/node/-/node-13.13.4.tgz", + "integrity": "sha512-x26ur3dSXgv5AwKS0lNfbjpCakGIduWU1DU91Zz58ONRWrIKGunmZBNv4P7N+e27sJkiGDsw/3fT4AtsqQBrBA==", + "dev": true + }, + "@types/normalize-package-data": { + "version": "2.4.0", + "resolved": "https://registry.npmjs.org/@types/normalize-package-data/-/normalize-package-data-2.4.0.tgz", + "integrity": "sha512-f5j5b/Gf71L+dbqxIpQ4Z2WlmI/mPJ0fOkGGmFgtb6sAu97EPczzbS3/tJKxmcYDj55OX6ssqwDAWOHIYDRDGA==", + "dev": true + }, + "@types/parse-json": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/@types/parse-json/-/parse-json-4.0.0.tgz", + "integrity": "sha512-//oorEZjL6sbPcKUaCdIGlIUeH26mgzimjBB77G6XRgnDl/L5wOnpyBGRe/Mmf5CVW3PwEBE1NjiMZ/ssFh4wA==", + "dev": true + }, + "@types/prettier": { + "version": "2.1.5", + "resolved": "https://registry.npmjs.org/@types/prettier/-/prettier-2.1.5.tgz", + "integrity": "sha512-UEyp8LwZ4Dg30kVU2Q3amHHyTn1jEdhCIE59ANed76GaT1Vp76DD3ZWSAxgCrw6wJ0TqeoBpqmfUHiUDPs//HQ==", + "dev": true + }, + "@types/q": { + "version": "1.5.4", + "resolved": "https://registry.npmjs.org/@types/q/-/q-1.5.4.tgz", + "integrity": "sha512-1HcDas8SEj4z1Wc696tH56G8OlRaH/sqZOynNNB+HF0WOeXPaxTtbYzJY2oEfiUxjSKjhCKr+MvR7dCHcEelug==", + "dev": true + }, + "@types/qs": { + "version": "6.9.5", + "resolved": "https://registry.npmjs.org/@types/qs/-/qs-6.9.5.tgz", + "integrity": "sha512-/JHkVHtx/REVG0VVToGRGH2+23hsYLHdyG+GrvoUGlGAd0ErauXDyvHtRI/7H7mzLm+tBCKA7pfcpkQ1lf58iQ==", + "dev": true + }, + "@types/resolve": { + "version": "0.0.8", + "resolved": "https://registry.npmjs.org/@types/resolve/-/resolve-0.0.8.tgz", + "integrity": "sha512-auApPaJf3NPfe18hSoJkp8EbZzer2ISk7o8mCC3M9he/a04+gbMF97NkpD2S8riMGvm4BMRI59/SZQSaLTKpsQ==", + "dev": true, + "requires": { + "@types/node": "*" + } + }, + "@types/source-list-map": { + "version": "0.1.2", + "resolved": "https://registry.npmjs.org/@types/source-list-map/-/source-list-map-0.1.2.tgz", + "integrity": "sha512-K5K+yml8LTo9bWJI/rECfIPrGgxdpeNbj+d53lwN4QjW1MCwlkhUms+gtdzigTeUyBr09+u8BwOIY3MXvHdcsA==", + "dev": true + }, + "@types/stack-utils": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/@types/stack-utils/-/stack-utils-2.0.0.tgz", + "integrity": "sha512-RJJrrySY7A8havqpGObOB4W92QXKJo63/jFLLgpvOtsGUqbQZ9Sbgl35KMm1DjC6j7AvmmU2bIno+3IyEaemaw==", + "dev": true + }, + "@types/tapable": { + "version": "1.0.6", + "resolved": "https://registry.npmjs.org/@types/tapable/-/tapable-1.0.6.tgz", + "integrity": "sha512-W+bw9ds02rAQaMvaLYxAbJ6cvguW/iJXNT6lTssS1ps6QdrMKttqEAMEG/b5CR8TZl3/L7/lH0ZV5nNR1LXikA==", + "dev": true + }, + "@types/uglify-js": { + "version": "3.11.1", + "resolved": "https://registry.npmjs.org/@types/uglify-js/-/uglify-js-3.11.1.tgz", + "integrity": "sha512-7npvPKV+jINLu1SpSYVWG8KvyJBhBa8tmzMMdDoVc2pWUYHN8KIXlPJhjJ4LT97c4dXJA2SHL/q6ADbDriZN+Q==", + "dev": true, + "requires": { + "source-map": "^0.6.1" + }, + "dependencies": { + "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 + } + } + }, + "@types/webpack": { + "version": "4.41.25", + "resolved": "https://registry.npmjs.org/@types/webpack/-/webpack-4.41.25.tgz", + "integrity": "sha512-cr6kZ+4m9lp86ytQc1jPOJXgINQyz3kLLunZ57jznW+WIAL0JqZbGubQk4GlD42MuQL5JGOABrxdpqqWeovlVQ==", + "dev": true, + "requires": { + "@types/anymatch": "*", + "@types/node": "*", + "@types/tapable": "*", + "@types/uglify-js": "*", + "@types/webpack-sources": "*", + "source-map": "^0.6.0" + }, + "dependencies": { + "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 + } + } + }, + "@types/webpack-sources": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/@types/webpack-sources/-/webpack-sources-2.0.0.tgz", + "integrity": "sha512-a5kPx98CNFRKQ+wqawroFunvFqv7GHm/3KOI52NY9xWADgc8smu4R6prt4EU/M4QfVjvgBkMqU4fBhw3QfMVkg==", + "dev": true, + "requires": { + "@types/node": "*", + "@types/source-list-map": "*", + "source-map": "^0.7.3" + }, + "dependencies": { + "source-map": { + "version": "0.7.3", + "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.7.3.tgz", + "integrity": "sha512-CkCj6giN3S+n9qrYiBTX5gystlENnRW5jZeNLHpe6aue+SrHcG5VYwujhW9s4dY31mEGsxBDrHR6oI69fTXsaQ==", + "dev": true + } + } + }, + "@types/yargs": { + "version": "15.0.10", + "resolved": "https://registry.npmjs.org/@types/yargs/-/yargs-15.0.10.tgz", + "integrity": "sha512-z8PNtlhrj7eJNLmrAivM7rjBESG6JwC5xP3RVk12i/8HVP7Xnx/sEmERnRImyEuUaJfO942X0qMOYsoupaJbZQ==", + "dev": true, + "requires": { + "@types/yargs-parser": "*" + } + }, + "@types/yargs-parser": { + "version": "15.0.0", + "resolved": "https://registry.npmjs.org/@types/yargs-parser/-/yargs-parser-15.0.0.tgz", + "integrity": "sha512-FA/BWv8t8ZWJ+gEOnLLd8ygxH/2UFbAvgEonyfN6yWGLKc7zVjbpl2Y4CTjid9h2RfgPP6SEt6uHwEOply00yw==", + "dev": true + }, + "@typescript-eslint/eslint-plugin": { + "version": "2.30.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/eslint-plugin/-/eslint-plugin-2.30.0.tgz", + "integrity": "sha512-PGejii0qIZ9Q40RB2jIHyUpRWs1GJuHP1pkoCiaeicfwO9z7Fx03NQzupuyzAmv+q9/gFNHu7lo1ByMXe8PNyg==", + "dev": true, + "requires": { + "@typescript-eslint/experimental-utils": "2.30.0", + "functional-red-black-tree": "^1.0.1", + "regexpp": "^3.0.0", + "tsutils": "^3.17.1" + } + }, + "@typescript-eslint/eslint-plugin-tslint": { + "version": "2.30.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/eslint-plugin-tslint/-/eslint-plugin-tslint-2.30.0.tgz", + "integrity": "sha512-phARGRY1SyAkG9uVhF7o0yjK1eqmGYCwM7JpNkOo/50d68ZG0V/P9VyYMSKAj+IbRlZ/k2rKFibKZvfLJPcFGw==", + "dev": true, + "requires": { + "@typescript-eslint/experimental-utils": "2.30.0", + "lodash": "^4.17.15" + } + }, + "@typescript-eslint/experimental-utils": { + "version": "2.30.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/experimental-utils/-/experimental-utils-2.30.0.tgz", + "integrity": "sha512-L3/tS9t+hAHksy8xuorhOzhdefN0ERPDWmR9CclsIGOUqGKy6tqc/P+SoXeJRye5gazkuPO0cK9MQRnolykzkA==", + "dev": true, + "requires": { + "@types/json-schema": "^7.0.3", + "@typescript-eslint/typescript-estree": "2.30.0", + "eslint-scope": "^5.0.0", + "eslint-utils": "^2.0.0" + } + }, + "@typescript-eslint/parser": { + "version": "2.30.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/parser/-/parser-2.30.0.tgz", + "integrity": "sha512-9kDOxzp0K85UnpmPJqUzdWaCNorYYgk1yZmf4IKzpeTlSAclnFsrLjfwD9mQExctLoLoGAUXq1co+fbr+3HeFw==", + "dev": true, + "requires": { + "@types/eslint-visitor-keys": "^1.0.0", + "@typescript-eslint/experimental-utils": "2.30.0", + "@typescript-eslint/typescript-estree": "2.30.0", + "eslint-visitor-keys": "^1.1.0" + } + }, + "@typescript-eslint/typescript-estree": { + "version": "2.30.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/typescript-estree/-/typescript-estree-2.30.0.tgz", + "integrity": "sha512-nI5WOechrA0qAhnr+DzqwmqHsx7Ulr/+0H7bWCcClDhhWkSyZR5BmTvnBEyONwJCTWHfc5PAQExX24VD26IAVw==", + "dev": true, + "requires": { + "debug": "^4.1.1", + "eslint-visitor-keys": "^1.1.0", + "glob": "^7.1.6", + "is-glob": "^4.0.1", + "lodash": "^4.17.15", + "semver": "^6.3.0", + "tsutils": "^3.17.1" + }, + "dependencies": { + "glob": { + "version": "7.1.6", + "resolved": "https://registry.npmjs.org/glob/-/glob-7.1.6.tgz", + "integrity": "sha512-LwaxwyZ72Lk7vZINtNNrywX0ZuLyStrdDtabefZKAY5ZGJhVtgdznluResxNmPitE0SAO+O26sWTHeKSI2wMBA==", + "dev": true, + "requires": { + "fs.realpath": "^1.0.0", + "inflight": "^1.0.4", + "inherits": "2", + "minimatch": "^3.0.4", + "once": "^1.3.0", + "path-is-absolute": "^1.0.0" + } + }, + "semver": { + "version": "6.3.0", + "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.0.tgz", + "integrity": "sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw==", + "dev": true + } + } + }, + "@webassemblyjs/ast": { + "version": "1.9.0", + "resolved": "https://registry.npmjs.org/@webassemblyjs/ast/-/ast-1.9.0.tgz", + "integrity": "sha512-C6wW5L+b7ogSDVqymbkkvuW9kruN//YisMED04xzeBBqjHa2FYnmvOlS6Xj68xWQRgWvI9cIglsjFowH/RJyEA==", + "dev": true, + "requires": { + "@webassemblyjs/helper-module-context": "1.9.0", + "@webassemblyjs/helper-wasm-bytecode": "1.9.0", + "@webassemblyjs/wast-parser": "1.9.0" + } + }, + "@webassemblyjs/floating-point-hex-parser": { + "version": "1.9.0", + "resolved": "https://registry.npmjs.org/@webassemblyjs/floating-point-hex-parser/-/floating-point-hex-parser-1.9.0.tgz", + "integrity": "sha512-TG5qcFsS8QB4g4MhrxK5TqfdNe7Ey/7YL/xN+36rRjl/BlGE/NcBvJcqsRgCP6Z92mRE+7N50pRIi8SmKUbcQA==", + "dev": true + }, + "@webassemblyjs/helper-api-error": { + "version": "1.9.0", + "resolved": "https://registry.npmjs.org/@webassemblyjs/helper-api-error/-/helper-api-error-1.9.0.tgz", + "integrity": "sha512-NcMLjoFMXpsASZFxJ5h2HZRcEhDkvnNFOAKneP5RbKRzaWJN36NC4jqQHKwStIhGXu5mUWlUUk7ygdtrO8lbmw==", + "dev": true + }, + "@webassemblyjs/helper-buffer": { + "version": "1.9.0", + "resolved": "https://registry.npmjs.org/@webassemblyjs/helper-buffer/-/helper-buffer-1.9.0.tgz", + "integrity": "sha512-qZol43oqhq6yBPx7YM3m9Bv7WMV9Eevj6kMi6InKOuZxhw+q9hOkvq5e/PpKSiLfyetpaBnogSbNCfBwyB00CA==", + "dev": true + }, + "@webassemblyjs/helper-code-frame": { + "version": "1.9.0", + "resolved": "https://registry.npmjs.org/@webassemblyjs/helper-code-frame/-/helper-code-frame-1.9.0.tgz", + "integrity": "sha512-ERCYdJBkD9Vu4vtjUYe8LZruWuNIToYq/ME22igL+2vj2dQ2OOujIZr3MEFvfEaqKoVqpsFKAGsRdBSBjrIvZA==", + "dev": true, + "requires": { + "@webassemblyjs/wast-printer": "1.9.0" + } + }, + "@webassemblyjs/helper-fsm": { + "version": "1.9.0", + "resolved": "https://registry.npmjs.org/@webassemblyjs/helper-fsm/-/helper-fsm-1.9.0.tgz", + "integrity": "sha512-OPRowhGbshCb5PxJ8LocpdX9Kl0uB4XsAjl6jH/dWKlk/mzsANvhwbiULsaiqT5GZGT9qinTICdj6PLuM5gslw==", + "dev": true + }, + "@webassemblyjs/helper-module-context": { + "version": "1.9.0", + "resolved": "https://registry.npmjs.org/@webassemblyjs/helper-module-context/-/helper-module-context-1.9.0.tgz", + "integrity": "sha512-MJCW8iGC08tMk2enck1aPW+BE5Cw8/7ph/VGZxwyvGbJwjktKkDK7vy7gAmMDx88D7mhDTCNKAW5tED+gZ0W8g==", + "dev": true, + "requires": { + "@webassemblyjs/ast": "1.9.0" + } + }, + "@webassemblyjs/helper-wasm-bytecode": { + "version": "1.9.0", + "resolved": "https://registry.npmjs.org/@webassemblyjs/helper-wasm-bytecode/-/helper-wasm-bytecode-1.9.0.tgz", + "integrity": "sha512-R7FStIzyNcd7xKxCZH5lE0Bqy+hGTwS3LJjuv1ZVxd9O7eHCedSdrId/hMOd20I+v8wDXEn+bjfKDLzTepoaUw==", + "dev": true + }, + "@webassemblyjs/helper-wasm-section": { + "version": "1.9.0", + "resolved": "https://registry.npmjs.org/@webassemblyjs/helper-wasm-section/-/helper-wasm-section-1.9.0.tgz", + "integrity": "sha512-XnMB8l3ek4tvrKUUku+IVaXNHz2YsJyOOmz+MMkZvh8h1uSJpSen6vYnw3IoQ7WwEuAhL8Efjms1ZWjqh2agvw==", + "dev": true, + "requires": { + "@webassemblyjs/ast": "1.9.0", + "@webassemblyjs/helper-buffer": "1.9.0", + "@webassemblyjs/helper-wasm-bytecode": "1.9.0", + "@webassemblyjs/wasm-gen": "1.9.0" + } + }, + "@webassemblyjs/ieee754": { + "version": "1.9.0", + "resolved": "https://registry.npmjs.org/@webassemblyjs/ieee754/-/ieee754-1.9.0.tgz", + "integrity": "sha512-dcX8JuYU/gvymzIHc9DgxTzUUTLexWwt8uCTWP3otys596io0L5aW02Gb1RjYpx2+0Jus1h4ZFqjla7umFniTg==", + "dev": true, + "requires": { + "@xtuc/ieee754": "^1.2.0" + } + }, + "@webassemblyjs/leb128": { + "version": "1.9.0", + "resolved": "https://registry.npmjs.org/@webassemblyjs/leb128/-/leb128-1.9.0.tgz", + "integrity": "sha512-ENVzM5VwV1ojs9jam6vPys97B/S65YQtv/aanqnU7D8aSoHFX8GyhGg0CMfyKNIHBuAVjy3tlzd5QMMINa7wpw==", + "dev": true, + "requires": { + "@xtuc/long": "4.2.2" + } + }, + "@webassemblyjs/utf8": { + "version": "1.9.0", + "resolved": "https://registry.npmjs.org/@webassemblyjs/utf8/-/utf8-1.9.0.tgz", + "integrity": "sha512-GZbQlWtopBTP0u7cHrEx+73yZKrQoBMpwkGEIqlacljhXCkVM1kMQge/Mf+csMJAjEdSwhOyLAS0AoR3AG5P8w==", + "dev": true + }, + "@webassemblyjs/wasm-edit": { + "version": "1.9.0", + "resolved": "https://registry.npmjs.org/@webassemblyjs/wasm-edit/-/wasm-edit-1.9.0.tgz", + "integrity": "sha512-FgHzBm80uwz5M8WKnMTn6j/sVbqilPdQXTWraSjBwFXSYGirpkSWE2R9Qvz9tNiTKQvoKILpCuTjBKzOIm0nxw==", + "dev": true, + "requires": { + "@webassemblyjs/ast": "1.9.0", + "@webassemblyjs/helper-buffer": "1.9.0", + "@webassemblyjs/helper-wasm-bytecode": "1.9.0", + "@webassemblyjs/helper-wasm-section": "1.9.0", + "@webassemblyjs/wasm-gen": "1.9.0", + "@webassemblyjs/wasm-opt": "1.9.0", + "@webassemblyjs/wasm-parser": "1.9.0", + "@webassemblyjs/wast-printer": "1.9.0" + } + }, + "@webassemblyjs/wasm-gen": { + "version": "1.9.0", + "resolved": "https://registry.npmjs.org/@webassemblyjs/wasm-gen/-/wasm-gen-1.9.0.tgz", + "integrity": "sha512-cPE3o44YzOOHvlsb4+E9qSqjc9Qf9Na1OO/BHFy4OI91XDE14MjFN4lTMezzaIWdPqHnsTodGGNP+iRSYfGkjA==", + "dev": true, + "requires": { + "@webassemblyjs/ast": "1.9.0", + "@webassemblyjs/helper-wasm-bytecode": "1.9.0", + "@webassemblyjs/ieee754": "1.9.0", + "@webassemblyjs/leb128": "1.9.0", + "@webassemblyjs/utf8": "1.9.0" + } + }, + "@webassemblyjs/wasm-opt": { + "version": "1.9.0", + "resolved": "https://registry.npmjs.org/@webassemblyjs/wasm-opt/-/wasm-opt-1.9.0.tgz", + "integrity": "sha512-Qkjgm6Anhm+OMbIL0iokO7meajkzQD71ioelnfPEj6r4eOFuqm4YC3VBPqXjFyyNwowzbMD+hizmprP/Fwkl2A==", + "dev": true, + "requires": { + "@webassemblyjs/ast": "1.9.0", + "@webassemblyjs/helper-buffer": "1.9.0", + "@webassemblyjs/wasm-gen": "1.9.0", + "@webassemblyjs/wasm-parser": "1.9.0" + } + }, + "@webassemblyjs/wasm-parser": { + "version": "1.9.0", + "resolved": "https://registry.npmjs.org/@webassemblyjs/wasm-parser/-/wasm-parser-1.9.0.tgz", + "integrity": "sha512-9+wkMowR2AmdSWQzsPEjFU7njh8HTO5MqO8vjwEHuM+AMHioNqSBONRdr0NQQ3dVQrzp0s8lTcYqzUdb7YgELA==", + "dev": true, + "requires": { + "@webassemblyjs/ast": "1.9.0", + "@webassemblyjs/helper-api-error": "1.9.0", + "@webassemblyjs/helper-wasm-bytecode": "1.9.0", + "@webassemblyjs/ieee754": "1.9.0", + "@webassemblyjs/leb128": "1.9.0", + "@webassemblyjs/utf8": "1.9.0" + } + }, + "@webassemblyjs/wast-parser": { + "version": "1.9.0", + "resolved": "https://registry.npmjs.org/@webassemblyjs/wast-parser/-/wast-parser-1.9.0.tgz", + "integrity": "sha512-qsqSAP3QQ3LyZjNC/0jBJ/ToSxfYJ8kYyuiGvtn/8MK89VrNEfwj7BPQzJVHi0jGTRK2dGdJ5PRqhtjzoww+bw==", + "dev": true, + "requires": { + "@webassemblyjs/ast": "1.9.0", + "@webassemblyjs/floating-point-hex-parser": "1.9.0", + "@webassemblyjs/helper-api-error": "1.9.0", + "@webassemblyjs/helper-code-frame": "1.9.0", + "@webassemblyjs/helper-fsm": "1.9.0", + "@xtuc/long": "4.2.2" + } + }, + "@webassemblyjs/wast-printer": { + "version": "1.9.0", + "resolved": "https://registry.npmjs.org/@webassemblyjs/wast-printer/-/wast-printer-1.9.0.tgz", + "integrity": "sha512-2J0nE95rHXHyQ24cWjMKJ1tqB/ds8z/cyeOZxJhcb+rW+SQASVjuznUSmdz5GpVJTzU8JkhYut0D3siFDD6wsA==", + "dev": true, + "requires": { + "@webassemblyjs/ast": "1.9.0", + "@webassemblyjs/wast-parser": "1.9.0", + "@xtuc/long": "4.2.2" + } + }, + "@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 + }, + "@yarnpkg/lockfile": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/@yarnpkg/lockfile/-/lockfile-1.1.0.tgz", + "integrity": "sha512-GpSwvyXOcOOlV70vbnzjj4fW5xW/FdUF6nQEt1ENy7m4ZCczi1+/buVUPAqmGfqznsORNFzUMjctTIp8a9tuCQ==", + "dev": true + }, + "JSONStream": { + "version": "1.3.5", + "resolved": "https://registry.npmjs.org/JSONStream/-/JSONStream-1.3.5.tgz", + "integrity": "sha512-E+iruNOY8VV9s4JEbe1aNEm6MiszPRr/UfcHMz0TQh1BXSxHK+ASV1R6W4HpjBhSeS+54PIsAMCBmwD06LLsqQ==", + "dev": true, + "requires": { + "jsonparse": "^1.2.0", + "through": ">=2.2.7 <3" + } + }, + "abab": { + "version": "2.0.5", + "resolved": "https://registry.npmjs.org/abab/-/abab-2.0.5.tgz", + "integrity": "sha512-9IK9EadsbHo6jLWIpxpR6pL0sazTXV6+SQv25ZB+F7Bj9mJNaOc4nCRabwd5M/JwmUa8idz6Eci6eKfJryPs6Q==", + "dev": true + }, + "accepts": { + "version": "1.3.7", + "resolved": "https://registry.npmjs.org/accepts/-/accepts-1.3.7.tgz", + "integrity": "sha512-Il80Qs2WjYlJIBNzNkK6KYqlVMTbZLXgHx2oT0pU/fjRHyEp+PEfEPY0R3WCwAGVOtauxh1hOxNgIf5bv7dQpA==", + "dev": true, + "requires": { + "mime-types": "~2.1.24", + "negotiator": "0.6.2" + } + }, + "acorn": { + "version": "7.4.1", + "resolved": "https://registry.npmjs.org/acorn/-/acorn-7.4.1.tgz", + "integrity": "sha512-nQyp0o1/mNdbTO1PO6kHkwSrmgZ0MT/jCCpNiwbUjGoRN4dlBhqJtoQuCnEOKzgTVwg0ZWiCoQy6SxMebQVh8A==", + "dev": true + }, + "acorn-globals": { + "version": "6.0.0", + "resolved": "https://registry.npmjs.org/acorn-globals/-/acorn-globals-6.0.0.tgz", + "integrity": "sha512-ZQl7LOWaF5ePqqcX4hLuv/bLXYQNfNWw2c0/yX/TsPRKamzHcTGQnlCjHT3TsmkOUVEPS3crCxiPfdzE/Trlhg==", + "dev": true, + "requires": { + "acorn": "^7.1.1", + "acorn-walk": "^7.1.1" + } + }, + "acorn-jsx": { + "version": "5.3.1", + "resolved": "https://registry.npmjs.org/acorn-jsx/-/acorn-jsx-5.3.1.tgz", + "integrity": "sha512-K0Ptm/47OKfQRpNQ2J/oIN/3QYiK6FwW+eJbILhsdxh2WTLdl+30o8aGdTbm5JbffpFFAg/g+zi1E+jvJha5ng==", + "dev": true + }, + "acorn-walk": { + "version": "7.2.0", + "resolved": "https://registry.npmjs.org/acorn-walk/-/acorn-walk-7.2.0.tgz", + "integrity": "sha512-OPdCF6GsMIP+Az+aWfAAOEt2/+iVDKE7oy6lJ098aoe59oAmK76qV6Gw60SbZ8jHuG2wH058GF4pLFbYamYrVA==", + "dev": true + }, + "after": { + "version": "0.8.2", + "resolved": "https://registry.npmjs.org/after/-/after-0.8.2.tgz", + "integrity": "sha1-/ts5T58OAqqXaOcCvaI7UF+ufh8=", + "dev": true + }, + "agent-base": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/agent-base/-/agent-base-4.3.0.tgz", + "integrity": "sha512-salcGninV0nPrwpGNn4VTXBb1SOuXQBiqbrNXoeizJsHrsL6ERFM2Ne3JUSBWRE6aeNJI2ROP/WEEIDUiDe3cg==", + "dev": true, + "requires": { + "es6-promisify": "^5.0.0" + } + }, + "agentkeepalive": { + "version": "3.5.2", + "resolved": "https://registry.npmjs.org/agentkeepalive/-/agentkeepalive-3.5.2.tgz", + "integrity": "sha512-e0L/HNe6qkQ7H19kTlRRqUibEAwDK5AFk6y3PtMsuut2VAH6+Q4xZml1tNDJD7kSAyqmbG/K08K5WEJYtUrSlQ==", + "dev": true, + "requires": { + "humanize-ms": "^1.2.1" + } + }, + "aggregate-error": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/aggregate-error/-/aggregate-error-3.1.0.tgz", + "integrity": "sha512-4I7Td01quW/RpocfNayFdFVk1qSuoh0E7JrbRJ16nH01HhKFQ88INq9Sd+nd72zqRySlr9BmDA8xlEJ6vJMrYA==", + "dev": true, + "requires": { + "clean-stack": "^2.0.0", + "indent-string": "^4.0.0" + } + }, + "ajv": { + "version": "6.12.2", + "resolved": "https://registry.npmjs.org/ajv/-/ajv-6.12.2.tgz", + "integrity": "sha512-k+V+hzjm5q/Mr8ef/1Y9goCmlsK4I6Sm74teeyGvFk1XrOsbsKLjEdrvny42CZ+a8sXbk8KWpY/bDwS+FLL2UQ==", + "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-errors": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/ajv-errors/-/ajv-errors-1.0.1.tgz", + "integrity": "sha512-DCRfO/4nQ+89p/RK43i8Ezd41EqdGIU4ld7nGF8OQ14oc/we5rEntLCUa7+jrn3nn83BosfwZA0wb4pon2o8iQ==", + "dev": 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 + }, + "alphanum-sort": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/alphanum-sort/-/alphanum-sort-1.0.2.tgz", + "integrity": "sha1-l6ERlkmyEa0zaR2fn0hqjsn74KM=", + "dev": true + }, + "ansi": { + "version": "0.3.1", + "resolved": "https://registry.npmjs.org/ansi/-/ansi-0.3.1.tgz", + "integrity": "sha1-DELU+xcWDVqa8eSEus4cZpIsGyE=", + "dev": true + }, + "ansi-colors": { + "version": "4.1.1", + "resolved": "https://registry.npmjs.org/ansi-colors/-/ansi-colors-4.1.1.tgz", + "integrity": "sha512-JoX0apGbHaUJBNl6yF+p6JAFYZ666/hhCGKN5t9QFjbJQKUU/g8MNbFDbvfrgKXvI1QpZplPOnwIo99lX/AAmA==", + "dev": true + }, + "ansi-escapes": { + "version": "4.3.1", + "resolved": "https://registry.npmjs.org/ansi-escapes/-/ansi-escapes-4.3.1.tgz", + "integrity": "sha512-JWF7ocqNrp8u9oqpgV+wH5ftbt+cfvv+PTjOvKLT3AdYly/LmORARfEVT1iyjwN+4MqE5UmVKoAdIBqeoCHgLA==", + "dev": true, + "requires": { + "type-fest": "^0.11.0" + } + }, + "ansi-font": { + "version": "0.0.2", + "resolved": "https://registry.npmjs.org/ansi-font/-/ansi-font-0.0.2.tgz", + "integrity": "sha1-iQMBvVhBRi/TnAt3Ca/R9SUUMzE=", + "dev": true + }, + "ansi-html": { + "version": "0.0.7", + "resolved": "https://registry.npmjs.org/ansi-html/-/ansi-html-0.0.7.tgz", + "integrity": "sha1-gTWEAhliqenm/QOflA0S9WynhZ4=", + "dev": true + }, + "ansi-regex": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.0.tgz", + "integrity": "sha512-bY6fj56OUQ0hU1KjFNDQuJFezqKdrAyFdIevADiqrWHwSlbmBNMHp5ak2f40Pm8JTFyM2mqxkG6ngkHO11f/lg==" + }, + "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==", + "requires": { + "color-convert": "^1.9.0" + } + }, + "ansi-wrap": { + "version": "0.1.0", + "resolved": "https://registry.npmjs.org/ansi-wrap/-/ansi-wrap-0.1.0.tgz", + "integrity": "sha1-qCJQ3bABXponyoLoLqYDu/pF768=", + "dev": true + }, + "any-observable": { + "version": "0.3.0", + "resolved": "https://registry.npmjs.org/any-observable/-/any-observable-0.3.0.tgz", + "integrity": "sha512-/FQM1EDkTsf63Ub2C6O7GuYFDsSXUwsaZDurV0np41ocwq0jthUAYCmhBX9f+KwlaCgIuWyr/4WlUQUBfKfZog==", + "dev": true + }, + "anymatch": { + "version": "3.1.1", + "resolved": "https://registry.npmjs.org/anymatch/-/anymatch-3.1.1.tgz", + "integrity": "sha512-mM8522psRCqzV+6LhomX5wgp25YVibjh8Wj23I5RPkPppSVSjyKD2A2mBJmWGa+KN7f2D6LNh9jkBCeyLktzjg==", + "dev": true, + "requires": { + "normalize-path": "^3.0.0", + "picomatch": "^2.0.4" + } + }, + "app-root-path": { + "version": "2.2.1", + "resolved": "https://registry.npmjs.org/app-root-path/-/app-root-path-2.2.1.tgz", + "integrity": "sha512-91IFKeKk7FjfmezPKkwtaRvSpnUc4gDwPAjA1YZ9Gn0q0PPeW+vbeUsZuyDwjI7+QTHhcLen2v25fi/AmhvbJA==", + "dev": true + }, + "aproba": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/aproba/-/aproba-1.2.0.tgz", + "integrity": "sha512-Y9J6ZjXtoYh8RnXVCMOU/ttDmk1aBjunq9vO0ta5x85WDQiQfUF9sIPBITdbiiIVcBo03Hi3jMxigBtsddlXRw==", + "dev": true + }, + "are-we-there-yet": { + "version": "1.1.5", + "resolved": "https://registry.npmjs.org/are-we-there-yet/-/are-we-there-yet-1.1.5.tgz", + "integrity": "sha512-5hYdAkZlcG8tOLujVDTgCT+uPX0VnpAH28gWsLfzpXYm7wP6mp5Q/gYyR7YQ0cKVJcXJnl3j2kpBan13PtQf6w==", + "dev": true, + "requires": { + "delegates": "^1.0.0", + "readable-stream": "^2.0.6" + } + }, + "argparse": { + "version": "1.0.10", + "resolved": "https://registry.npmjs.org/argparse/-/argparse-1.0.10.tgz", + "integrity": "sha512-o5Roy6tNG4SL/FOkCAN6RzjiakZS25RLYFrcMttJqbdd8BWrnA+fGz57iN5Pb06pvBGvl5gQ0B48dJlslXvoTg==", + "dev": true, + "requires": { + "sprintf-js": "~1.0.2" + }, + "dependencies": { + "sprintf-js": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/sprintf-js/-/sprintf-js-1.0.3.tgz", + "integrity": "sha1-BOaSb2YolTVPPdAVIDYzuFcpfiw=", + "dev": true + } + } + }, + "aria-query": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/aria-query/-/aria-query-3.0.0.tgz", + "integrity": "sha1-ZbP8wcoRVajJrmTW7uKX8V1RM8w=", + "dev": true, + "requires": { + "ast-types-flow": "0.0.7", + "commander": "^2.11.0" + } + }, + "arr-diff": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/arr-diff/-/arr-diff-4.0.0.tgz", + "integrity": "sha1-1kYQdP6/7HHn4VI1dhoyml3HxSA=", + "dev": true + }, + "arr-flatten": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/arr-flatten/-/arr-flatten-1.1.0.tgz", + "integrity": "sha512-L3hKV5R/p5o81R7O02IGnwpDmkp6E982XhtbuwSe3O4qOtMMMtodicASA1Cny2U+aCXcNpml+m4dPsvsJ3jatg==", + "dev": true + }, + "arr-union": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/arr-union/-/arr-union-3.1.0.tgz", + "integrity": "sha1-45sJrqne+Gao8gbiiK9jkZuuOcQ=", + "dev": true + }, + "array-differ": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/array-differ/-/array-differ-3.0.0.tgz", + "integrity": "sha512-THtfYS6KtME/yIAhKjZ2ul7XI96lQGHRputJQHO80LAWQnuGP4iCIN8vdMRboGbIEYBwU33q8Tch1os2+X0kMg==", + "dev": true + }, + "array-flatten": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/array-flatten/-/array-flatten-1.1.1.tgz", + "integrity": "sha1-ml9pkFGx5wczKPKgCJaLZOopVdI=", + "dev": true + }, + "array-union": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/array-union/-/array-union-2.1.0.tgz", + "integrity": "sha512-HGyxoOTYUyCM6stUe6EJgnd4EoewAI7zMdfqO+kGjnlZmBDz/cR5pf8r/cR4Wq60sL/p0IkcjUEEPwS3GFrIyw==", + "dev": true + }, + "array-uniq": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/array-uniq/-/array-uniq-1.0.3.tgz", + "integrity": "sha1-r2rId6Jcx/dOBYiUdThY39sk/bY=", + "dev": true + }, + "array-unique": { + "version": "0.3.2", + "resolved": "https://registry.npmjs.org/array-unique/-/array-unique-0.3.2.tgz", + "integrity": "sha1-qJS3XUvE9s1nnvMkSp/Y9Gri1Cg=", + "dev": true + }, + "arraybuffer.slice": { + "version": "0.0.7", + "resolved": "https://registry.npmjs.org/arraybuffer.slice/-/arraybuffer.slice-0.0.7.tgz", + "integrity": "sha512-wGUIVQXuehL5TCqQun8OW81jGzAWycqzFF8lFp+GOM5BXLYj3bKNsYC4daB7n6XjCqxQA/qgTJ+8ANR3acjrog==", + "dev": true + }, + "arrify": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/arrify/-/arrify-2.0.1.tgz", + "integrity": "sha512-3duEwti880xqi4eAMN8AyR4a0ByT90zoYdLlevfrvU43vb0YZwZVfxOgxWrLXXXpyugL0hNZc9G6BiB5B3nUug==", + "dev": true + }, + "asap": { + "version": "2.0.6", + "resolved": "https://registry.npmjs.org/asap/-/asap-2.0.6.tgz", + "integrity": "sha1-5QNHYR1+aQlDIIu9r+vLwvuGbUY=", + "dev": true + }, + "asn1": { + "version": "0.2.4", + "resolved": "https://registry.npmjs.org/asn1/-/asn1-0.2.4.tgz", + "integrity": "sha512-jxwzQpLQjSmWXgwaCZE9Nz+glAG01yF1QnWgbhGwHI5A6FRIEY6IVqtHhIepHqI7/kyEyQEagBC5mBEFlIYvdg==", + "dev": true, + "requires": { + "safer-buffer": "~2.1.0" + } + }, + "asn1.js": { + "version": "5.4.1", + "resolved": "https://registry.npmjs.org/asn1.js/-/asn1.js-5.4.1.tgz", + "integrity": "sha512-+I//4cYPccV8LdmBLiX8CYvf9Sp3vQsrqu2QNXRcrbiWvcx/UdlFiqUJJzxRQxgsZmvhXhn4cSKeSmoFjVdupA==", + "dev": true, + "requires": { + "bn.js": "^4.0.0", + "inherits": "^2.0.1", + "minimalistic-assert": "^1.0.0", + "safer-buffer": "^2.1.0" + }, + "dependencies": { + "bn.js": { + "version": "4.11.9", + "resolved": "https://registry.npmjs.org/bn.js/-/bn.js-4.11.9.tgz", + "integrity": "sha512-E6QoYqCKZfgatHTdHzs1RRKP7ip4vvm+EyRUeE2RF0NblwVvb0p6jSVeNTOFxPn26QXN2o6SMfNxKp6kU8zQaw==", + "dev": true + } + } + }, + "assert": { + "version": "1.5.0", + "resolved": "https://registry.npmjs.org/assert/-/assert-1.5.0.tgz", + "integrity": "sha512-EDsgawzwoun2CZkCgtxJbv392v4nbk9XDD06zI+kQYoBM/3RBWLlEyJARDOmhAAosBjWACEkKL6S+lIZtcAubA==", + "dev": true, + "requires": { + "object-assign": "^4.1.1", + "util": "0.10.3" + }, + "dependencies": { + "inherits": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.1.tgz", + "integrity": "sha1-sX0I0ya0Qj5Wjv9xn5GwscvfafE=", + "dev": true + }, + "util": { + "version": "0.10.3", + "resolved": "https://registry.npmjs.org/util/-/util-0.10.3.tgz", + "integrity": "sha1-evsa/lCAUkZInj23/g7TeTNqwPk=", + "dev": true, + "requires": { + "inherits": "2.0.1" + } + } + } + }, + "assert-plus": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/assert-plus/-/assert-plus-1.0.0.tgz", + "integrity": "sha1-8S4PPF13sLHN2RRpQuTpbB5N1SU=", + "dev": true + }, + "assign-symbols": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/assign-symbols/-/assign-symbols-1.0.0.tgz", + "integrity": "sha1-WWZ/QfrdTyDMvCu5a41Pf3jsA2c=", + "dev": true + }, + "ast-types-flow": { + "version": "0.0.7", + "resolved": "https://registry.npmjs.org/ast-types-flow/-/ast-types-flow-0.0.7.tgz", + "integrity": "sha1-9wtzXGvKGlycItmCw+Oef+ujva0=", + "dev": true + }, + "astral-regex": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/astral-regex/-/astral-regex-1.0.0.tgz", + "integrity": "sha512-+Ryf6g3BKoRc7jfp7ad8tM4TtMiaWvbF/1/sQcZPkkS7ag3D5nMBCe2UfOTONtAkaG0tO0ij3C5Lwmf1EiyjHg==", + "dev": true + }, + "async": { + "version": "1.5.2", + "resolved": "https://registry.npmjs.org/async/-/async-1.5.2.tgz", + "integrity": "sha1-7GphrlZIDAw8skHJVhjiCJL5Zyo=", + "dev": true + }, + "async-each": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/async-each/-/async-each-1.0.3.tgz", + "integrity": "sha512-z/WhQ5FPySLdvREByI2vZiTWwCnF0moMJ1hK9YQwDTHKh6I7/uSckMetoRGb5UBZPC1z0jlw+n/XCgjeH7y1AQ==", + "dev": true + }, + "async-each-series": { + "version": "0.1.1", + "resolved": "https://registry.npmjs.org/async-each-series/-/async-each-series-0.1.1.tgz", + "integrity": "sha1-dhfBkXQB/Yykooqtzj266Yr+tDI=", + "dev": true + }, + "async-limiter": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/async-limiter/-/async-limiter-1.0.1.tgz", + "integrity": "sha512-csOlWGAcRFJaI6m+F2WKdnMKr4HhdhFVBk0H/QbJFMCr+uO2kwohwXQPxw/9OCxp05r5ghVBFSyioixx3gfkNQ==", + "dev": true + }, + "asynckit": { + "version": "0.4.0", + "resolved": "https://registry.npmjs.org/asynckit/-/asynckit-0.4.0.tgz", + "integrity": "sha1-x57Zf380y48robyXkLzDZkdLS3k=", + "dev": true + }, + "atob": { + "version": "2.1.2", + "resolved": "https://registry.npmjs.org/atob/-/atob-2.1.2.tgz", + "integrity": "sha512-Wm6ukoaOGJi/73p/cl2GvLjTI5JM1k/O14isD73YML8StrH/7/lRFgmg8nICZgD3bZZvjwCGxtMOD3wWNAu8cg==", + "dev": true + }, + "autoprefixer": { + "version": "9.8.4", + "resolved": "https://registry.npmjs.org/autoprefixer/-/autoprefixer-9.8.4.tgz", + "integrity": "sha512-84aYfXlpUe45lvmS+HoAWKCkirI/sw4JK0/bTeeqgHYco3dcsOn0NqdejISjptsYwNji/21dnkDri9PsYKk89A==", + "dev": true, + "requires": { + "browserslist": "^4.12.0", + "caniuse-lite": "^1.0.30001087", + "colorette": "^1.2.0", + "normalize-range": "^0.1.2", + "num2fraction": "^1.2.2", + "postcss": "^7.0.32", + "postcss-value-parser": "^4.1.0" + } + }, + "aws-sdk": { + "version": "2.706.0", + "resolved": "https://registry.npmjs.org/aws-sdk/-/aws-sdk-2.706.0.tgz", + "integrity": "sha512-7GT+yrB5Wb/zOReRdv/Pzkb2Qt+hz6B/8FGMVaoysX3NryHvQUdz7EQWi5yhg9CxOjKxdw5lFwYSs69YlSp1KA==", + "dev": true, + "requires": { + "buffer": "4.9.2", + "events": "1.1.1", + "ieee754": "1.1.13", + "jmespath": "0.15.0", + "querystring": "0.2.0", + "sax": "1.2.1", + "url": "0.10.3", + "uuid": "3.3.2", + "xml2js": "0.4.19" + }, + "dependencies": { + "uuid": { + "version": "3.3.2", + "resolved": "https://registry.npmjs.org/uuid/-/uuid-3.3.2.tgz", + "integrity": "sha512-yXJmeNaw3DnnKAOKJE51sL/ZaYfWJRl1pK9dr19YFCu0ObS231AB1/LbqTKRAQ5kw8A90rA6fr4riOUpTZvQZA==", + "dev": true + } + } + }, + "aws-sign2": { + "version": "0.7.0", + "resolved": "https://registry.npmjs.org/aws-sign2/-/aws-sign2-0.7.0.tgz", + "integrity": "sha1-tG6JCTSpWR8tL2+G1+ap8bP+dqg=", + "dev": true + }, + "aws4": { + "version": "1.11.0", + "resolved": "https://registry.npmjs.org/aws4/-/aws4-1.11.0.tgz", + "integrity": "sha512-xh1Rl34h6Fi1DC2WWKfxUTVqRsNnr6LsKz2+hfwDxQJWmrx8+c7ylaqBMcHfl1U1r2dsifOvKX3LQuLNZ+XSvA==", + "dev": true + }, + "axios": { + "version": "0.19.0", + "resolved": "https://registry.npmjs.org/axios/-/axios-0.19.0.tgz", + "integrity": "sha512-1uvKqKQta3KBxIz14F2v06AEHZ/dIoeKfbTRkK1E5oqjDnuEerLmYTgJB5AiQZHJcljpg1TuRzdjDR06qNk0DQ==", + "dev": true, + "requires": { + "follow-redirects": "1.5.10", + "is-buffer": "^2.0.2" + }, + "dependencies": { + "is-buffer": { + "version": "2.0.5", + "resolved": "https://registry.npmjs.org/is-buffer/-/is-buffer-2.0.5.tgz", + "integrity": "sha512-i2R6zNFDwgEHJyQUtJEk0XFi1i0dPFn/oqjK3/vPCcDeJvW5NQ83V8QbicfF1SupOaB0h8ntgBC2YiE7dfyctQ==", + "dev": true + } + } + }, + "axobject-query": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/axobject-query/-/axobject-query-2.0.2.tgz", + "integrity": "sha512-MCeek8ZH7hKyO1rWUbKNQBbl4l2eY0ntk7OGi+q0RlafrCnfPxC06WZA+uebCfmYp4mNU9jRBP1AhGyf8+W3ww==", + "dev": true, + "requires": { + "ast-types-flow": "0.0.7" + } + }, + "babel-extract-comments": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/babel-extract-comments/-/babel-extract-comments-1.0.0.tgz", + "integrity": "sha512-qWWzi4TlddohA91bFwgt6zO/J0X+io7Qp184Fw0m2JYRSTZnJbFR8+07KmzudHCZgOiKRCrjhylwv9Xd8gfhVQ==", + "dev": true, + "requires": { + "babylon": "^6.18.0" + } + }, + "babel-jest": { + "version": "26.6.3", + "resolved": "https://registry.npmjs.org/babel-jest/-/babel-jest-26.6.3.tgz", + "integrity": "sha512-pl4Q+GAVOHwvjrck6jKjvmGhnO3jHX/xuB9d27f+EJZ/6k+6nMuPjorrYp7s++bKKdANwzElBWnLWaObvTnaZA==", + "dev": true, + "requires": { + "@jest/transform": "^26.6.2", + "@jest/types": "^26.6.2", + "@types/babel__core": "^7.1.7", + "babel-plugin-istanbul": "^6.0.0", + "babel-preset-jest": "^26.6.2", + "chalk": "^4.0.0", + "graceful-fs": "^4.2.4", + "slash": "^3.0.0" + }, + "dependencies": { + "@jest/types": { + "version": "26.6.2", + "resolved": "https://registry.npmjs.org/@jest/types/-/types-26.6.2.tgz", + "integrity": "sha512-fC6QCp7Sc5sX6g8Tvbmj4XUTbyrik0akgRy03yjXbQaBWWNWGE7SGtJk98m0N8nzegD/7SggrUlivxo5ax4KWQ==", + "dev": true, + "requires": { + "@types/istanbul-lib-coverage": "^2.0.0", + "@types/istanbul-reports": "^3.0.0", + "@types/node": "*", + "@types/yargs": "^15.0.0", + "chalk": "^4.0.0" + } + }, + "@types/istanbul-reports": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/@types/istanbul-reports/-/istanbul-reports-3.0.0.tgz", + "integrity": "sha512-nwKNbvnwJ2/mndE9ItP/zc2TCzw6uuodnF4EHYWD+gCQDVBuRQL5UzbZD0/ezy1iKsFU2ZQiDqg4M9dN4+wZgA==", + "dev": true, + "requires": { + "@types/istanbul-lib-report": "*" + } + }, + "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.0", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.0.tgz", + "integrity": "sha512-qwx12AxXe2Q5xQ43Ac//I6v5aXTipYrSESdOgzrN+9XjgEpyjpKuvSGaN4qE93f7TQTlerQQ8S+EQ0EyDoVL1A==", + "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 + }, + "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" + } + } + } + }, + "babel-plugin-dynamic-import-node": { + "version": "2.3.3", + "resolved": "https://registry.npmjs.org/babel-plugin-dynamic-import-node/-/babel-plugin-dynamic-import-node-2.3.3.tgz", + "integrity": "sha512-jZVI+s9Zg3IqA/kdi0i6UDCybUI3aSBLnglhYbSSjKlV7yF1F/5LWv8MakQmvYpnbJDS6fcBL2KzHSxNCMtWSQ==", + "dev": true, + "requires": { + "object.assign": "^4.1.0" + } + }, + "babel-plugin-istanbul": { + "version": "6.0.0", + "resolved": "https://registry.npmjs.org/babel-plugin-istanbul/-/babel-plugin-istanbul-6.0.0.tgz", + "integrity": "sha512-AF55rZXpe7trmEylbaE1Gv54wn6rwU03aptvRoVIGP8YykoSxqdVLV1TfwflBCE/QtHmqtP8SWlTENqbK8GCSQ==", + "dev": true, + "requires": { + "@babel/helper-plugin-utils": "^7.0.0", + "@istanbuljs/load-nyc-config": "^1.0.0", + "@istanbuljs/schema": "^0.1.2", + "istanbul-lib-instrument": "^4.0.0", + "test-exclude": "^6.0.0" + } + }, + "babel-plugin-jest-hoist": { + "version": "26.6.2", + "resolved": "https://registry.npmjs.org/babel-plugin-jest-hoist/-/babel-plugin-jest-hoist-26.6.2.tgz", + "integrity": "sha512-PO9t0697lNTmcEHH69mdtYiOIkkOlj9fySqfO3K1eCcdISevLAE0xY59VLLUj0SoiPiTX/JU2CYFpILydUa5Lw==", + "dev": true, + "requires": { + "@babel/template": "^7.3.3", + "@babel/types": "^7.3.3", + "@types/babel__core": "^7.0.0", + "@types/babel__traverse": "^7.0.6" + } + }, + "babel-plugin-syntax-object-rest-spread": { + "version": "6.13.0", + "resolved": "https://registry.npmjs.org/babel-plugin-syntax-object-rest-spread/-/babel-plugin-syntax-object-rest-spread-6.13.0.tgz", + "integrity": "sha1-/WU28rzhODb/o6VFjEkDpZe7O/U=", + "dev": true + }, + "babel-plugin-transform-object-rest-spread": { + "version": "6.26.0", + "resolved": "https://registry.npmjs.org/babel-plugin-transform-object-rest-spread/-/babel-plugin-transform-object-rest-spread-6.26.0.tgz", + "integrity": "sha1-DzZpLVD+9rfi1LOsFHgTepY7ewY=", + "dev": true, + "requires": { + "babel-plugin-syntax-object-rest-spread": "^6.8.0", + "babel-runtime": "^6.26.0" + } + }, + "babel-preset-current-node-syntax": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/babel-preset-current-node-syntax/-/babel-preset-current-node-syntax-1.0.0.tgz", + "integrity": "sha512-mGkvkpocWJes1CmMKtgGUwCeeq0pOhALyymozzDWYomHTbDLwueDYG6p4TK1YOeYHCzBzYPsWkgTto10JubI1Q==", + "dev": true, + "requires": { + "@babel/plugin-syntax-async-generators": "^7.8.4", + "@babel/plugin-syntax-bigint": "^7.8.3", + "@babel/plugin-syntax-class-properties": "^7.8.3", + "@babel/plugin-syntax-import-meta": "^7.8.3", + "@babel/plugin-syntax-json-strings": "^7.8.3", + "@babel/plugin-syntax-logical-assignment-operators": "^7.8.3", + "@babel/plugin-syntax-nullish-coalescing-operator": "^7.8.3", + "@babel/plugin-syntax-numeric-separator": "^7.8.3", + "@babel/plugin-syntax-object-rest-spread": "^7.8.3", + "@babel/plugin-syntax-optional-catch-binding": "^7.8.3", + "@babel/plugin-syntax-optional-chaining": "^7.8.3", + "@babel/plugin-syntax-top-level-await": "^7.8.3" + } + }, + "babel-preset-jest": { + "version": "26.6.2", + "resolved": "https://registry.npmjs.org/babel-preset-jest/-/babel-preset-jest-26.6.2.tgz", + "integrity": "sha512-YvdtlVm9t3k777c5NPQIv6cxFFFapys25HiUmuSgHwIZhfifweR5c5Sf5nwE3MAbfu327CYSvps8Yx6ANLyleQ==", + "dev": true, + "requires": { + "babel-plugin-jest-hoist": "^26.6.2", + "babel-preset-current-node-syntax": "^1.0.0" + } + }, + "babel-runtime": { + "version": "6.26.0", + "resolved": "https://registry.npmjs.org/babel-runtime/-/babel-runtime-6.26.0.tgz", + "integrity": "sha1-llxwWGaOgrVde/4E/yM3vItWR/4=", + "dev": true, + "requires": { + "core-js": "^2.4.0", + "regenerator-runtime": "^0.11.0" + }, + "dependencies": { + "regenerator-runtime": { + "version": "0.11.1", + "resolved": "https://registry.npmjs.org/regenerator-runtime/-/regenerator-runtime-0.11.1.tgz", + "integrity": "sha512-MguG95oij0fC3QV3URf4V2SDYGJhJnJGqvIIgdECeODCT98wSWDAJ94SSuVpYQUoTcGUIL6L4yNB7j1DFFHSBg==", + "dev": true + } + } + }, + "babylon": { + "version": "6.18.0", + "resolved": "https://registry.npmjs.org/babylon/-/babylon-6.18.0.tgz", + "integrity": "sha512-q/UEjfGJ2Cm3oKV71DJz9d25TPnq5rhBVL2Q4fA5wcC3jcrdn7+SssEybFIxwAvvP+YCsCYNKughoF33GxgycQ==", + "dev": true + }, + "backo2": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/backo2/-/backo2-1.0.2.tgz", + "integrity": "sha1-MasayLEpNjRj41s+u2n038+6eUc=", + "dev": true + }, + "balanced-match": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.0.tgz", + "integrity": "sha1-ibTRmasr7kneFk6gK4nORi1xt2c=" + }, + "base": { + "version": "0.11.2", + "resolved": "https://registry.npmjs.org/base/-/base-0.11.2.tgz", + "integrity": "sha512-5T6P4xPgpp0YDFvSWwEZ4NoE3aM4QBQXDzmVbraCkFj8zHM+mba8SyqB5DbZWyR7mYHo6Y7BdQo3MoA4m0TeQg==", + "dev": true, + "requires": { + "cache-base": "^1.0.1", + "class-utils": "^0.3.5", + "component-emitter": "^1.2.1", + "define-property": "^1.0.0", + "isobject": "^3.0.1", + "mixin-deep": "^1.2.0", + "pascalcase": "^0.1.1" + }, + "dependencies": { + "define-property": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/define-property/-/define-property-1.0.0.tgz", + "integrity": "sha1-dp66rz9KY6rTr56NMEybvnm/sOY=", + "dev": true, + "requires": { + "is-descriptor": "^1.0.0" + } + }, + "is-accessor-descriptor": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/is-accessor-descriptor/-/is-accessor-descriptor-1.0.0.tgz", + "integrity": "sha512-m5hnHTkcVsPfqx3AKlyttIPb7J+XykHvJP2B9bZDjlhLIoEq4XoK64Vg7boZlVWYK6LUY94dYPEE7Lh0ZkZKcQ==", + "dev": true, + "requires": { + "kind-of": "^6.0.0" + } + }, + "is-data-descriptor": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/is-data-descriptor/-/is-data-descriptor-1.0.0.tgz", + "integrity": "sha512-jbRXy1FmtAoCjQkVmIVYwuuqDFUbaOeDjmed1tOGPrsMhtJA4rD9tkgA0F1qJ3gRFRXcHYVkdeaP50Q5rE/jLQ==", + "dev": true, + "requires": { + "kind-of": "^6.0.0" + } + }, + "is-descriptor": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/is-descriptor/-/is-descriptor-1.0.2.tgz", + "integrity": "sha512-2eis5WqQGV7peooDyLmNEPUrps9+SXX5c9pL3xEB+4e9HnGuDa7mB7kHxHw4CbqS9k1T2hOH3miL8n8WtiYVtg==", + "dev": true, + "requires": { + "is-accessor-descriptor": "^1.0.0", + "is-data-descriptor": "^1.0.0", + "kind-of": "^6.0.2" + } + } + } + }, + "base64-arraybuffer": { + "version": "0.1.4", + "resolved": "https://registry.npmjs.org/base64-arraybuffer/-/base64-arraybuffer-0.1.4.tgz", + "integrity": "sha1-mBjHngWbE1X5fgQooBfIOOkLqBI=", + "dev": true + }, + "base64-js": { + "version": "1.5.1", + "resolved": "https://registry.npmjs.org/base64-js/-/base64-js-1.5.1.tgz", + "integrity": "sha512-AKpaYlHn8t4SVbOHCy+b5+KKgvR4vrsD8vbvrbiQJps7fKDTkjkDry6ji0rUJjC0kzbNePLwzxq8iypo41qeWA==", + "dev": true + }, + "base64id": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/base64id/-/base64id-1.0.0.tgz", + "integrity": "sha1-R2iMuZu2gE8OBtPnY7HDLlfY5rY=", + "dev": true + }, + "batch": { + "version": "0.6.1", + "resolved": "https://registry.npmjs.org/batch/-/batch-0.6.1.tgz", + "integrity": "sha1-3DQxT05nkxgJP8dgJyUl+UvyXBY=", + "dev": true + }, + "bcrypt-pbkdf": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/bcrypt-pbkdf/-/bcrypt-pbkdf-1.0.2.tgz", + "integrity": "sha1-pDAdOJtqQ/m2f/PKEaP2Y342Dp4=", + "dev": true, + "requires": { + "tweetnacl": "^0.14.3" + } + }, + "better-assert": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/better-assert/-/better-assert-1.0.2.tgz", + "integrity": "sha1-QIZrnhueC1W0gYlDEeaPr/rrxSI=", + "dev": true, + "requires": { + "callsite": "1.0.0" + } + }, + "bfj": { + "version": "6.1.2", + "resolved": "https://registry.npmjs.org/bfj/-/bfj-6.1.2.tgz", + "integrity": "sha512-BmBJa4Lip6BPRINSZ0BPEIfB1wUY/9rwbwvIHQA1KjX9om29B6id0wnWXq7m3bn5JrUVjeOTnVuhPT1FiHwPGw==", + "dev": true, + "requires": { + "bluebird": "^3.5.5", + "check-types": "^8.0.3", + "hoopy": "^0.1.4", + "tryer": "^1.0.1" + } + }, + "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.1.0", + "resolved": "https://registry.npmjs.org/binary-extensions/-/binary-extensions-2.1.0.tgz", + "integrity": "sha512-1Yj8h9Q+QDF5FzhMs/c9+6UntbD5MkRfRwac8DoEm9ZfUBZ7tZ55YcGVAzEe4bXsdQHEk+s9S5wsOKVdZrw0tQ==", + "dev": true + }, + "binaryextensions": { + "version": "2.3.0", + "resolved": "https://registry.npmjs.org/binaryextensions/-/binaryextensions-2.3.0.tgz", + "integrity": "sha512-nAihlQsYGyc5Bwq6+EsubvANYGExeJKHDO3RjnvwU042fawQTQfM3Kxn7IHUXQOz4bzfwsGYYHGSvXyW4zOGLg==", + "dev": true + }, + "blob": { + "version": "0.0.5", + "resolved": "https://registry.npmjs.org/blob/-/blob-0.0.5.tgz", + "integrity": "sha512-gaqbzQPqOoamawKg0LGVd7SzLgXS+JH61oWprSLH+P+abTczqJbhTR8CmJ2u9/bUYNmHTGJx/UEmn6doAvvuig==", + "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 + }, + "bn.js": { + "version": "5.1.3", + "resolved": "https://registry.npmjs.org/bn.js/-/bn.js-5.1.3.tgz", + "integrity": "sha512-GkTiFpjFtUzU9CbMeJ5iazkCzGL3jrhzerzZIuqLABjbwRaFt33I9tUdSNryIptM+RxDet6OKm2WnLXzW51KsQ==", + "dev": true + }, + "body-parser": { + "version": "1.19.0", + "resolved": "https://registry.npmjs.org/body-parser/-/body-parser-1.19.0.tgz", + "integrity": "sha512-dhEPs72UPbDnAQJ9ZKMNTP6ptJaionhP5cBb541nXPlW60Jepo9RV/a4fX4XWW9CuFNK22krhrj1+rgzifNCsw==", + "dev": true, + "requires": { + "bytes": "3.1.0", + "content-type": "~1.0.4", + "debug": "2.6.9", + "depd": "~1.1.2", + "http-errors": "1.7.2", + "iconv-lite": "0.4.24", + "on-finished": "~2.3.0", + "qs": "6.7.0", + "raw-body": "2.4.0", + "type-is": "~1.6.17" + }, + "dependencies": { + "debug": { + "version": "2.6.9", + "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", + "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", + "dev": true, + "requires": { + "ms": "2.0.0" + } + }, + "http-errors": { + "version": "1.7.2", + "resolved": "https://registry.npmjs.org/http-errors/-/http-errors-1.7.2.tgz", + "integrity": "sha512-uUQBt3H/cSIVfch6i1EuPNy/YsRSOUBXTVfZ+yR7Zjez3qjBz6i9+i4zjNaoqcoFVI4lQJ5plg63TvGfRSDCRg==", + "dev": true, + "requires": { + "depd": "~1.1.2", + "inherits": "2.0.3", + "setprototypeof": "1.1.1", + "statuses": ">= 1.5.0 < 2", + "toidentifier": "1.0.0" + } + }, + "iconv-lite": { + "version": "0.4.24", + "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.4.24.tgz", + "integrity": "sha512-v3MXnZAcvnywkTUEZomIActle7RXXeedOR31wwl7VlyoXO4Qi9arvSenNQWne1TcRwhCL1HwLI21bEqdpj8/rA==", + "dev": true, + "requires": { + "safer-buffer": ">= 2.1.2 < 3" + } + }, + "inherits": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.3.tgz", + "integrity": "sha1-Yzwsg+PaQqUC9SRmAiSA9CCCYd4=", + "dev": true + }, + "ms": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", + "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=", + "dev": true + }, + "qs": { + "version": "6.7.0", + "resolved": "https://registry.npmjs.org/qs/-/qs-6.7.0.tgz", + "integrity": "sha512-VCdBRNFTX1fyE7Nb6FYoURo/SPe62QCaAyzJvUjwRaIsc+NePBEniHlvxFmmX56+HZphIGtV0XeCirBtpDrTyQ==", + "dev": true + }, + "raw-body": { + "version": "2.4.0", + "resolved": "https://registry.npmjs.org/raw-body/-/raw-body-2.4.0.tgz", + "integrity": "sha512-4Oz8DUIwdvoa5qMJelxipzi/iJIi40O5cGV1wNYp5hvZP8ZN0T+jiNkL0QepXs+EsQ9XJ8ipEDoiH70ySUJP3Q==", + "dev": true, + "requires": { + "bytes": "3.1.0", + "http-errors": "1.7.2", + "iconv-lite": "0.4.24", + "unpipe": "1.0.0" + } + }, + "statuses": { + "version": "1.5.0", + "resolved": "https://registry.npmjs.org/statuses/-/statuses-1.5.0.tgz", + "integrity": "sha1-Fhx9rBd2Wf2YEfQ3cfqZOBR4Yow=", + "dev": true + } + } + }, + "bonjour": { + "version": "3.5.0", + "resolved": "https://registry.npmjs.org/bonjour/-/bonjour-3.5.0.tgz", + "integrity": "sha1-jokKGD2O6aI5OzhExpGkK897yfU=", + "dev": true, + "requires": { + "array-flatten": "^2.1.0", + "deep-equal": "^1.0.1", + "dns-equal": "^1.0.0", + "dns-txt": "^2.0.2", + "multicast-dns": "^6.0.1", + "multicast-dns-service-types": "^1.1.0" + }, + "dependencies": { + "array-flatten": { + "version": "2.1.2", + "resolved": "https://registry.npmjs.org/array-flatten/-/array-flatten-2.1.2.tgz", + "integrity": "sha512-hNfzcOV8W4NdualtqBFPyVO+54DSJuZGY9qT4pRroB6S9e3iiido2ISIC5h9R2sPJ8H3FHCIiEnsv1lPXO3KtQ==", + "dev": true + } + } + }, + "boolbase": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/boolbase/-/boolbase-1.0.0.tgz", + "integrity": "sha1-aN/1++YMUes3cl6p4+0xDcwed24=", + "dev": true + }, + "bootstrap": { + "version": "4.5.0", + "resolved": "https://registry.npmjs.org/bootstrap/-/bootstrap-4.5.0.tgz", + "integrity": "sha512-Z93QoXvodoVslA+PWNdk23Hze4RBYIkpb5h8I2HY2Tu2h7A0LpAgLcyrhrSUyo2/Oxm2l1fRZPs1e5hnxnliXA==" + }, + "brace-expansion": { + "version": "1.1.11", + "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.11.tgz", + "integrity": "sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA==", + "requires": { + "balanced-match": "^1.0.0", + "concat-map": "0.0.1" + } + }, + "braces": { + "version": "3.0.2", + "resolved": "https://registry.npmjs.org/braces/-/braces-3.0.2.tgz", + "integrity": "sha512-b8um+L1RzM3WDSzvhm6gIz1yfTbBt6YTlcEKAvsmqCZZFw46z626lVj9j1yEPW33H5H+lBQpZMP1k8l+78Ha0A==", + "dev": true, + "requires": { + "fill-range": "^7.0.1" + } + }, + "brorand": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/brorand/-/brorand-1.1.0.tgz", + "integrity": "sha1-EsJe/kCkXjwyPrhnWgoM5XsiNx8=", + "dev": true + }, + "browser-process-hrtime": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/browser-process-hrtime/-/browser-process-hrtime-1.0.0.tgz", + "integrity": "sha512-9o5UecI3GhkpM6DrXr69PblIuWxPKk9Y0jHBRhdocZ2y7YECBFCsHm79Pr3OyR2AvjhDkabFJaDJMYRazHgsow==", + "dev": true + }, + "browser-sync": { + "version": "2.26.7", + "resolved": "https://registry.npmjs.org/browser-sync/-/browser-sync-2.26.7.tgz", + "integrity": "sha512-lY3emme0OyvA2ujEMpRmyRy9LY6gHLuTr2/ABxhIm3lADOiRXzP4dgekvnDrQqZ/Ec2Fz19lEjm6kglSG5766w==", + "dev": true, + "requires": { + "browser-sync-client": "^2.26.6", + "browser-sync-ui": "^2.26.4", + "bs-recipes": "1.3.4", + "bs-snippet-injector": "^2.0.1", + "chokidar": "^2.0.4", + "connect": "3.6.6", + "connect-history-api-fallback": "^1", + "dev-ip": "^1.0.1", + "easy-extender": "^2.3.4", + "eazy-logger": "^3", + "etag": "^1.8.1", + "fresh": "^0.5.2", + "fs-extra": "3.0.1", + "http-proxy": "1.15.2", + "immutable": "^3", + "localtunnel": "1.9.2", + "micromatch": "^3.1.10", + "opn": "5.3.0", + "portscanner": "2.1.1", + "qs": "6.2.3", + "raw-body": "^2.3.2", + "resp-modifier": "6.0.2", + "rx": "4.1.0", + "send": "0.16.2", + "serve-index": "1.9.1", + "serve-static": "1.13.2", + "server-destroy": "1.0.1", + "socket.io": "2.1.1", + "ua-parser-js": "0.7.17", + "yargs": "6.4.0" + }, + "dependencies": { + "ansi-regex": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-2.1.1.tgz", + "integrity": "sha1-w7M6te42DYbg5ijwRorn7yfWVN8=", + "dev": true + }, + "anymatch": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/anymatch/-/anymatch-2.0.0.tgz", + "integrity": "sha512-5teOsQWABXHHBFP9y3skS5P3d/WfWXpv3FUpy+LorMrNYaT9pI4oLMQX7jzQ2KklNpGpWHzdCXTDT2Y3XGlZBw==", + "dev": true, + "requires": { + "micromatch": "^3.1.4", + "normalize-path": "^2.1.1" + }, + "dependencies": { + "normalize-path": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/normalize-path/-/normalize-path-2.1.1.tgz", + "integrity": "sha1-GrKLVW4Zg2Oowab35vogE3/mrtk=", + "dev": true, + "requires": { + "remove-trailing-separator": "^1.0.1" + } + } + } + }, + "binary-extensions": { + "version": "1.13.1", + "resolved": "https://registry.npmjs.org/binary-extensions/-/binary-extensions-1.13.1.tgz", + "integrity": "sha512-Un7MIEDdUC5gNpcGDV97op1Ywk748MpHcFTHoYs6qnj1Z3j7I53VG3nwZhKzoBZmbdRNnb6WRdFlwl7tSDuZGw==", + "dev": true + }, + "braces": { + "version": "2.3.2", + "resolved": "https://registry.npmjs.org/braces/-/braces-2.3.2.tgz", + "integrity": "sha512-aNdbnj9P8PjdXU4ybaWLK2IF3jc/EoDYbC7AazW6to3TRsfXxscC9UXOB5iDiEQrkyIbWp2SLQda4+QAa7nc3w==", + "dev": true, + "requires": { + "arr-flatten": "^1.1.0", + "array-unique": "^0.3.2", + "extend-shallow": "^2.0.1", + "fill-range": "^4.0.0", + "isobject": "^3.0.1", + "repeat-element": "^1.1.2", + "snapdragon": "^0.8.1", + "snapdragon-node": "^2.0.1", + "split-string": "^3.0.2", + "to-regex": "^3.0.1" + } + }, + "camelcase": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/camelcase/-/camelcase-3.0.0.tgz", + "integrity": "sha1-MvxLn82vhF/N9+c7uXysImHwqwo=", + "dev": true + }, + "chokidar": { + "version": "2.1.8", + "resolved": "https://registry.npmjs.org/chokidar/-/chokidar-2.1.8.tgz", + "integrity": "sha512-ZmZUazfOzf0Nve7duiCKD23PFSCs4JPoYyccjUFF3aQkQadqBhfzhjkwBH2mNOG9cTBwhamM37EIsIkZw3nRgg==", + "dev": true, + "requires": { + "anymatch": "^2.0.0", + "async-each": "^1.0.1", + "braces": "^2.3.2", + "fsevents": "^1.2.7", + "glob-parent": "^3.1.0", + "inherits": "^2.0.3", + "is-binary-path": "^1.0.0", + "is-glob": "^4.0.0", + "normalize-path": "^3.0.0", + "path-is-absolute": "^1.0.0", + "readdirp": "^2.2.1", + "upath": "^1.1.1" + } + }, + "cliui": { + "version": "3.2.0", + "resolved": "https://registry.npmjs.org/cliui/-/cliui-3.2.0.tgz", + "integrity": "sha1-EgYBU3qRbSmUD5NNo7SNWFo5IT0=", + "dev": true, + "requires": { + "string-width": "^1.0.1", + "strip-ansi": "^3.0.1", + "wrap-ansi": "^2.0.0" + } + }, + "extend-shallow": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/extend-shallow/-/extend-shallow-2.0.1.tgz", + "integrity": "sha1-Ua99YUrZqfYQ6huvu5idaxxWiQ8=", + "dev": true, + "requires": { + "is-extendable": "^0.1.0" + } + }, + "fill-range": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/fill-range/-/fill-range-4.0.0.tgz", + "integrity": "sha1-1USBHUKPmOsGpj3EAtJAPDKMOPc=", + "dev": true, + "requires": { + "extend-shallow": "^2.0.1", + "is-number": "^3.0.0", + "repeat-string": "^1.6.1", + "to-regex-range": "^2.1.0" + } + }, + "fs-extra": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/fs-extra/-/fs-extra-3.0.1.tgz", + "integrity": "sha1-N5TzeMWLNC6n27sjCVEJxLO2IpE=", + "dev": true, + "requires": { + "graceful-fs": "^4.1.2", + "jsonfile": "^3.0.0", + "universalify": "^0.1.0" + } + }, + "fsevents": { + "version": "1.2.13", + "resolved": "https://registry.npmjs.org/fsevents/-/fsevents-1.2.13.tgz", + "integrity": "sha512-oWb1Z6mkHIskLzEJ/XWX0srkpkTQ7vaopMQkyaEIoq0fmtFVxOthb8cCxeT+p3ynTdkk/RZwbgG4brR5BeWECw==", + "dev": true, + "optional": true + }, + "get-caller-file": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/get-caller-file/-/get-caller-file-1.0.3.tgz", + "integrity": "sha512-3t6rVToeoZfYSGd8YoLFR2DJkiQrIiUrGcjvFX2mDw3bn6k2OtwHN0TNCLbBO+w8qTvimhDkv+LSscbJY1vE6w==", + "dev": true + }, + "glob-parent": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-3.1.0.tgz", + "integrity": "sha1-nmr2KZ2NO9K9QEMIMr0RPfkGxa4=", + "dev": true, + "requires": { + "is-glob": "^3.1.0", + "path-dirname": "^1.0.0" + }, + "dependencies": { + "is-glob": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-3.1.0.tgz", + "integrity": "sha1-e6WuJCF4BKxwcHuWkiVnSGzD6Eo=", + "dev": true, + "requires": { + "is-extglob": "^2.1.0" + } + } + } + }, + "is-binary-path": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/is-binary-path/-/is-binary-path-1.0.1.tgz", + "integrity": "sha1-dfFmQrSA8YenEcgUFh/TpKdlWJg=", + "dev": true, + "requires": { + "binary-extensions": "^1.0.0" + } + }, + "is-fullwidth-code-point": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-1.0.0.tgz", + "integrity": "sha1-754xOG8DGn8NZDr4L95QxFfvAMs=", + "dev": true, + "requires": { + "number-is-nan": "^1.0.0" + } + }, + "is-number": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/is-number/-/is-number-3.0.0.tgz", + "integrity": "sha1-JP1iAaR4LPUFYcgQJ2r8fRLXEZU=", + "dev": true, + "requires": { + "kind-of": "^3.0.2" + } + }, + "jsonfile": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/jsonfile/-/jsonfile-3.0.1.tgz", + "integrity": "sha1-pezG9l9T9mLEQVx2daAzHQmS7GY=", + "dev": true, + "requires": { + "graceful-fs": "^4.1.6" + } + }, + "kind-of": { + "version": "3.2.2", + "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-3.2.2.tgz", + "integrity": "sha1-MeohpzS6ubuw8yRm2JOupR5KPGQ=", + "dev": true, + "requires": { + "is-buffer": "^1.1.5" + } + }, + "qs": { + "version": "6.2.3", + "resolved": "https://registry.npmjs.org/qs/-/qs-6.2.3.tgz", + "integrity": "sha1-HPyyXBCpsrSDBT/zn138kjOQjP4=", + "dev": true + }, + "readdirp": { + "version": "2.2.1", + "resolved": "https://registry.npmjs.org/readdirp/-/readdirp-2.2.1.tgz", + "integrity": "sha512-1JU/8q+VgFZyxwrJ+SVIOsh+KywWGpds3NTqikiKpDMZWScmAYyKIgqkO+ARvNWJfXeXR1zxz7aHF4u4CyH6vQ==", + "dev": true, + "requires": { + "graceful-fs": "^4.1.11", + "micromatch": "^3.1.10", + "readable-stream": "^2.0.2" + } + }, + "require-main-filename": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/require-main-filename/-/require-main-filename-1.0.1.tgz", + "integrity": "sha1-l/cXtp1IeE9fUmpsWqj/3aBVpNE=", + "dev": true + }, + "string-width": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/string-width/-/string-width-1.0.2.tgz", + "integrity": "sha1-EYvfW4zcUaKn5w0hHgfisLmxB9M=", + "dev": true, + "requires": { + "code-point-at": "^1.0.0", + "is-fullwidth-code-point": "^1.0.0", + "strip-ansi": "^3.0.0" + } + }, + "strip-ansi": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-3.0.1.tgz", + "integrity": "sha1-ajhfuIU9lS1f8F0Oiq+UJ43GPc8=", + "dev": true, + "requires": { + "ansi-regex": "^2.0.0" + } + }, + "to-regex-range": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/to-regex-range/-/to-regex-range-2.1.1.tgz", + "integrity": "sha1-fIDBe53+vlmeJzZ+DU3VWQFB2zg=", + "dev": true, + "requires": { + "is-number": "^3.0.0", + "repeat-string": "^1.6.1" + } + }, + "which-module": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/which-module/-/which-module-1.0.0.tgz", + "integrity": "sha1-u6Y8qGGUiZT/MHc2CJ47lgJsKk8=", + "dev": true + }, + "wrap-ansi": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/wrap-ansi/-/wrap-ansi-2.1.0.tgz", + "integrity": "sha1-2Pw9KE3QV5T+hJc8rs3Rz4JP3YU=", + "dev": true, + "requires": { + "string-width": "^1.0.1", + "strip-ansi": "^3.0.1" + } + }, + "y18n": { + "version": "3.2.1", + "resolved": "https://registry.npmjs.org/y18n/-/y18n-3.2.1.tgz", + "integrity": "sha1-bRX7qITAhnnA136I53WegR4H+kE=", + "dev": true + }, + "yargs": { + "version": "6.4.0", + "resolved": "https://registry.npmjs.org/yargs/-/yargs-6.4.0.tgz", + "integrity": "sha1-gW4ahm1VmMzzTlWW3c4i2S2kkNQ=", + "dev": true, + "requires": { + "camelcase": "^3.0.0", + "cliui": "^3.2.0", + "decamelize": "^1.1.1", + "get-caller-file": "^1.0.1", + "os-locale": "^1.4.0", + "read-pkg-up": "^1.0.1", + "require-directory": "^2.1.1", + "require-main-filename": "^1.0.1", + "set-blocking": "^2.0.0", + "string-width": "^1.0.2", + "which-module": "^1.0.0", + "window-size": "^0.2.0", + "y18n": "^3.2.1", + "yargs-parser": "^4.1.0" + } + }, + "yargs-parser": { + "version": "4.2.1", + "resolved": "https://registry.npmjs.org/yargs-parser/-/yargs-parser-4.2.1.tgz", + "integrity": "sha1-KczqwNxPA8bIe0qfIX3RjJ90hxw=", + "dev": true, + "requires": { + "camelcase": "^3.0.0" + } + } + } + }, + "browser-sync-client": { + "version": "2.26.13", + "resolved": "https://registry.npmjs.org/browser-sync-client/-/browser-sync-client-2.26.13.tgz", + "integrity": "sha512-p2VbZoYrpuDhkreq+/Sv1MkToHklh7T1OaIntDwpG6Iy2q/XkBcgwPcWjX+WwRNiZjN8MEehxIjEUh12LweLmQ==", + "dev": true, + "requires": { + "etag": "1.8.1", + "fresh": "0.5.2", + "mitt": "^1.1.3", + "rxjs": "^5.5.6" + }, + "dependencies": { + "rxjs": { + "version": "5.5.12", + "resolved": "https://registry.npmjs.org/rxjs/-/rxjs-5.5.12.tgz", + "integrity": "sha512-xx2itnL5sBbqeeiVgNPVuQQ1nC8Jp2WfNJhXWHmElW9YmrpS9UVnNzhP3EH3HFqexO5Tlp8GhYY+WEcqcVMvGw==", + "dev": true, + "requires": { + "symbol-observable": "1.0.1" + } + }, + "symbol-observable": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/symbol-observable/-/symbol-observable-1.0.1.tgz", + "integrity": "sha1-g0D8RwLDEi310iKI+IKD9RPT/dQ=", + "dev": true + } + } + }, + "browser-sync-ui": { + "version": "2.26.13", + "resolved": "https://registry.npmjs.org/browser-sync-ui/-/browser-sync-ui-2.26.13.tgz", + "integrity": "sha512-6NJ/pCnhCnBMzaty1opWo7ipDmFAIk8U71JMQGKJxblCUaGfdsbF2shf6XNZSkXYia1yS0vwKu9LIOzpXqQZCA==", + "dev": true, + "requires": { + "async-each-series": "0.1.1", + "connect-history-api-fallback": "^1", + "immutable": "^3", + "server-destroy": "1.0.1", + "socket.io-client": "^2.0.4", + "stream-throttle": "^0.1.3" + } + }, + "browser-sync-webpack-plugin": { + "version": "2.2.2", + "resolved": "https://registry.npmjs.org/browser-sync-webpack-plugin/-/browser-sync-webpack-plugin-2.2.2.tgz", + "integrity": "sha512-x92kl8LdBi4dp6YVXYqrSoDkOCOLCeBOrYSY0h9Sk1VcCDSoZC1Vc62eae6TfC2ljN4/L+aYlkzE46kirHzbgA==", + "dev": true, + "requires": { + "lodash": "^4" + } + }, + "browserify-aes": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/browserify-aes/-/browserify-aes-1.2.0.tgz", + "integrity": "sha512-+7CHXqGuspUn/Sl5aO7Ea0xWGAtETPXNSAjHo48JfLdPWcMng33Xe4znFvQweqc/uzk5zSOI3H52CYnjCfb5hA==", + "dev": true, + "requires": { + "buffer-xor": "^1.0.3", + "cipher-base": "^1.0.0", + "create-hash": "^1.1.0", + "evp_bytestokey": "^1.0.3", + "inherits": "^2.0.1", + "safe-buffer": "^5.0.1" + } + }, + "browserify-cipher": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/browserify-cipher/-/browserify-cipher-1.0.1.tgz", + "integrity": "sha512-sPhkz0ARKbf4rRQt2hTpAHqn47X3llLkUGn+xEJzLjwY8LRs2p0v7ljvI5EyoRO/mexrNunNECisZs+gw2zz1w==", + "dev": true, + "requires": { + "browserify-aes": "^1.0.4", + "browserify-des": "^1.0.0", + "evp_bytestokey": "^1.0.0" + } + }, + "browserify-des": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/browserify-des/-/browserify-des-1.0.2.tgz", + "integrity": "sha512-BioO1xf3hFwz4kc6iBhI3ieDFompMhrMlnDFC4/0/vd5MokpuAc3R+LYbwTA9A5Yc9pq9UYPqffKpW2ObuwX5A==", + "dev": true, + "requires": { + "cipher-base": "^1.0.1", + "des.js": "^1.0.0", + "inherits": "^2.0.1", + "safe-buffer": "^5.1.2" + } + }, + "browserify-rsa": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/browserify-rsa/-/browserify-rsa-4.1.0.tgz", + "integrity": "sha512-AdEER0Hkspgno2aR97SAf6vi0y0k8NuOpGnVH3O99rcA5Q6sh8QxcngtHuJ6uXwnfAXNM4Gn1Gb7/MV1+Ymbog==", + "dev": true, + "requires": { + "bn.js": "^5.0.0", + "randombytes": "^2.0.1" + } + }, + "browserify-sign": { + "version": "4.2.1", + "resolved": "https://registry.npmjs.org/browserify-sign/-/browserify-sign-4.2.1.tgz", + "integrity": "sha512-/vrA5fguVAKKAVTNJjgSm1tRQDHUU6DbwO9IROu/0WAzC8PKhucDSh18J0RMvVeHAn5puMd+QHC2erPRNf8lmg==", + "dev": true, + "requires": { + "bn.js": "^5.1.1", + "browserify-rsa": "^4.0.1", + "create-hash": "^1.2.0", + "create-hmac": "^1.1.7", + "elliptic": "^6.5.3", + "inherits": "^2.0.4", + "parse-asn1": "^5.1.5", + "readable-stream": "^3.6.0", + "safe-buffer": "^5.2.0" + }, + "dependencies": { + "readable-stream": { + "version": "3.6.0", + "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-3.6.0.tgz", + "integrity": "sha512-BViHy7LKeTz4oNnkcLJ+lVSL6vpiFeX6/d3oSH8zCW7UxP2onchk+vTGB143xuFjHS3deTgkKoXXymXqymiIdA==", + "dev": true, + "requires": { + "inherits": "^2.0.3", + "string_decoder": "^1.1.1", + "util-deprecate": "^1.0.1" + } + }, + "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 + } + } + }, + "browserify-zlib": { + "version": "0.2.0", + "resolved": "https://registry.npmjs.org/browserify-zlib/-/browserify-zlib-0.2.0.tgz", + "integrity": "sha512-Z942RysHXmJrhqk88FmKBVq/v5tqmSkDz7p54G/MGyjMnCFFnC79XWNbg+Vta8W6Wb2qtSZTSxIGkJrRpCFEiA==", + "dev": true, + "requires": { + "pako": "~1.0.5" + } + }, + "browserslist": { + "version": "4.14.7", + "resolved": "https://registry.npmjs.org/browserslist/-/browserslist-4.14.7.tgz", + "integrity": "sha512-BSVRLCeG3Xt/j/1cCGj1019Wbty0H+Yvu2AOuZSuoaUWn3RatbL33Cxk+Q4jRMRAbOm0p7SLravLjpnT6s0vzQ==", + "dev": true, + "requires": { + "caniuse-lite": "^1.0.30001157", + "colorette": "^1.2.1", + "electron-to-chromium": "^1.3.591", + "escalade": "^3.1.1", + "node-releases": "^1.1.66" + } + }, + "bs-logger": { + "version": "0.2.6", + "resolved": "https://registry.npmjs.org/bs-logger/-/bs-logger-0.2.6.tgz", + "integrity": "sha512-pd8DCoxmbgc7hyPKOvxtqNcjYoOsABPQdcCUjGp3d42VR2CX1ORhk2A87oqqu5R1kk+76nsxZupkmyd+MVtCog==", + "dev": true, + "requires": { + "fast-json-stable-stringify": "2.x" + } + }, + "bs-recipes": { + "version": "1.3.4", + "resolved": "https://registry.npmjs.org/bs-recipes/-/bs-recipes-1.3.4.tgz", + "integrity": "sha1-DS1NSKcYyMBEdp/cT4lZLci2lYU=", + "dev": true + }, + "bs-snippet-injector": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/bs-snippet-injector/-/bs-snippet-injector-2.0.1.tgz", + "integrity": "sha1-YbU5PxH1JVntEgaTEANDtu2wTdU=", + "dev": true + }, + "bser": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/bser/-/bser-2.1.1.tgz", + "integrity": "sha512-gQxTNE/GAfIIrmHLUE3oJyp5FO6HRBfhjnw4/wMmA63ZGDJnWBmgY/lyQBpnDUkGmAhbSe39tx2d/iTOAfglwQ==", + "dev": true, + "requires": { + "node-int64": "^0.4.0" + } + }, + "buffer": { + "version": "4.9.2", + "resolved": "https://registry.npmjs.org/buffer/-/buffer-4.9.2.tgz", + "integrity": "sha512-xq+q3SRMOxGivLhBNaUdC64hDTQwejJ+H0T/NB1XMtTVEwNTrfFF3gAxiyW0Bu/xWEGhjVKgUcMhCrUy2+uCWg==", + "dev": true, + "requires": { + "base64-js": "^1.0.2", + "ieee754": "^1.1.4", + "isarray": "^1.0.0" + } + }, + "buffer-from": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/buffer-from/-/buffer-from-1.1.1.tgz", + "integrity": "sha512-MQcXEUbCKtEo7bhqEs6560Hyd4XaovZlO/k9V3hjVUF/zwW7KBVdSK4gIt/bzwS9MbR5qob+F5jusZsb0YQK2A==", + "dev": true + }, + "buffer-indexof": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/buffer-indexof/-/buffer-indexof-1.1.1.tgz", + "integrity": "sha512-4/rOEg86jivtPTeOUUT61jJO1Ya1TrR/OkqCSZDyq84WJh3LuuiphBYJN+fm5xufIk4XAFcEwte/8WzC8If/1g==", + "dev": true + }, + "buffer-xor": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/buffer-xor/-/buffer-xor-1.0.3.tgz", + "integrity": "sha1-JuYe0UIvtw3ULm42cp7VHYVf6Nk=", + "dev": true + }, + "builtin-modules": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/builtin-modules/-/builtin-modules-1.1.1.tgz", + "integrity": "sha1-Jw8HbFpywC9bZaR9+Uxf46J4iS8=", + "dev": true + }, + "builtin-status-codes": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/builtin-status-codes/-/builtin-status-codes-3.0.0.tgz", + "integrity": "sha1-hZgoeOIbmOHGZCXgPQF0eI9Wnug=", + "dev": true + }, + "builtins": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/builtins/-/builtins-1.0.3.tgz", + "integrity": "sha1-y5T662HIaWRR2zZTThQi+U8K7og=", + "dev": true + }, + "bytes": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/bytes/-/bytes-3.1.0.tgz", + "integrity": "sha512-zauLjrfCG+xvoyaqLoV8bLVXXNGC4JqlxFCutSDWA6fJrTo2ZuvLYTqZ7aHBLZSMOopbzwv8f+wZcVzfVTI2Dg==", + "dev": true + }, + "cacache": { + "version": "12.0.4", + "resolved": "https://registry.npmjs.org/cacache/-/cacache-12.0.4.tgz", + "integrity": "sha512-a0tMB40oefvuInr4Cwb3GerbL9xTj1D5yg0T5xrjGCGyfvbxseIXX7BAO/u/hIXdafzOI5JC3wDwHyf24buOAQ==", + "dev": true, + "requires": { + "bluebird": "^3.5.5", + "chownr": "^1.1.1", + "figgy-pudding": "^3.5.1", + "glob": "^7.1.4", + "graceful-fs": "^4.1.15", + "infer-owner": "^1.0.3", + "lru-cache": "^5.1.1", + "mississippi": "^3.0.0", + "mkdirp": "^0.5.1", + "move-concurrently": "^1.0.1", + "promise-inflight": "^1.0.1", + "rimraf": "^2.6.3", + "ssri": "^6.0.1", + "unique-filename": "^1.1.1", + "y18n": "^4.0.0" + }, + "dependencies": { + "glob": { + "version": "7.1.6", + "resolved": "https://registry.npmjs.org/glob/-/glob-7.1.6.tgz", + "integrity": "sha512-LwaxwyZ72Lk7vZINtNNrywX0ZuLyStrdDtabefZKAY5ZGJhVtgdznluResxNmPitE0SAO+O26sWTHeKSI2wMBA==", + "dev": true, + "requires": { + "fs.realpath": "^1.0.0", + "inflight": "^1.0.4", + "inherits": "2", + "minimatch": "^3.0.4", + "once": "^1.3.0", + "path-is-absolute": "^1.0.0" + } + }, + "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, + "requires": { + "yallist": "^3.0.2" + } + }, + "rimraf": { + "version": "2.7.1", + "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-2.7.1.tgz", + "integrity": "sha512-uWjbaKIK3T1OSVptzX7Nl6PvQ3qAGtKEtVRjRuazjfL3Bx5eI409VZSqgND+4UNnmzLVdPj9FqFJNPqBZFve4w==", + "dev": true, + "requires": { + "glob": "^7.1.3" + } + }, + "yallist": { + "version": "3.1.1", + "resolved": "https://registry.npmjs.org/yallist/-/yallist-3.1.1.tgz", + "integrity": "sha512-a4UGQaWPH59mOXUYnAG2ewncQS4i4F43Tv3JoAM+s2VDAmS9NsK8GpDMLrCHPksFT7h3K6TOoUNn2pb7RoXx4g==", + "dev": true + } + } + }, + "cache-base": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/cache-base/-/cache-base-1.0.1.tgz", + "integrity": "sha512-AKcdTnFSWATd5/GCPRxr2ChwIJ85CeyrEyjRHlKxQ56d4XJMGym0uAiKn0xbLOGOl3+yRpOTi484dVCEc5AUzQ==", + "dev": true, + "requires": { + "collection-visit": "^1.0.0", + "component-emitter": "^1.2.1", + "get-value": "^2.0.6", + "has-value": "^1.0.0", + "isobject": "^3.0.1", + "set-value": "^2.0.0", + "to-object-path": "^0.3.0", + "union-value": "^1.0.0", + "unset-value": "^1.0.0" + } + }, + "call-bind": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/call-bind/-/call-bind-1.0.0.tgz", + "integrity": "sha512-AEXsYIyyDY3MCzbwdhzG3Jx1R0J2wetQyUynn6dYHAO+bg8l1k7jwZtRv4ryryFs7EP+NDlikJlVe59jr0cM2w==", + "dev": true, + "requires": { + "function-bind": "^1.1.1", + "get-intrinsic": "^1.0.0" + } + }, + "call-me-maybe": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/call-me-maybe/-/call-me-maybe-1.0.1.tgz", + "integrity": "sha1-JtII6onje1y95gJQoV8DHBak1ms=", + "dev": true + }, + "caller-callsite": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/caller-callsite/-/caller-callsite-2.0.0.tgz", + "integrity": "sha1-hH4PzgoiN1CpoCfFSzNzGtMVQTQ=", + "dev": true, + "requires": { + "callsites": "^2.0.0" + }, + "dependencies": { + "callsites": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/callsites/-/callsites-2.0.0.tgz", + "integrity": "sha1-BuuE8A7qQT2oav/vrL/7Ngk7PFA=", + "dev": true + } + } + }, + "caller-path": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/caller-path/-/caller-path-2.0.0.tgz", + "integrity": "sha1-Ro+DBE42mrIBD6xfBs7uFbsssfQ=", + "dev": true, + "requires": { + "caller-callsite": "^2.0.0" + } + }, + "callsite": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/callsite/-/callsite-1.0.0.tgz", + "integrity": "sha1-KAOY5dZkvXQDi28JBRU+borxvCA=", + "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 + }, + "camel-case": { + "version": "4.1.1", + "resolved": "https://registry.npmjs.org/camel-case/-/camel-case-4.1.1.tgz", + "integrity": "sha512-7fa2WcG4fYFkclIvEmxBbTvmibwF2/agfEBc6q3lOpVu0A13ltLsA+Hr/8Hp6kp5f+G7hKi6t8lys6XxP+1K6Q==", + "dev": true, + "requires": { + "pascal-case": "^3.1.1", + "tslib": "^1.10.0" + }, + "dependencies": { + "tslib": { + "version": "1.14.1", + "resolved": "https://registry.npmjs.org/tslib/-/tslib-1.14.1.tgz", + "integrity": "sha512-Xni35NKzjgMrwevysHTCArtLDpPvye8zV/0E4EyYn43P7/7qvQwPh9BGkHewbMulVntbigmcT7rdX3BNo9wRJg==", + "dev": true + } + } + }, + "camelcase": { + "version": "5.3.1", + "resolved": "https://registry.npmjs.org/camelcase/-/camelcase-5.3.1.tgz", + "integrity": "sha512-L28STB170nwWS63UjtlEOE3dldQApaJXZkOI1uMFfzf3rRuPegHaHesyee+YxQ+W6SvRDQV6UrdOdRiR153wJg==" + }, + "camelcase-keys": { + "version": "6.2.2", + "resolved": "https://registry.npmjs.org/camelcase-keys/-/camelcase-keys-6.2.2.tgz", + "integrity": "sha512-YrwaA0vEKazPBkn0ipTiMpSajYDSe+KjQfrjhcBMxJt/znbvlHd8Pw/Vamaz5EB4Wfhs3SUR3Z9mwRu/P3s3Yg==", + "dev": true, + "requires": { + "camelcase": "^5.3.1", + "map-obj": "^4.0.0", + "quick-lru": "^4.0.1" + } + }, + "caniuse-api": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/caniuse-api/-/caniuse-api-3.0.0.tgz", + "integrity": "sha512-bsTwuIg/BZZK/vreVTYYbSWoe2F+71P7K5QGEX+pT250DZbfU1MQ5prOKpPR+LL6uWKK3KMwMCAS74QB3Um1uw==", + "dev": true, + "requires": { + "browserslist": "^4.0.0", + "caniuse-lite": "^1.0.0", + "lodash.memoize": "^4.1.2", + "lodash.uniq": "^4.5.0" + } + }, + "caniuse-lite": { + "version": "1.0.30001161", + "resolved": "https://registry.npmjs.org/caniuse-lite/-/caniuse-lite-1.0.30001161.tgz", + "integrity": "sha512-JharrCDxOqPLBULF9/SPa6yMcBRTjZARJ6sc3cuKrPfyIk64JN6kuMINWqA99Xc8uElMFcROliwtz0n9pYej+g==", + "dev": true + }, + "canonical-path": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/canonical-path/-/canonical-path-1.0.0.tgz", + "integrity": "sha512-feylzsbDxi1gPZ1IjystzIQZagYYLvfKrSuygUCgf7z6x790VEzze5QEkdSV1U58RA7Hi0+v6fv4K54atOzATg==", + "dev": true + }, + "capture-exit": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/capture-exit/-/capture-exit-2.0.0.tgz", + "integrity": "sha512-PiT/hQmTonHhl/HFGN+Lx3JJUznrVYJ3+AQsnthneZbvW7x+f08Tk7yLJTLEOUvBTbduLeeBkxEaYXUOUrRq6g==", + "dev": true, + "requires": { + "rsvp": "^4.8.4" + } + }, + "capture-stack-trace": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/capture-stack-trace/-/capture-stack-trace-1.0.1.tgz", + "integrity": "sha512-mYQLZnx5Qt1JgB1WEiMCf2647plpGeQ2NMR/5L0HNZzGQo4fuSPnK+wjfPnKZV0aiJDgzmWqqkV/g7JD+DW0qw==", + "dev": true + }, + "caseless": { + "version": "0.12.0", + "resolved": "https://registry.npmjs.org/caseless/-/caseless-0.12.0.tgz", + "integrity": "sha1-G2gcIf+EAzyCZUMJBolCDRhxUdw=", + "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==", + "requires": { + "ansi-styles": "^3.2.1", + "escape-string-regexp": "^1.0.5", + "supports-color": "^5.3.0" + } + }, + "char-regex": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/char-regex/-/char-regex-1.0.2.tgz", + "integrity": "sha512-kWWXztvZ5SBQV+eRgKFeh8q5sLuZY2+8WUIzlxWVTg+oGwY14qylx1KbKzHd8P6ZYkAg0xyIDU9JMHhyJMZ1jw==", + "dev": true + }, + "chardet": { + "version": "0.7.0", + "resolved": "https://registry.npmjs.org/chardet/-/chardet-0.7.0.tgz", + "integrity": "sha512-mT8iDcrh03qDGRRmoA2hmBJnxpllMR+0/0qlzjqZES6NdiWDcZkCNAk4rPFZ9Q85r27unkiNNg8ZOiwZXBHwcA==", + "dev": true + }, + "check-types": { + "version": "8.0.3", + "resolved": "https://registry.npmjs.org/check-types/-/check-types-8.0.3.tgz", + "integrity": "sha512-YpeKZngUmG65rLudJ4taU7VLkOCTMhNl/u4ctNC56LQS/zJTyNH0Lrtwm1tfTsbLlwvlfsA2d1c8vCf/Kh2KwQ==", + "dev": true + }, + "chevrotain": { + "version": "7.0.1", + "resolved": "https://registry.npmjs.org/chevrotain/-/chevrotain-7.0.1.tgz", + "integrity": "sha512-B/44jrdw5GAzy483LEeVSgXSX0qOYM8lUd3l5+yf6Vl6OQjEUCm2BUiYbHRCIK6xCEvCLAFe1kj8uyV6+zdaVw==", + "dev": true, + "requires": { + "regexp-to-ast": "0.5.0" + } + }, + "chokidar": { + "version": "3.4.3", + "resolved": "https://registry.npmjs.org/chokidar/-/chokidar-3.4.3.tgz", + "integrity": "sha512-DtM3g7juCXQxFVSNPNByEC2+NImtBuxQQvWlHunpJIS5Ocr0lG306cC7FCi7cEA0fzmybPUIl4txBIobk1gGOQ==", + "dev": true, + "requires": { + "anymatch": "~3.1.1", + "braces": "~3.0.2", + "fsevents": "~2.1.2", + "glob-parent": "~5.1.0", + "is-binary-path": "~2.1.0", + "is-glob": "~4.0.1", + "normalize-path": "~3.0.0", + "readdirp": "~3.5.0" + } + }, + "chownr": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/chownr/-/chownr-1.1.4.tgz", + "integrity": "sha512-jJ0bqzaylmJtVnNgzTeSOs8DPavpbYgEr/b0YL8/2GO3xJEhInFmhKMUnEJQjZumK7KXGFhUy89PrsJWlakBVg==", + "dev": true + }, + "chrome-trace-event": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/chrome-trace-event/-/chrome-trace-event-1.0.2.tgz", + "integrity": "sha512-9e/zx1jw7B4CO+c/RXoCsfg/x1AfUBioy4owYH0bJprEYAx5hRFLRhWBqHAG57D0ZM4H7vxbP7bPe0VwhQRYDQ==", + "dev": true, + "requires": { + "tslib": "^1.9.0" + }, + "dependencies": { + "tslib": { + "version": "1.14.1", + "resolved": "https://registry.npmjs.org/tslib/-/tslib-1.14.1.tgz", + "integrity": "sha512-Xni35NKzjgMrwevysHTCArtLDpPvye8zV/0E4EyYn43P7/7qvQwPh9BGkHewbMulVntbigmcT7rdX3BNo9wRJg==", + "dev": true + } + } + }, + "ci-info": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/ci-info/-/ci-info-2.0.0.tgz", + "integrity": "sha512-5tK7EtrZ0N+OLFMthtqOj4fI2Jeb88C4CAZPu25LDVUgXJ0A3Js4PMGqrn0JU1W0Mh1/Z8wZzYPxqUrXeBboCQ==", + "dev": true + }, + "cipher-base": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/cipher-base/-/cipher-base-1.0.4.tgz", + "integrity": "sha512-Kkht5ye6ZGmwv40uUDZztayT2ThLQGfnj/T71N/XzeZeo3nf8foyW7zGTsPYkEya3m5f3cAypH+qe7YOrM1U2Q==", + "dev": true, + "requires": { + "inherits": "^2.0.1", + "safe-buffer": "^5.0.1" + } + }, + "cjs-module-lexer": { + "version": "0.6.0", + "resolved": "https://registry.npmjs.org/cjs-module-lexer/-/cjs-module-lexer-0.6.0.tgz", + "integrity": "sha512-uc2Vix1frTfnuzxxu1Hp4ktSvM3QaI4oXl4ZUqL1wjTu/BGki9TrCWoqLTg/drR1KwAEarXuRFCG2Svr1GxPFw==", + "dev": true + }, + "class-utils": { + "version": "0.3.6", + "resolved": "https://registry.npmjs.org/class-utils/-/class-utils-0.3.6.tgz", + "integrity": "sha512-qOhPa/Fj7s6TY8H8esGu5QNpMMQxz79h+urzrNYN6mn+9BnxlDGf5QZ+XeCDsxSjPqsSR56XOZOJmpeurnLMeg==", + "dev": true, + "requires": { + "arr-union": "^3.1.0", + "define-property": "^0.2.5", + "isobject": "^3.0.0", + "static-extend": "^0.1.1" + }, + "dependencies": { + "define-property": { + "version": "0.2.5", + "resolved": "https://registry.npmjs.org/define-property/-/define-property-0.2.5.tgz", + "integrity": "sha1-w1se+RjsPJkPmlvFe+BKrOxcgRY=", + "dev": true, + "requires": { + "is-descriptor": "^0.1.0" + } + } + } + }, + "clean-css": { + "version": "4.2.3", + "resolved": "https://registry.npmjs.org/clean-css/-/clean-css-4.2.3.tgz", + "integrity": "sha512-VcMWDN54ZN/DS+g58HYL5/n4Zrqe8vHJpGA8KdgUXFU4fuP/aHNw8eld9SyEIyabIMJX/0RaY/fplOo5hYLSFA==", + "dev": true, + "requires": { + "source-map": "~0.6.0" + }, + "dependencies": { + "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 + } + } + }, + "clean-stack": { + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/clean-stack/-/clean-stack-2.2.0.tgz", + "integrity": "sha512-4diC9HaTE+KRAMWhDhrGOECgWZxoevMc5TlkObMqNSsVU62PYzXZ/SMTjzyGAFF1YusgxGcSWTEXBhp0CPwQ1A==", + "dev": true + }, + "cli-cursor": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/cli-cursor/-/cli-cursor-3.1.0.tgz", + "integrity": "sha512-I/zHAwsKf9FqGoXM4WWRACob9+SNukZTd94DWF57E4toouRulbCxcUh6RKUEOQlYTHJnzkPMySvPNaaSLNfLZw==", + "dev": true, + "requires": { + "restore-cursor": "^3.1.0" + } + }, + "cli-spinners": { + "version": "2.5.0", + "resolved": "https://registry.npmjs.org/cli-spinners/-/cli-spinners-2.5.0.tgz", + "integrity": "sha512-PC+AmIuK04E6aeSs/pUccSujsTzBhu4HzC2dL+CfJB/Jcc2qTRbEwZQDfIUpt2Xl8BodYBEq8w4fc0kU2I9DjQ==", + "dev": true + }, + "cli-table": { + "version": "0.3.1", + "resolved": "https://registry.npmjs.org/cli-table/-/cli-table-0.3.1.tgz", + "integrity": "sha1-9TsFJmqLGguTSz0IIebi3FkUriM=", + "dev": true, + "requires": { + "colors": "1.0.3" + }, + "dependencies": { + "colors": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/colors/-/colors-1.0.3.tgz", + "integrity": "sha1-BDP0TYCWgP3rYO0mDxsMJi6CpAs=", + "dev": true + } + } + }, + "cli-truncate": { + "version": "0.2.1", + "resolved": "https://registry.npmjs.org/cli-truncate/-/cli-truncate-0.2.1.tgz", + "integrity": "sha1-nxXPuwcFAFNpIWxiasfQWrkN1XQ=", + "dev": true, + "requires": { + "slice-ansi": "0.0.4", + "string-width": "^1.0.1" + }, + "dependencies": { + "ansi-regex": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-2.1.1.tgz", + "integrity": "sha1-w7M6te42DYbg5ijwRorn7yfWVN8=", + "dev": true + }, + "is-fullwidth-code-point": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-1.0.0.tgz", + "integrity": "sha1-754xOG8DGn8NZDr4L95QxFfvAMs=", + "dev": true, + "requires": { + "number-is-nan": "^1.0.0" + } + }, + "slice-ansi": { + "version": "0.0.4", + "resolved": "https://registry.npmjs.org/slice-ansi/-/slice-ansi-0.0.4.tgz", + "integrity": "sha1-7b+JA/ZvfOL46v1s7tZeJkyDGzU=", + "dev": true + }, + "string-width": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/string-width/-/string-width-1.0.2.tgz", + "integrity": "sha1-EYvfW4zcUaKn5w0hHgfisLmxB9M=", + "dev": true, + "requires": { + "code-point-at": "^1.0.0", + "is-fullwidth-code-point": "^1.0.0", + "strip-ansi": "^3.0.0" + } + }, + "strip-ansi": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-3.0.1.tgz", + "integrity": "sha1-ajhfuIU9lS1f8F0Oiq+UJ43GPc8=", + "dev": true, + "requires": { + "ansi-regex": "^2.0.0" + } + } + } + }, + "cli-width": { + "version": "2.2.1", + "resolved": "https://registry.npmjs.org/cli-width/-/cli-width-2.2.1.tgz", + "integrity": "sha512-GRMWDxpOB6Dgk2E5Uo+3eEBvtOOlimMmpbFiKuLFnQzYDavtLFY3K5ona41jgN/WdRZtG7utuVSVTL4HbZHGkw==", + "dev": true + }, + "cliui": { + "version": "6.0.0", + "resolved": "https://registry.npmjs.org/cliui/-/cliui-6.0.0.tgz", + "integrity": "sha512-t6wbgtoCXvAzst7QgXxJYqPt0usEfbgQdftEPbLL/cvv6HPE5VgvqCuAIDR0NgU52ds6rFwqrgakNLrHEjCbrQ==", + "requires": { + "string-width": "^4.2.0", + "strip-ansi": "^6.0.0", + "wrap-ansi": "^6.2.0" + } + }, + "clone": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/clone/-/clone-1.0.4.tgz", + "integrity": "sha1-2jCcwmPfFZlMaIypAheco8fNfH4=", + "dev": true + }, + "clone-buffer": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/clone-buffer/-/clone-buffer-1.0.0.tgz", + "integrity": "sha1-4+JbIHrE5wGvch4staFnksrD3Fg=", + "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" + } + }, + "clone-stats": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/clone-stats/-/clone-stats-1.0.0.tgz", + "integrity": "sha1-s3gt/4u1R04Yuba/D9/ngvh3doA=", + "dev": true + }, + "cloneable-readable": { + "version": "1.1.3", + "resolved": "https://registry.npmjs.org/cloneable-readable/-/cloneable-readable-1.1.3.tgz", + "integrity": "sha512-2EF8zTQOxYq70Y4XKtorQupqF0m49MBz2/yf5Bj+MHjvpG3Hy7sImifnqD6UA+TKYxeSV+u6qqQPawN5UvnpKQ==", + "dev": true, + "requires": { + "inherits": "^2.0.1", + "process-nextick-args": "^2.0.0", + "readable-stream": "^2.3.5" + } + }, + "co": { + "version": "4.6.0", + "resolved": "https://registry.npmjs.org/co/-/co-4.6.0.tgz", + "integrity": "sha1-bqa989hTrlTMuOR7+gvz+QMfsYQ=", + "dev": true + }, + "coa": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/coa/-/coa-2.0.2.tgz", + "integrity": "sha512-q5/jG+YQnSy4nRTV4F7lPepBJZ8qBNJJDBuJdoejDyLXgmL7IEo+Le2JDZudFTFt7mrCqIRaSjws4ygRCTCAXA==", + "dev": true, + "requires": { + "@types/q": "^1.5.1", + "chalk": "^2.4.1", + "q": "^1.1.2" + } + }, + "code-point-at": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/code-point-at/-/code-point-at-1.1.0.tgz", + "integrity": "sha1-DQcLTQQ6W+ozovGkDi7bPZpMz3c=", + "dev": true + }, + "codelyzer": { + "version": "5.2.2", + "resolved": "https://registry.npmjs.org/codelyzer/-/codelyzer-5.2.2.tgz", + "integrity": "sha512-jB4FZ1Sx7kZhvZVdf+N2BaKTdrrNZOL0Bj10RRfrhHrb3zEvXjJvvq298JPMJAiyiCS/v4zs1QlGU0ip7xGqeA==", + "dev": true, + "requires": { + "app-root-path": "^2.2.1", + "aria-query": "^3.0.0", + "axobject-query": "2.0.2", + "css-selector-tokenizer": "^0.7.1", + "cssauron": "^1.4.0", + "damerau-levenshtein": "^1.0.4", + "semver-dsl": "^1.0.1", + "source-map": "^0.5.7", + "sprintf-js": "^1.1.2" + } + }, + "collect-v8-coverage": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/collect-v8-coverage/-/collect-v8-coverage-1.0.1.tgz", + "integrity": "sha512-iBPtljfCNcTKNAto0KEtDfZ3qzjJvqE3aTGZsbhjSBlorqpXJlaWWtPO35D+ZImoC3KWejX64o+yPGxhWSTzfg==", + "dev": true + }, + "collection-visit": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/collection-visit/-/collection-visit-1.0.0.tgz", + "integrity": "sha1-S8A3PBZLwykbTTaMgpzxqApZ3KA=", + "dev": true, + "requires": { + "map-visit": "^1.0.0", + "object-visit": "^1.0.0" + } + }, + "color": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/color/-/color-3.0.0.tgz", + "integrity": "sha512-jCpd5+s0s0t7p3pHQKpnJ0TpQKKdleP71LWcA0aqiljpiuAkOSUFN/dyH8ZwF0hRmFlrIuRhufds1QyEP9EB+w==", + "dev": true, + "requires": { + "color-convert": "^1.9.1", + "color-string": "^1.5.2" + } + }, + "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==", + "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": "sha1-p9BVi9icQveV3UIyj3QIMcpTvCU=" + }, + "color-string": { + "version": "1.5.4", + "resolved": "https://registry.npmjs.org/color-string/-/color-string-1.5.4.tgz", + "integrity": "sha512-57yF5yt8Xa3czSEW1jfQDE79Idk0+AkN/4KWad6tbdxUmAs3MvjxlWSWD4deYytcRfoZ9nhKyFl1kj5tBvidbw==", + "dev": true, + "requires": { + "color-name": "^1.0.0", + "simple-swizzle": "^0.2.2" + } + }, + "colorette": { + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/colorette/-/colorette-1.2.1.tgz", + "integrity": "sha512-puCDz0CzydiSYOrnXpz/PKd69zRrribezjtE9yd4zvytoRc8+RY/KJPvtPFKZS3E3wP6neGyMe0vOTlHO5L3Pw==", + "dev": true + }, + "colornames": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/colornames/-/colornames-1.1.1.tgz", + "integrity": "sha1-+IiQMGhcfE/54qVZ9Qd+t2qBb5Y=", + "dev": true + }, + "colors": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/colors/-/colors-1.4.0.tgz", + "integrity": "sha512-a+UqTh4kgZg/SlGvfbzDHpgRu7AAQOmmqRHJnxhRZICKFUT91brVhNNt58CMWU9PsBbv3PDCZUHbVxuDiH2mtA==", + "dev": true + }, + "colorspace": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/colorspace/-/colorspace-1.1.2.tgz", + "integrity": "sha512-vt+OoIP2d76xLhjwbBaucYlNSpPsrJWPlBTtwCpQKIu6/CSMutyzX93O/Do0qzpH3YoHEes8YEFXyZ797rEhzQ==", + "dev": true, + "requires": { + "color": "3.0.x", + "text-hex": "1.0.x" + } + }, + "combined-stream": { + "version": "1.0.8", + "resolved": "https://registry.npmjs.org/combined-stream/-/combined-stream-1.0.8.tgz", + "integrity": "sha512-FQN4MRfuJeHf7cBbBMJFXhKSDq+2kAArBlmRBvcvFE5BB1HZKXtSFASDhdlz9zOYwxh8lDdnvmMOe/+5cdoEdg==", + "dev": true, + "requires": { + "delayed-stream": "~1.0.0" + } + }, + "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 + }, + "common-tags": { + "version": "1.8.0", + "resolved": "https://registry.npmjs.org/common-tags/-/common-tags-1.8.0.tgz", + "integrity": "sha512-6P6g0uetGpW/sdyUy/iQQCbFF0kWVMSIVSyYz7Zgjcgh8mgw8PQzDNZeyZ5DQ2gM7LBoZPHmnjz8rUthkBG5tw==", + "dev": true + }, + "commondir": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/commondir/-/commondir-1.0.1.tgz", + "integrity": "sha1-3dgA2gxmEnOTzKWVDqloo6rxJTs=", + "dev": true + }, + "compare-versions": { + "version": "3.6.0", + "resolved": "https://registry.npmjs.org/compare-versions/-/compare-versions-3.6.0.tgz", + "integrity": "sha512-W6Af2Iw1z4CB7q4uU4hv646dW9GQuBM+YpC0UvUCWSD8w90SJjp+ujJuXaEMtAXBtSqGfMPuFOVn4/+FlaqfBA==", + "dev": true + }, + "component-bind": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/component-bind/-/component-bind-1.0.0.tgz", + "integrity": "sha1-AMYIq33Nk4l8AAllGx06jh5zu9E=", + "dev": true + }, + "component-emitter": { + "version": "1.3.0", + "resolved": "https://registry.npmjs.org/component-emitter/-/component-emitter-1.3.0.tgz", + "integrity": "sha512-Rd3se6QB+sO1TwqZjscQrurpEPIfO0/yYnSin6Q/rD3mOutHvUrCAhJub3r90uNb+SESBuE0QYoB90YdfatsRg==", + "dev": true + }, + "component-inherit": { + "version": "0.0.3", + "resolved": "https://registry.npmjs.org/component-inherit/-/component-inherit-0.0.3.tgz", + "integrity": "sha1-ZF/ErfWLcrZJ1crmUTVhnbJv8UM=", + "dev": true + }, + "compressible": { + "version": "2.0.18", + "resolved": "https://registry.npmjs.org/compressible/-/compressible-2.0.18.tgz", + "integrity": "sha512-AF3r7P5dWxL8MxyITRMlORQNaOA2IkAFaTr4k7BUumjPtRpGDTZpl0Pb1XCO6JeDCBdp126Cgs9sMxqSjgYyRg==", + "dev": true, + "requires": { + "mime-db": ">= 1.43.0 < 2" + } + }, + "compression": { + "version": "1.7.4", + "resolved": "https://registry.npmjs.org/compression/-/compression-1.7.4.tgz", + "integrity": "sha512-jaSIDzP9pZVS4ZfQ+TzvtiWhdpFhE2RDHz8QJkpX9SIpLq88VueF5jJw6t+6CUQcAoA6t+x89MLrWAqpfDE8iQ==", + "dev": true, + "requires": { + "accepts": "~1.3.5", + "bytes": "3.0.0", + "compressible": "~2.0.16", + "debug": "2.6.9", + "on-headers": "~1.0.2", + "safe-buffer": "5.1.2", + "vary": "~1.1.2" + }, + "dependencies": { + "bytes": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/bytes/-/bytes-3.0.0.tgz", + "integrity": "sha1-0ygVQE1olpn4Wk6k+odV3ROpYEg=", + "dev": true + }, + "debug": { + "version": "2.6.9", + "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", + "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", + "dev": true, + "requires": { + "ms": "2.0.0" + } + }, + "ms": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", + "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=", + "dev": true + } + } + }, + "concat-map": { + "version": "0.0.1", + "resolved": "https://registry.npmjs.org/concat-map/-/concat-map-0.0.1.tgz", + "integrity": "sha1-2Klr13/Wjfd5OnMDajug1UBdR3s=" + }, + "concat-stream": { + "version": "1.6.2", + "resolved": "https://registry.npmjs.org/concat-stream/-/concat-stream-1.6.2.tgz", + "integrity": "sha512-27HBghJxjiZtIk3Ycvn/4kbJk/1uZuJFfuPEns6LaEvpvG1f0hTea8lilrouyo9mVc2GWdcEZ8OLoGmSADlrCw==", + "dev": true, + "requires": { + "buffer-from": "^1.0.0", + "inherits": "^2.0.3", + "readable-stream": "^2.2.2", + "typedarray": "^0.0.6" + } + }, + "conf": { + "version": "6.2.4", + "resolved": "https://registry.npmjs.org/conf/-/conf-6.2.4.tgz", + "integrity": "sha512-GjgyPRLo1qK1LR9RWAdUagqo+DP18f5HWCFk4va7GS+wpxQTOzfuKTwKOvGW2c01/YXNicAyyoyuSddmdkBzZQ==", + "dev": true, + "requires": { + "ajv": "^6.10.2", + "debounce-fn": "^3.0.1", + "dot-prop": "^5.0.0", + "env-paths": "^2.2.0", + "json-schema-typed": "^7.0.1", + "make-dir": "^3.0.0", + "onetime": "^5.1.0", + "pkg-up": "^3.0.1", + "semver": "^6.2.0", + "write-file-atomic": "^3.0.0" + }, + "dependencies": { + "semver": { + "version": "6.3.0", + "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.0.tgz", + "integrity": "sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw==", + "dev": true + } + } + }, + "connect": { + "version": "3.6.6", + "resolved": "https://registry.npmjs.org/connect/-/connect-3.6.6.tgz", + "integrity": "sha1-Ce/2xVr3I24TcTWnJXSFi2eG9SQ=", + "dev": true, + "requires": { + "debug": "2.6.9", + "finalhandler": "1.1.0", + "parseurl": "~1.3.2", + "utils-merge": "1.0.1" + }, + "dependencies": { + "debug": { + "version": "2.6.9", + "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", + "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", + "dev": true, + "requires": { + "ms": "2.0.0" + } + }, + "ms": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", + "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=", + "dev": true + } + } + }, + "connect-history-api-fallback": { + "version": "1.6.0", + "resolved": "https://registry.npmjs.org/connect-history-api-fallback/-/connect-history-api-fallback-1.6.0.tgz", + "integrity": "sha512-e54B99q/OUoH64zYYRf3HBP5z24G38h5D3qXu23JGRoigpX5Ss4r9ZnDk3g0Z8uQC2x2lPaJ+UlWBc1ZWBWdLg==", + "dev": true + }, + "console-browserify": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/console-browserify/-/console-browserify-1.2.0.tgz", + "integrity": "sha512-ZMkYO/LkF17QvCPqM0gxw8yUzigAOZOSWSHg91FH6orS7vcEj5dVZTidN2fQ14yBSdg97RqhSNwLUXInd52OTA==", + "dev": true + }, + "constants-browserify": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/constants-browserify/-/constants-browserify-1.0.0.tgz", + "integrity": "sha1-wguW2MYXdIqvHBYCF2DNJ/y4y3U=", + "dev": true + }, + "content-disposition": { + "version": "0.5.3", + "resolved": "https://registry.npmjs.org/content-disposition/-/content-disposition-0.5.3.tgz", + "integrity": "sha512-ExO0774ikEObIAEV9kDo50o+79VCUdEB6n6lzKgGwupcVeRlhrj3qGAfwq8G6uBJjkqLrhT0qEYFcWng8z1z0g==", + "dev": true, + "requires": { + "safe-buffer": "5.1.2" + } + }, + "content-type": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/content-type/-/content-type-1.0.4.tgz", + "integrity": "sha512-hIP3EEPs8tB9AT1L+NUqtwOAps4mk2Zob89MWXMHjHWg9milF/j4osnnQLXBCBFBk/tvIG/tUc9mOUJiPBhPXA==", + "dev": true + }, + "convert-source-map": { + "version": "1.7.0", + "resolved": "https://registry.npmjs.org/convert-source-map/-/convert-source-map-1.7.0.tgz", + "integrity": "sha512-4FJkXzKXEDB1snCFZlLP4gpC3JILicCpGbzG9f9G7tGqGCzETQ2hWPrcinA9oU4wtf2biUaEH5065UnMeR33oA==", + "requires": { + "safe-buffer": "~5.1.1" + } + }, + "cookie": { + "version": "0.3.1", + "resolved": "https://registry.npmjs.org/cookie/-/cookie-0.3.1.tgz", + "integrity": "sha1-5+Ch+e9DtMi6klxcWpboBtFoc7s=", + "dev": true + }, + "cookie-signature": { + "version": "1.0.6", + "resolved": "https://registry.npmjs.org/cookie-signature/-/cookie-signature-1.0.6.tgz", + "integrity": "sha1-4wOogrNCzD7oylE6eZmXNNqzriw=", + "dev": true + }, + "copy-concurrently": { + "version": "1.0.5", + "resolved": "https://registry.npmjs.org/copy-concurrently/-/copy-concurrently-1.0.5.tgz", + "integrity": "sha512-f2domd9fsVDFtaFcbaRZuYXwtdmnzqbADSwhSWYxYB/Q8zsdUUFMXVRwXGDMWmbEzAn1kdRrtI1T/KTFOL4X2A==", + "dev": true, + "requires": { + "aproba": "^1.1.1", + "fs-write-stream-atomic": "^1.0.8", + "iferr": "^0.1.5", + "mkdirp": "^0.5.1", + "rimraf": "^2.5.4", + "run-queue": "^1.0.0" + }, + "dependencies": { + "glob": { + "version": "7.1.6", + "resolved": "https://registry.npmjs.org/glob/-/glob-7.1.6.tgz", + "integrity": "sha512-LwaxwyZ72Lk7vZINtNNrywX0ZuLyStrdDtabefZKAY5ZGJhVtgdznluResxNmPitE0SAO+O26sWTHeKSI2wMBA==", + "dev": true, + "requires": { + "fs.realpath": "^1.0.0", + "inflight": "^1.0.4", + "inherits": "2", + "minimatch": "^3.0.4", + "once": "^1.3.0", + "path-is-absolute": "^1.0.0" + } + }, + "rimraf": { + "version": "2.7.1", + "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-2.7.1.tgz", + "integrity": "sha512-uWjbaKIK3T1OSVptzX7Nl6PvQ3qAGtKEtVRjRuazjfL3Bx5eI409VZSqgND+4UNnmzLVdPj9FqFJNPqBZFve4w==", + "dev": true, + "requires": { + "glob": "^7.1.3" + } + } + } + }, + "copy-descriptor": { + "version": "0.1.1", + "resolved": "https://registry.npmjs.org/copy-descriptor/-/copy-descriptor-0.1.1.tgz", + "integrity": "sha1-Z29us8OZl8LuGsOpJP1hJHSPV40=", + "dev": true + }, + "copy-webpack-plugin": { + "version": "6.0.2", + "resolved": "https://registry.npmjs.org/copy-webpack-plugin/-/copy-webpack-plugin-6.0.2.tgz", + "integrity": "sha512-9Gm8X0c6eXlKnmltMPFCBeGOKjtcRIyTt4VaO3k1TkNgVTe5Ov2lYsYVuyLp0kp8DItO3apewflM+1GYgh6V2Q==", + "dev": true, + "requires": { + "cacache": "^15.0.4", + "fast-glob": "^3.2.2", + "find-cache-dir": "^3.3.1", + "glob-parent": "^5.1.1", + "globby": "^11.0.1", + "loader-utils": "^2.0.0", + "normalize-path": "^3.0.0", + "p-limit": "^2.3.0", + "schema-utils": "^2.7.0", + "serialize-javascript": "^3.1.0", + "webpack-sources": "^1.4.3" + }, + "dependencies": { + "cacache": { + "version": "15.0.5", + "resolved": "https://registry.npmjs.org/cacache/-/cacache-15.0.5.tgz", + "integrity": "sha512-lloiL22n7sOjEEXdL8NAjTgv9a1u43xICE9/203qonkZUCj5X1UEWIdf2/Y0d6QcCtMzbKQyhrcDbdvlZTs/+A==", + "dev": true, + "requires": { + "@npmcli/move-file": "^1.0.1", + "chownr": "^2.0.0", + "fs-minipass": "^2.0.0", + "glob": "^7.1.4", + "infer-owner": "^1.0.4", + "lru-cache": "^6.0.0", + "minipass": "^3.1.1", + "minipass-collect": "^1.0.2", + "minipass-flush": "^1.0.5", + "minipass-pipeline": "^1.2.2", + "mkdirp": "^1.0.3", + "p-map": "^4.0.0", + "promise-inflight": "^1.0.1", + "rimraf": "^3.0.2", + "ssri": "^8.0.0", + "tar": "^6.0.2", + "unique-filename": "^1.1.1" + } + }, + "chownr": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/chownr/-/chownr-2.0.0.tgz", + "integrity": "sha512-bIomtDF5KGpdogkLd9VspvFzk9KfpyyGlS8YFVZl7TGPBHL5snIOnxeshwVgPteQ9b4Eydl+pVbIyE1DcvCWgQ==", + "dev": true + }, + "fs-minipass": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/fs-minipass/-/fs-minipass-2.1.0.tgz", + "integrity": "sha512-V/JgOLFCS+R6Vcq0slCuaeWEdNC3ouDlJMNIsacH2VtALiu9mV4LPrHc5cDl8k5aw6J8jwgWWpiTo5RYhmIzvg==", + "dev": true, + "requires": { + "minipass": "^3.0.0" + } + }, + "glob": { + "version": "7.1.6", + "resolved": "https://registry.npmjs.org/glob/-/glob-7.1.6.tgz", + "integrity": "sha512-LwaxwyZ72Lk7vZINtNNrywX0ZuLyStrdDtabefZKAY5ZGJhVtgdznluResxNmPitE0SAO+O26sWTHeKSI2wMBA==", + "dev": true, + "requires": { + "fs.realpath": "^1.0.0", + "inflight": "^1.0.4", + "inherits": "2", + "minimatch": "^3.0.4", + "once": "^1.3.0", + "path-is-absolute": "^1.0.0" + } + }, + "minipass": { + "version": "3.1.3", + "resolved": "https://registry.npmjs.org/minipass/-/minipass-3.1.3.tgz", + "integrity": "sha512-Mgd2GdMVzY+x3IJ+oHnVM+KG3lA5c8tnabyJKmHSaG2kAGpudxuOf8ToDkhumF7UzME7DecbQE9uOZhNm7PuJg==", + "dev": true, + "requires": { + "yallist": "^4.0.0" + } + }, + "minizlib": { + "version": "2.1.2", + "resolved": "https://registry.npmjs.org/minizlib/-/minizlib-2.1.2.tgz", + "integrity": "sha512-bAxsR8BVfj60DWXHE3u30oHzfl4G7khkSuPW+qvpd7jFRHm7dLxOjUk1EHACJ/hxLY8phGJ0YhYHZo7jil7Qdg==", + "dev": true, + "requires": { + "minipass": "^3.0.0", + "yallist": "^4.0.0" + } + }, + "mkdirp": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/mkdirp/-/mkdirp-1.0.4.tgz", + "integrity": "sha512-vVqVZQyf3WLx2Shd0qJ9xuvqgAyKPLAiqITEtqW0oIUjzo3PePDd6fW9iFz30ef7Ysp/oiWqbhszeGWW2T6Gzw==", + "dev": true + }, + "ssri": { + "version": "8.0.0", + "resolved": "https://registry.npmjs.org/ssri/-/ssri-8.0.0.tgz", + "integrity": "sha512-aq/pz989nxVYwn16Tsbj1TqFpD5LLrQxHf5zaHuieFV+R0Bbr4y8qUsOA45hXT/N4/9UNXTarBjnjVmjSOVaAA==", + "dev": true, + "requires": { + "minipass": "^3.1.1" + } + }, + "tar": { + "version": "6.0.5", + "resolved": "https://registry.npmjs.org/tar/-/tar-6.0.5.tgz", + "integrity": "sha512-0b4HOimQHj9nXNEAA7zWwMM91Zhhba3pspja6sQbgTpynOJf+bkjBnfybNYzbpLbnwXnbyB4LOREvlyXLkCHSg==", + "dev": true, + "requires": { + "chownr": "^2.0.0", + "fs-minipass": "^2.0.0", + "minipass": "^3.0.0", + "minizlib": "^2.1.1", + "mkdirp": "^1.0.3", + "yallist": "^4.0.0" + } + } + } + }, + "core-js": { + "version": "2.6.11", + "resolved": "https://registry.npmjs.org/core-js/-/core-js-2.6.11.tgz", + "integrity": "sha512-5wjnpaT/3dV+XB4borEsnAYQchn00XSgTAWKDkEqv+K8KevjbzmofK6hfJ9TZIlpj2N0xQpazy7PiRQiWHqzWg==", + "dev": true + }, + "core-js-compat": { + "version": "3.7.0", + "resolved": "https://registry.npmjs.org/core-js-compat/-/core-js-compat-3.7.0.tgz", + "integrity": "sha512-V8yBI3+ZLDVomoWICO6kq/CD28Y4r1M7CWeO4AGpMdMfseu8bkSubBmUPySMGKRTS+su4XQ07zUkAsiu9FCWTg==", + "dev": true, + "requires": { + "browserslist": "^4.14.6", + "semver": "7.0.0" + }, + "dependencies": { + "semver": { + "version": "7.0.0", + "resolved": "https://registry.npmjs.org/semver/-/semver-7.0.0.tgz", + "integrity": "sha512-+GB6zVA9LWh6zovYQLALHwv5rb2PHGlJi3lfiqIHxR0uuwCgefcOJc59v9fv1w8GbStwxuuqqAjI9NMAOOgq1A==", + "dev": true + } + } + }, + "core-util-is": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/core-util-is/-/core-util-is-1.0.2.tgz", + "integrity": "sha1-tf1UIgqivFq1eqtxQMlAdUUDwac=", + "dev": true + }, + "cosmiconfig": { + "version": "6.0.0", + "resolved": "https://registry.npmjs.org/cosmiconfig/-/cosmiconfig-6.0.0.tgz", + "integrity": "sha512-xb3ZL6+L8b9JLLCx3ZdoZy4+2ECphCMo2PwqgP1tlfVq6M6YReyzBJtvWWtbDSpNr9hn96pkCiZqUcFEc+54Qg==", + "dev": true, + "requires": { + "@types/parse-json": "^4.0.0", + "import-fresh": "^3.1.0", + "parse-json": "^5.0.0", + "path-type": "^4.0.0", + "yaml": "^1.7.2" + }, + "dependencies": { + "parse-json": { + "version": "5.1.0", + "resolved": "https://registry.npmjs.org/parse-json/-/parse-json-5.1.0.tgz", + "integrity": "sha512-+mi/lmVVNKFNVyLXV31ERiy2CY5E1/F6QtJFEzoChPRwwngMNXRDQ9GJ5WdE2Z2P4AujsOi0/+2qHID68KwfIQ==", + "dev": true, + "requires": { + "@babel/code-frame": "^7.0.0", + "error-ex": "^1.3.1", + "json-parse-even-better-errors": "^2.3.0", + "lines-and-columns": "^1.1.6" + } + }, + "path-type": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/path-type/-/path-type-4.0.0.tgz", + "integrity": "sha512-gDKb8aZMDeD/tZWs9P6+q0J9Mwkdl6xMV8TjnGP3qJVJ06bdMgkbBlLU8IdfOsIsFz2BW1rNVT3XuNEl8zPAvw==", + "dev": true + } + } + }, + "create-ecdh": { + "version": "4.0.4", + "resolved": "https://registry.npmjs.org/create-ecdh/-/create-ecdh-4.0.4.tgz", + "integrity": "sha512-mf+TCx8wWc9VpuxfP2ht0iSISLZnt0JgWlrOKZiNqyUZWnjIaCIVNQArMHnCZKfEYRg6IM7A+NeJoN8gf/Ws0A==", + "dev": true, + "requires": { + "bn.js": "^4.1.0", + "elliptic": "^6.5.3" + }, + "dependencies": { + "bn.js": { + "version": "4.11.9", + "resolved": "https://registry.npmjs.org/bn.js/-/bn.js-4.11.9.tgz", + "integrity": "sha512-E6QoYqCKZfgatHTdHzs1RRKP7ip4vvm+EyRUeE2RF0NblwVvb0p6jSVeNTOFxPn26QXN2o6SMfNxKp6kU8zQaw==", + "dev": true + } + } + }, + "create-error-class": { + "version": "3.0.2", + "resolved": "https://registry.npmjs.org/create-error-class/-/create-error-class-3.0.2.tgz", + "integrity": "sha1-Br56vvlHo/FKMP1hBnHUAbyot7Y=", + "dev": true, + "requires": { + "capture-stack-trace": "^1.0.0" + } + }, + "create-hash": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/create-hash/-/create-hash-1.2.0.tgz", + "integrity": "sha512-z00bCGNHDG8mHAkP7CtT1qVu+bFQUPjYq/4Iv3C3kWjTFV10zIjfSoeqXo9Asws8gwSHDGj/hl2u4OGIjapeCg==", + "dev": true, + "requires": { + "cipher-base": "^1.0.1", + "inherits": "^2.0.1", + "md5.js": "^1.3.4", + "ripemd160": "^2.0.1", + "sha.js": "^2.4.0" + } + }, + "create-hmac": { + "version": "1.1.7", + "resolved": "https://registry.npmjs.org/create-hmac/-/create-hmac-1.1.7.tgz", + "integrity": "sha512-MJG9liiZ+ogc4TzUwuvbER1JRdgvUFSB5+VR/g5h82fGaIRWMWddtKBHi7/sVhfjQZ6SehlyhvQYrcYkaUIpLg==", + "dev": true, + "requires": { + "cipher-base": "^1.0.3", + "create-hash": "^1.1.0", + "inherits": "^2.0.1", + "ripemd160": "^2.0.0", + "safe-buffer": "^5.0.1", + "sha.js": "^2.4.8" + } + }, + "cross-spawn": { + "version": "6.0.5", + "resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-6.0.5.tgz", + "integrity": "sha512-eTVLrBSt7fjbDygz805pMnstIs2VTBNkRm0qxZd+M7A5XDdxVRWO5MxGBXZhjY4cqLYLdtrGqRf8mBPmzwSpWQ==", + "dev": true, + "requires": { + "nice-try": "^1.0.4", + "path-key": "^2.0.1", + "semver": "^5.5.0", + "shebang-command": "^1.2.0", + "which": "^1.2.9" + } + }, + "crypto-browserify": { + "version": "3.12.0", + "resolved": "https://registry.npmjs.org/crypto-browserify/-/crypto-browserify-3.12.0.tgz", + "integrity": "sha512-fz4spIh+znjO2VjL+IdhEpRJ3YN6sMzITSBijk6FK2UvTqruSQW+/cCZTSNsMiZNvUeq0CqurF+dAbyiGOY6Wg==", + "dev": true, + "requires": { + "browserify-cipher": "^1.0.0", + "browserify-sign": "^4.0.0", + "create-ecdh": "^4.0.0", + "create-hash": "^1.1.0", + "create-hmac": "^1.1.0", + "diffie-hellman": "^5.0.0", + "inherits": "^2.0.1", + "pbkdf2": "^3.0.3", + "public-encrypt": "^4.0.0", + "randombytes": "^2.0.0", + "randomfill": "^1.0.3" + } + }, + "crypto-random-string": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/crypto-random-string/-/crypto-random-string-1.0.0.tgz", + "integrity": "sha1-ojD2T1aDEOFJgAmUB5DsmVRbyn4=", + "dev": true + }, + "css-color-names": { + "version": "0.0.4", + "resolved": "https://registry.npmjs.org/css-color-names/-/css-color-names-0.0.4.tgz", + "integrity": "sha1-gIrcLnnPhHOAabZGyyDsJ762KeA=", + "dev": true + }, + "css-declaration-sorter": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/css-declaration-sorter/-/css-declaration-sorter-4.0.1.tgz", + "integrity": "sha512-BcxQSKTSEEQUftYpBVnsH4SF05NTuBokb19/sBt6asXGKZ/6VP7PLG1CBCkFDYOnhXhPh0jMhO6xZ71oYHXHBA==", + "dev": true, + "requires": { + "postcss": "^7.0.1", + "timsort": "^0.3.0" + } + }, + "css-loader": { + "version": "3.6.0", + "resolved": "https://registry.npmjs.org/css-loader/-/css-loader-3.6.0.tgz", + "integrity": "sha512-M5lSukoWi1If8dhQAUCvj4H8vUt3vOnwbQBH9DdTm/s4Ym2B/3dPMtYZeJmq7Q3S3Pa+I94DcZ7pc9bP14cWIQ==", + "dev": true, + "requires": { + "camelcase": "^5.3.1", + "cssesc": "^3.0.0", + "icss-utils": "^4.1.1", + "loader-utils": "^1.2.3", + "normalize-path": "^3.0.0", + "postcss": "^7.0.32", + "postcss-modules-extract-imports": "^2.0.0", + "postcss-modules-local-by-default": "^3.0.2", + "postcss-modules-scope": "^2.2.0", + "postcss-modules-values": "^3.0.0", + "postcss-value-parser": "^4.1.0", + "schema-utils": "^2.7.0", + "semver": "^6.3.0" + }, + "dependencies": { + "json5": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/json5/-/json5-1.0.1.tgz", + "integrity": "sha512-aKS4WQjPenRxiQsC93MNfjx+nbF4PAdYzmd/1JIj8HYzqfbu86beTuNgXDzPknWk0n0uARlyewZo4s++ES36Ow==", + "dev": true, + "requires": { + "minimist": "^1.2.0" + } + }, + "loader-utils": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/loader-utils/-/loader-utils-1.4.0.tgz", + "integrity": "sha512-qH0WSMBtn/oHuwjy/NucEgbx5dbxxnxup9s4PVXJUDHZBQY+s0NWA9rJf53RBnQZxfch7euUui7hpoAPvALZdA==", + "dev": true, + "requires": { + "big.js": "^5.2.2", + "emojis-list": "^3.0.0", + "json5": "^1.0.1" + } + }, + "semver": { + "version": "6.3.0", + "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.0.tgz", + "integrity": "sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw==", + "dev": true + } + } + }, + "css-select": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/css-select/-/css-select-1.2.0.tgz", + "integrity": "sha1-KzoRBTnFNV8c2NMUYj6HCxIeyFg=", + "dev": true, + "requires": { + "boolbase": "~1.0.0", + "css-what": "2.1", + "domutils": "1.5.1", + "nth-check": "~1.0.1" + }, + "dependencies": { + "dom-serializer": { + "version": "0.2.2", + "resolved": "https://registry.npmjs.org/dom-serializer/-/dom-serializer-0.2.2.tgz", + "integrity": "sha512-2/xPb3ORsQ42nHYiSunXkDjPLBaEj/xTwUO4B7XCZQTRk7EBtTOPaygh10YAAh2OI1Qrp6NWfpAhzswj0ydt9g==", + "dev": true, + "requires": { + "domelementtype": "^2.0.1", + "entities": "^2.0.0" + }, + "dependencies": { + "domelementtype": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/domelementtype/-/domelementtype-2.0.2.tgz", + "integrity": "sha512-wFwTwCVebUrMgGeAwRL/NhZtHAUyT9n9yg4IMDwf10+6iCMxSkVq9MGCVEH+QZWo1nNidy8kNvwmv4zWHDTqvA==", + "dev": true + } + } + }, + "domelementtype": { + "version": "1.3.1", + "resolved": "https://registry.npmjs.org/domelementtype/-/domelementtype-1.3.1.tgz", + "integrity": "sha512-BSKB+TSpMpFI/HOxCNr1O8aMOTZ8hT3pM3GQ0w/mWRmkhEDSFJkkyzz4XQsBV44BChwGkrDfMyjVD0eA2aFV3w==", + "dev": true + }, + "domutils": { + "version": "1.5.1", + "resolved": "https://registry.npmjs.org/domutils/-/domutils-1.5.1.tgz", + "integrity": "sha1-3NhIiib1Y9YQeeSMn3t+Mjc2gs8=", + "dev": true, + "requires": { + "dom-serializer": "0", + "domelementtype": "1" + } + } + } + }, + "css-select-base-adapter": { + "version": "0.1.1", + "resolved": "https://registry.npmjs.org/css-select-base-adapter/-/css-select-base-adapter-0.1.1.tgz", + "integrity": "sha512-jQVeeRG70QI08vSTwf1jHxp74JoZsr2XSgETae8/xC8ovSnL2WF87GTLO86Sbwdt2lK4Umg4HnnwMO4YF3Ce7w==", + "dev": true + }, + "css-selector-tokenizer": { + "version": "0.7.3", + "resolved": "https://registry.npmjs.org/css-selector-tokenizer/-/css-selector-tokenizer-0.7.3.tgz", + "integrity": "sha512-jWQv3oCEL5kMErj4wRnK/OPoBi0D+P1FR2cDCKYPaMeD2eW3/mttav8HT4hT1CKopiJI/psEULjkClhvJo4Lvg==", + "dev": true, + "requires": { + "cssesc": "^3.0.0", + "fastparse": "^1.1.2" + } + }, + "css-tree": { + "version": "1.0.0-alpha.37", + "resolved": "https://registry.npmjs.org/css-tree/-/css-tree-1.0.0-alpha.37.tgz", + "integrity": "sha512-DMxWJg0rnz7UgxKT0Q1HU/L9BeJI0M6ksor0OgqOnF+aRCDWg/N2641HmVyU9KVIu0OVVWOb2IpC9A+BJRnejg==", + "dev": true, + "requires": { + "mdn-data": "2.0.4", + "source-map": "^0.6.1" + }, + "dependencies": { + "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 + } + } + }, + "css-what": { + "version": "2.1.3", + "resolved": "https://registry.npmjs.org/css-what/-/css-what-2.1.3.tgz", + "integrity": "sha512-a+EPoD+uZiNfh+5fxw2nO9QwFa6nJe2Or35fGY6Ipw1R3R4AGz1d1TEZrCegvw2YTmZ0jXirGYlzxxpYSHwpEg==", + "dev": true + }, + "cssauron": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/cssauron/-/cssauron-1.4.0.tgz", + "integrity": "sha1-pmAt/34EqDBtwNuaVR6S6LVmKtg=", + "dev": true, + "requires": { + "through": "X.X.X" + } + }, + "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 + }, + "cssnano": { + "version": "4.1.10", + "resolved": "https://registry.npmjs.org/cssnano/-/cssnano-4.1.10.tgz", + "integrity": "sha512-5wny+F6H4/8RgNlaqab4ktc3e0/blKutmq8yNlBFXA//nSFFAqAngjNVRzUvCgYROULmZZUoosL/KSoZo5aUaQ==", + "dev": true, + "requires": { + "cosmiconfig": "^5.0.0", + "cssnano-preset-default": "^4.0.7", + "is-resolvable": "^1.0.0", + "postcss": "^7.0.0" + }, + "dependencies": { + "cosmiconfig": { + "version": "5.2.1", + "resolved": "https://registry.npmjs.org/cosmiconfig/-/cosmiconfig-5.2.1.tgz", + "integrity": "sha512-H65gsXo1SKjf8zmrJ67eJk8aIRKV5ff2D4uKZIBZShbhGSpEmsQOPW/SKMKYhSTrqR7ufy6RP69rPogdaPh/kA==", + "dev": true, + "requires": { + "import-fresh": "^2.0.0", + "is-directory": "^0.3.1", + "js-yaml": "^3.13.1", + "parse-json": "^4.0.0" + } + }, + "import-fresh": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/import-fresh/-/import-fresh-2.0.0.tgz", + "integrity": "sha1-2BNVwVYS04bGH53dOSLUMEgipUY=", + "dev": true, + "requires": { + "caller-path": "^2.0.0", + "resolve-from": "^3.0.0" + } + }, + "parse-json": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/parse-json/-/parse-json-4.0.0.tgz", + "integrity": "sha1-vjX1Qlvh9/bHRxhPmKeIy5lHfuA=", + "dev": true, + "requires": { + "error-ex": "^1.3.1", + "json-parse-better-errors": "^1.0.1" + } + }, + "resolve-from": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/resolve-from/-/resolve-from-3.0.0.tgz", + "integrity": "sha1-six699nWiBvItuZTM17rywoYh0g=", + "dev": true + } + } + }, + "cssnano-preset-default": { + "version": "4.0.7", + "resolved": "https://registry.npmjs.org/cssnano-preset-default/-/cssnano-preset-default-4.0.7.tgz", + "integrity": "sha512-x0YHHx2h6p0fCl1zY9L9roD7rnlltugGu7zXSKQx6k2rYw0Hi3IqxcoAGF7u9Q5w1nt7vK0ulxV8Lo+EvllGsA==", + "dev": true, + "requires": { + "css-declaration-sorter": "^4.0.1", + "cssnano-util-raw-cache": "^4.0.1", + "postcss": "^7.0.0", + "postcss-calc": "^7.0.1", + "postcss-colormin": "^4.0.3", + "postcss-convert-values": "^4.0.1", + "postcss-discard-comments": "^4.0.2", + "postcss-discard-duplicates": "^4.0.2", + "postcss-discard-empty": "^4.0.1", + "postcss-discard-overridden": "^4.0.1", + "postcss-merge-longhand": "^4.0.11", + "postcss-merge-rules": "^4.0.3", + "postcss-minify-font-values": "^4.0.2", + "postcss-minify-gradients": "^4.0.2", + "postcss-minify-params": "^4.0.2", + "postcss-minify-selectors": "^4.0.2", + "postcss-normalize-charset": "^4.0.1", + "postcss-normalize-display-values": "^4.0.2", + "postcss-normalize-positions": "^4.0.2", + "postcss-normalize-repeat-style": "^4.0.2", + "postcss-normalize-string": "^4.0.2", + "postcss-normalize-timing-functions": "^4.0.2", + "postcss-normalize-unicode": "^4.0.1", + "postcss-normalize-url": "^4.0.1", + "postcss-normalize-whitespace": "^4.0.2", + "postcss-ordered-values": "^4.1.2", + "postcss-reduce-initial": "^4.0.3", + "postcss-reduce-transforms": "^4.0.2", + "postcss-svgo": "^4.0.2", + "postcss-unique-selectors": "^4.0.1" + } + }, + "cssnano-util-get-arguments": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/cssnano-util-get-arguments/-/cssnano-util-get-arguments-4.0.0.tgz", + "integrity": "sha1-7ToIKZ8h11dBsg87gfGU7UnMFQ8=", + "dev": true + }, + "cssnano-util-get-match": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/cssnano-util-get-match/-/cssnano-util-get-match-4.0.0.tgz", + "integrity": "sha1-wOTKB/U4a7F+xeUiULT1lhNlFW0=", + "dev": true + }, + "cssnano-util-raw-cache": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/cssnano-util-raw-cache/-/cssnano-util-raw-cache-4.0.1.tgz", + "integrity": "sha512-qLuYtWK2b2Dy55I8ZX3ky1Z16WYsx544Q0UWViebptpwn/xDBmog2TLg4f+DBMg1rJ6JDWtn96WHbOKDWt1WQA==", + "dev": true, + "requires": { + "postcss": "^7.0.0" + } + }, + "cssnano-util-same-parent": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/cssnano-util-same-parent/-/cssnano-util-same-parent-4.0.1.tgz", + "integrity": "sha512-WcKx5OY+KoSIAxBW6UBBRay1U6vkYheCdjyVNDm85zt5K9mHoGOfsOsqIszfAqrQQFIIKgjh2+FDgIj/zsl21Q==", + "dev": true + }, + "csso": { + "version": "4.1.1", + "resolved": "https://registry.npmjs.org/csso/-/csso-4.1.1.tgz", + "integrity": "sha512-Rvq+e1e0TFB8E8X+8MQjHSY6vtol45s5gxtLI/018UsAn2IBMmwNEZRM/h+HVnAJRHjasLIKKUO3uvoMM28LvA==", + "dev": true, + "requires": { + "css-tree": "^1.0.0" + }, + "dependencies": { + "css-tree": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/css-tree/-/css-tree-1.1.1.tgz", + "integrity": "sha512-NVN42M2fjszcUNpDbdkvutgQSlFYsr1z7kqeuCagHnNLBfYor6uP1WL1KrkmdYZ5Y1vTBCIOI/C/+8T98fJ71w==", + "dev": true, + "requires": { + "mdn-data": "2.0.14", + "source-map": "^0.6.1" + } + }, + "mdn-data": { + "version": "2.0.14", + "resolved": "https://registry.npmjs.org/mdn-data/-/mdn-data-2.0.14.tgz", + "integrity": "sha512-dn6wd0uw5GsdswPFfsgMp5NSB0/aDe6fK94YJV/AJDYXL6HVLWBsxeq7js7Ad+mU2K9LAlwpk6kN2D5mwCPVow==", + "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 + } + } + }, + "cssom": { + "version": "0.4.4", + "resolved": "https://registry.npmjs.org/cssom/-/cssom-0.4.4.tgz", + "integrity": "sha512-p3pvU7r1MyyqbTk+WbNJIgJjG2VmTIaB10rI93LzVPrmDJKkzKYMtxxyAvQXR/NS6otuzveI7+7BBq3SjBS2mw==", + "dev": true + }, + "cssstyle": { + "version": "2.3.0", + "resolved": "https://registry.npmjs.org/cssstyle/-/cssstyle-2.3.0.tgz", + "integrity": "sha512-AZL67abkUzIuvcHqk7c09cezpGNcxUxU4Ioi/05xHk4DQeTkWmGYftIE6ctU6AEt+Gn4n1lDStOtj7FKycP71A==", + "dev": true, + "requires": { + "cssom": "~0.3.6" + }, + "dependencies": { + "cssom": { + "version": "0.3.8", + "resolved": "https://registry.npmjs.org/cssom/-/cssom-0.3.8.tgz", + "integrity": "sha512-b0tGHbfegbhPJpxpiBPU2sCkigAqtM9O121le6bbOlgyV+NyGyCmVfJ6QW9eRjz8CpNfWEOYBIMIGRYkLwsIYg==", + "dev": true + } + } + }, + "cyclist": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/cyclist/-/cyclist-1.0.1.tgz", + "integrity": "sha1-WW6WmP0MgOEgOMK4LW6xs1tiJNk=", + "dev": true + }, + "damerau-levenshtein": { + "version": "1.0.6", + "resolved": "https://registry.npmjs.org/damerau-levenshtein/-/damerau-levenshtein-1.0.6.tgz", + "integrity": "sha512-JVrozIeElnj3QzfUIt8tB8YMluBJom4Vw9qTPpjGYQ9fYlB3D/rb6OordUxf3xeFB35LKWs0xqcO5U6ySvBtug==", + "dev": true + }, + "dargs": { + "version": "6.1.0", + "resolved": "https://registry.npmjs.org/dargs/-/dargs-6.1.0.tgz", + "integrity": "sha512-5dVBvpBLBnPwSsYXqfybFyehMmC/EenKEcf23AhCTgTf48JFBbmJKqoZBsERDnjL0FyiVTYWdFsRfTLHxLyKdQ==", + "dev": true + }, + "dashdash": { + "version": "1.14.1", + "resolved": "https://registry.npmjs.org/dashdash/-/dashdash-1.14.1.tgz", + "integrity": "sha1-hTz6D3y+L+1d4gMmuN1YEDX24vA=", + "dev": true, + "requires": { + "assert-plus": "^1.0.0" + } + }, + "data-urls": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/data-urls/-/data-urls-2.0.0.tgz", + "integrity": "sha512-X5eWTSXO/BJmpdIKCRuKUgSCgAN0OwliVK3yPKbwIWU1Tdw5BRajxlzMidvh+gwko9AfQ9zIj52pzF91Q3YAvQ==", + "dev": true, + "requires": { + "abab": "^2.0.3", + "whatwg-mimetype": "^2.3.0", + "whatwg-url": "^8.0.0" + } + }, + "date-fns": { + "version": "1.30.1", + "resolved": "https://registry.npmjs.org/date-fns/-/date-fns-1.30.1.tgz", + "integrity": "sha512-hBSVCvSmWC+QypYObzwGOd9wqdDpOt+0wl0KbU+R+uuZBS1jN8VsD1ss3irQDknRj5NvxiTF6oj/nDRnN/UQNw==", + "dev": true + }, + "dateformat": { + "version": "3.0.3", + "resolved": "https://registry.npmjs.org/dateformat/-/dateformat-3.0.3.tgz", + "integrity": "sha512-jyCETtSl3VMZMWeRo7iY1FL19ges1t55hMo5yaam4Jrsm5EPL89UQkoQRyiI+Yf4k8r2ZpdngkV8hr1lIdjb3Q==", + "dev": true + }, + "debounce-fn": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/debounce-fn/-/debounce-fn-3.0.1.tgz", + "integrity": "sha512-aBoJh5AhpqlRoHZjHmOzZlRx+wz2xVwGL9rjs+Kj0EWUrL4/h4K7OD176thl2Tdoqui/AaA4xhHrNArGLAaI3Q==", + "dev": true, + "requires": { + "mimic-fn": "^2.1.0" + } + }, + "debug": { + "version": "4.3.1", + "resolved": "https://registry.npmjs.org/debug/-/debug-4.3.1.tgz", + "integrity": "sha512-doEwdvm4PCeK4K3RQN2ZC2BYUBaxwLARCqZmMjtF8a51J2Rb0xpVloFRnCODwqjpwnAoao4pelN8l3RJdv3gRQ==", + "requires": { + "ms": "2.1.2" + } + }, + "debuglog": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/debuglog/-/debuglog-1.0.1.tgz", + "integrity": "sha1-qiT/uaw9+aI1GDfPstJ5NgzXhJI=", + "dev": true + }, + "decamelize": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/decamelize/-/decamelize-1.2.0.tgz", + "integrity": "sha1-9lNNFRSCabIDUue+4m9QH5oZEpA=" + }, + "decamelize-keys": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/decamelize-keys/-/decamelize-keys-1.1.0.tgz", + "integrity": "sha1-0XGoeTMlKAfrPLYdwcFEXQeN8tk=", + "dev": true, + "requires": { + "decamelize": "^1.1.0", + "map-obj": "^1.0.0" + }, + "dependencies": { + "map-obj": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/map-obj/-/map-obj-1.0.1.tgz", + "integrity": "sha1-2TPOuSBdgr3PSIb2dCvcK03qFG0=", + "dev": true + } + } + }, + "decimal.js": { + "version": "10.2.1", + "resolved": "https://registry.npmjs.org/decimal.js/-/decimal.js-10.2.1.tgz", + "integrity": "sha512-KaL7+6Fw6i5A2XSnsbhm/6B+NuEA7TZ4vqxnd5tXz9sbKtrN9Srj8ab4vKVdK8YAqZO9P1kg45Y6YLoduPf+kw==", + "dev": true + }, + "decode-uri-component": { + "version": "0.2.0", + "resolved": "https://registry.npmjs.org/decode-uri-component/-/decode-uri-component-0.2.0.tgz", + "integrity": "sha1-6zkTMzRYd1y4TNGh+uBiEGu4dUU=", + "dev": true + }, + "dedent": { + "version": "0.7.0", + "resolved": "https://registry.npmjs.org/dedent/-/dedent-0.7.0.tgz", + "integrity": "sha1-JJXduvbrh0q7Dhvp3yLS5aVEMmw=", + "dev": true + }, + "deep-equal": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/deep-equal/-/deep-equal-1.1.1.tgz", + "integrity": "sha512-yd9c5AdiqVcR+JjcwUQb9DkhJc8ngNr0MahEBGvDiJw8puWab2yZlh+nkasOnZP+EGTAP6rRp2JzJhJZzvNF8g==", + "dev": true, + "requires": { + "is-arguments": "^1.0.4", + "is-date-object": "^1.0.1", + "is-regex": "^1.0.4", + "object-is": "^1.0.1", + "object-keys": "^1.1.1", + "regexp.prototype.flags": "^1.2.0" + } + }, + "deep-extend": { + "version": "0.6.0", + "resolved": "https://registry.npmjs.org/deep-extend/-/deep-extend-0.6.0.tgz", + "integrity": "sha512-LOHxIOaPYdHlJRtCQfDIVZtfw/ufM8+rVj649RIHzcm/vGwQRXFt6OPqIFWsm2XEMrNIEtWR64sY1LEKD2vAOA==", + "dev": true + }, + "deep-is": { + "version": "0.1.3", + "resolved": "https://registry.npmjs.org/deep-is/-/deep-is-0.1.3.tgz", + "integrity": "sha1-s2nW+128E+7PUk+RsHD+7cNXzzQ=", + "dev": true + }, + "deepmerge": { + "version": "4.2.2", + "resolved": "https://registry.npmjs.org/deepmerge/-/deepmerge-4.2.2.tgz", + "integrity": "sha512-FJ3UgI4gIl+PHZm53knsuSFpE+nESMr7M4v9QcgB7S63Kj/6WqMiFQJpBBYz1Pt+66bZpP3Q7Lye0Oo9MPKEdg==", + "dev": true + }, + "default-gateway": { + "version": "4.2.0", + "resolved": "https://registry.npmjs.org/default-gateway/-/default-gateway-4.2.0.tgz", + "integrity": "sha512-h6sMrVB1VMWVrW13mSc6ia/DwYYw5MN6+exNu1OaJeFac5aSAvwM7lZ0NVfTABuSkQelr4h5oebg3KB1XPdjgA==", + "dev": true, + "requires": { + "execa": "^1.0.0", + "ip-regex": "^2.1.0" + } + }, + "defaults": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/defaults/-/defaults-1.0.3.tgz", + "integrity": "sha1-xlYFHpgX2f8I7YgUd/P+QBnz730=", + "dev": true, + "requires": { + "clone": "^1.0.2" + } + }, + "define-properties": { + "version": "1.1.3", + "resolved": "https://registry.npmjs.org/define-properties/-/define-properties-1.1.3.tgz", + "integrity": "sha512-3MqfYKj2lLzdMSf8ZIZE/V+Zuy+BgD6f164e8K2w7dgnpKArBDerGYpM46IYYcjnkdPNMjPk9A6VFB8+3SKlXQ==", + "dev": true, + "requires": { + "object-keys": "^1.0.12" + } + }, + "define-property": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/define-property/-/define-property-2.0.2.tgz", + "integrity": "sha512-jwK2UV4cnPpbcG7+VRARKTZPUWowwXA8bzH5NP6ud0oeAxyYPuGZUAC7hMugpCdz4BeSZl2Dl9k66CHJ/46ZYQ==", + "dev": true, + "requires": { + "is-descriptor": "^1.0.2", + "isobject": "^3.0.1" + }, + "dependencies": { + "is-accessor-descriptor": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/is-accessor-descriptor/-/is-accessor-descriptor-1.0.0.tgz", + "integrity": "sha512-m5hnHTkcVsPfqx3AKlyttIPb7J+XykHvJP2B9bZDjlhLIoEq4XoK64Vg7boZlVWYK6LUY94dYPEE7Lh0ZkZKcQ==", + "dev": true, + "requires": { + "kind-of": "^6.0.0" + } + }, + "is-data-descriptor": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/is-data-descriptor/-/is-data-descriptor-1.0.0.tgz", + "integrity": "sha512-jbRXy1FmtAoCjQkVmIVYwuuqDFUbaOeDjmed1tOGPrsMhtJA4rD9tkgA0F1qJ3gRFRXcHYVkdeaP50Q5rE/jLQ==", + "dev": true, + "requires": { + "kind-of": "^6.0.0" + } + }, + "is-descriptor": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/is-descriptor/-/is-descriptor-1.0.2.tgz", + "integrity": "sha512-2eis5WqQGV7peooDyLmNEPUrps9+SXX5c9pL3xEB+4e9HnGuDa7mB7kHxHw4CbqS9k1T2hOH3miL8n8WtiYVtg==", + "dev": true, + "requires": { + "is-accessor-descriptor": "^1.0.0", + "is-data-descriptor": "^1.0.0", + "kind-of": "^6.0.2" + } + } + } + }, + "del": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/del/-/del-3.0.0.tgz", + "integrity": "sha1-U+z2mf/LyzljdpGrE7rxYIGXZuU=", + "dev": true, + "requires": { + "globby": "^6.1.0", + "is-path-cwd": "^1.0.0", + "is-path-in-cwd": "^1.0.0", + "p-map": "^1.1.1", + "pify": "^3.0.0", + "rimraf": "^2.2.8" + }, + "dependencies": { + "array-union": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/array-union/-/array-union-1.0.2.tgz", + "integrity": "sha1-mjRBDk9OPaI96jdb5b5w8kd47Dk=", + "dev": true, + "requires": { + "array-uniq": "^1.0.1" + } + }, + "globby": { + "version": "6.1.0", + "resolved": "https://registry.npmjs.org/globby/-/globby-6.1.0.tgz", + "integrity": "sha1-9abXDoOV4hyFj7BInWTfAkJNUGw=", + "dev": true, + "requires": { + "array-union": "^1.0.1", + "glob": "^7.0.3", + "object-assign": "^4.0.1", + "pify": "^2.0.0", + "pinkie-promise": "^2.0.0" + }, + "dependencies": { + "pify": { + "version": "2.3.0", + "resolved": "https://registry.npmjs.org/pify/-/pify-2.3.0.tgz", + "integrity": "sha1-7RQaasBDqEnqWISY59yosVMw6Qw=", + "dev": true + } + } + }, + "p-map": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/p-map/-/p-map-1.2.0.tgz", + "integrity": "sha512-r6zKACMNhjPJMTl8KcFH4li//gkrXWfbD6feV8l6doRHlzljFWGJ2AP6iKaCJXyZmAUMOPtvbW7EXkbWO/pLEA==", + "dev": true + }, + "pify": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/pify/-/pify-3.0.0.tgz", + "integrity": "sha1-5aSs0sEB/fPZpNB/DbxNtJ3SgXY=", + "dev": true + }, + "rimraf": { + "version": "2.7.1", + "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-2.7.1.tgz", + "integrity": "sha512-uWjbaKIK3T1OSVptzX7Nl6PvQ3qAGtKEtVRjRuazjfL3Bx5eI409VZSqgND+4UNnmzLVdPj9FqFJNPqBZFve4w==", + "dev": true, + "requires": { + "glob": "^7.1.3" + }, + "dependencies": { + "glob": { + "version": "7.1.6", + "resolved": "https://registry.npmjs.org/glob/-/glob-7.1.6.tgz", + "integrity": "sha512-LwaxwyZ72Lk7vZINtNNrywX0ZuLyStrdDtabefZKAY5ZGJhVtgdznluResxNmPitE0SAO+O26sWTHeKSI2wMBA==", + "dev": true, + "requires": { + "fs.realpath": "^1.0.0", + "inflight": "^1.0.4", + "inherits": "2", + "minimatch": "^3.0.4", + "once": "^1.3.0", + "path-is-absolute": "^1.0.0" + } + } + } + } + } + }, + "delayed-stream": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/delayed-stream/-/delayed-stream-1.0.0.tgz", + "integrity": "sha1-3zrhmayt+31ECqrgsp4icrJOxhk=", + "dev": true + }, + "delegates": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/delegates/-/delegates-1.0.0.tgz", + "integrity": "sha1-hMbhWbgZBP3KWaDvRM2HDTElD5o=", + "dev": true + }, + "depd": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/depd/-/depd-1.1.2.tgz", + "integrity": "sha1-m81S4UwJd2PnSbJ0xDRu0uVgtak=", + "dev": true + }, + "dependency-graph": { + "version": "0.7.2", + "resolved": "https://registry.npmjs.org/dependency-graph/-/dependency-graph-0.7.2.tgz", + "integrity": "sha512-KqtH4/EZdtdfWX0p6MGP9jljvxSY6msy/pRUD4jgNwVpv3v1QmNLlsB3LDSSUg79BRVSn7jI1QPRtArGABovAQ==", + "dev": true + }, + "des.js": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/des.js/-/des.js-1.0.1.tgz", + "integrity": "sha512-Q0I4pfFrv2VPd34/vfLrFOoRmlYj3OV50i7fskps1jZWK1kApMWWT9G6RRUeYedLcBDIhnSDaUvJMb3AhUlaEA==", + "dev": true, + "requires": { + "inherits": "^2.0.1", + "minimalistic-assert": "^1.0.0" + } + }, + "destroy": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/destroy/-/destroy-1.0.4.tgz", + "integrity": "sha1-l4hXRCxEdJ5CBmE+N5RiBYJqvYA=", + "dev": true + }, + "detect-file": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/detect-file/-/detect-file-1.0.0.tgz", + "integrity": "sha1-8NZtA2cqglyxtzvbP+YjEMjlUrc=", + "dev": true + }, + "detect-newline": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/detect-newline/-/detect-newline-3.1.0.tgz", + "integrity": "sha512-TLz+x/vEXm/Y7P7wn1EJFNLxYpUD4TgMosxY6fAVJUnJMbupHBOncxyWUG9OpTaH9EBD7uFI5LfEgmMOc54DsA==", + "dev": true + }, + "detect-node": { + "version": "2.0.4", + "resolved": "https://registry.npmjs.org/detect-node/-/detect-node-2.0.4.tgz", + "integrity": "sha512-ZIzRpLJrOj7jjP2miAtgqIfmzbxa4ZOr5jJc601zklsfEx9oTzmmj2nVpIPRpNlRTIh8lc1kyViIY7BWSGNmKw==", + "dev": true + }, + "dev-ip": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/dev-ip/-/dev-ip-1.0.1.tgz", + "integrity": "sha1-p2o+0YVb56ASu4rBbLgPPADcKPA=", + "dev": true + }, + "dezalgo": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/dezalgo/-/dezalgo-1.0.3.tgz", + "integrity": "sha1-f3Qt4Gb8dIvI24IFad3c5Jvw1FY=", + "dev": true, + "requires": { + "asap": "^2.0.0", + "wrappy": "1" + } + }, + "diagnostics": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/diagnostics/-/diagnostics-1.1.1.tgz", + "integrity": "sha512-8wn1PmdunLJ9Tqbx+Fx/ZEuHfJf4NKSN2ZBj7SJC/OWRWha843+WsTjqMe1B5E3p28jqBlp+mJ2fPVxPyNgYKQ==", + "dev": true, + "requires": { + "colorspace": "1.1.x", + "enabled": "1.0.x", + "kuler": "1.0.x" + } + }, + "didyoumean": { + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/didyoumean/-/didyoumean-1.2.1.tgz", + "integrity": "sha1-6S7f2tplN9SE1zwBcv0eugxJdv8=", + "dev": true + }, + "diff": { + "version": "3.5.0", + "resolved": "https://registry.npmjs.org/diff/-/diff-3.5.0.tgz", + "integrity": "sha512-A46qtFgd+g7pDZinpnwiRJtxbC1hpgf0uzP3iG89scHk0AUC7A1TGxf5OiiOUv/JMZR8GOt8hL900hV0bOy5xA==", + "dev": true + }, + "diff-sequences": { + "version": "25.2.6", + "resolved": "https://registry.npmjs.org/diff-sequences/-/diff-sequences-25.2.6.tgz", + "integrity": "sha512-Hq8o7+6GaZeoFjtpgvRBUknSXNeJiCx7V9Fr94ZMljNiCr9n9L8H8aJqgWOQiDDGdyn29fRNcDdRVJ5fdyihfg==", + "dev": true + }, + "diffie-hellman": { + "version": "5.0.3", + "resolved": "https://registry.npmjs.org/diffie-hellman/-/diffie-hellman-5.0.3.tgz", + "integrity": "sha512-kqag/Nl+f3GwyK25fhUMYj81BUOrZ9IuJsjIcDE5icNM9FJHAVm3VcUDxdLPoQtTuUylWm6ZIknYJwwaPxsUzg==", + "dev": true, + "requires": { + "bn.js": "^4.1.0", + "miller-rabin": "^4.0.0", + "randombytes": "^2.0.0" + }, + "dependencies": { + "bn.js": { + "version": "4.11.9", + "resolved": "https://registry.npmjs.org/bn.js/-/bn.js-4.11.9.tgz", + "integrity": "sha512-E6QoYqCKZfgatHTdHzs1RRKP7ip4vvm+EyRUeE2RF0NblwVvb0p6jSVeNTOFxPn26QXN2o6SMfNxKp6kU8zQaw==", + "dev": true + } + } + }, + "dir-glob": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/dir-glob/-/dir-glob-3.0.1.tgz", + "integrity": "sha512-WkrWp9GR4KXfKGYzOLmTuGVi1UWFfws377n9cc55/tb6DuqyF6pcQ5AbiHEshaDpY9v6oaSr2XCDidGmMwdzIA==", + "dev": true, + "requires": { + "path-type": "^4.0.0" + }, + "dependencies": { + "path-type": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/path-type/-/path-type-4.0.0.tgz", + "integrity": "sha512-gDKb8aZMDeD/tZWs9P6+q0J9Mwkdl6xMV8TjnGP3qJVJ06bdMgkbBlLU8IdfOsIsFz2BW1rNVT3XuNEl8zPAvw==", + "dev": true + } + } + }, + "dlv": { + "version": "1.1.3", + "resolved": "https://registry.npmjs.org/dlv/-/dlv-1.1.3.tgz", + "integrity": "sha512-+HlytyjlPKnIG8XuRG8WvmBP8xs8P71y+SKKS6ZXWoEgLuePxtDoUEiH7WkdePWrQ5JBpE6aoVqfZfJUQkjXwA==", + "dev": true + }, + "dns-equal": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/dns-equal/-/dns-equal-1.0.0.tgz", + "integrity": "sha1-s55/HabrCnW6nBcySzR1PEfgZU0=", + "dev": true + }, + "dns-packet": { + "version": "1.3.1", + "resolved": "https://registry.npmjs.org/dns-packet/-/dns-packet-1.3.1.tgz", + "integrity": "sha512-0UxfQkMhYAUaZI+xrNZOz/as5KgDU0M/fQ9b6SpkyLbk3GEswDi6PADJVaYJradtRVsRIlF1zLyOodbcTCDzUg==", + "dev": true, + "requires": { + "ip": "^1.1.0", + "safe-buffer": "^5.0.1" + } + }, + "dns-txt": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/dns-txt/-/dns-txt-2.0.2.tgz", + "integrity": "sha1-uR2Ab10nGI5Ks+fRB9iBocxGQrY=", + "dev": true, + "requires": { + "buffer-indexof": "^1.0.0" + } + }, + "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" + } + }, + "dom-converter": { + "version": "0.2.0", + "resolved": "https://registry.npmjs.org/dom-converter/-/dom-converter-0.2.0.tgz", + "integrity": "sha512-gd3ypIPfOMr9h5jIKq8E3sHOTCjeirnl0WK5ZdS1AW0Odt0b1PaWaHdJ4Qk4klv+YB9aJBS7mESXjFoDQPu6DA==", + "dev": true, + "requires": { + "utila": "~0.4" + } + }, + "dom-serializer": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/dom-serializer/-/dom-serializer-1.1.0.tgz", + "integrity": "sha512-ox7bvGXt2n+uLWtCRLybYx60IrOlWL/aCebWJk1T0d4m3y2tzf4U3ij9wBMUb6YJZpz06HCCYuyCDveE2xXmzQ==", + "dev": true, + "requires": { + "domelementtype": "^2.0.1", + "domhandler": "^3.0.0", + "entities": "^2.0.0" + } + }, + "domain-browser": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/domain-browser/-/domain-browser-1.2.0.tgz", + "integrity": "sha512-jnjyiM6eRyZl2H+W8Q/zLMA481hzi0eszAaBUzIVnmYVDBbnLxVNnfu1HgEBvCbL+71FrxMl3E6lpKH7Ge3OXA==", + "dev": true + }, + "domelementtype": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/domelementtype/-/domelementtype-2.0.2.tgz", + "integrity": "sha512-wFwTwCVebUrMgGeAwRL/NhZtHAUyT9n9yg4IMDwf10+6iCMxSkVq9MGCVEH+QZWo1nNidy8kNvwmv4zWHDTqvA==", + "dev": true + }, + "domexception": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/domexception/-/domexception-2.0.1.tgz", + "integrity": "sha512-yxJ2mFy/sibVQlu5qHjOkf9J3K6zgmCxgJ94u2EdvDOV09H+32LtRswEcUsmUWN72pVLOEnTSRaIVVzVQgS0dg==", + "dev": true, + "requires": { + "webidl-conversions": "^5.0.0" + }, + "dependencies": { + "webidl-conversions": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/webidl-conversions/-/webidl-conversions-5.0.0.tgz", + "integrity": "sha512-VlZwKPCkYKxQgeSbH5EyngOmRp7Ww7I9rQLERETtf5ofd9pGeswWiOtogpEO850jziPRarreGxn5QIiTqpb2wA==", + "dev": true + } + } + }, + "domhandler": { + "version": "3.3.0", + "resolved": "https://registry.npmjs.org/domhandler/-/domhandler-3.3.0.tgz", + "integrity": "sha512-J1C5rIANUbuYK+FuFL98650rihynUOEzRLxW+90bKZRWB6A1X1Tf82GxR1qAWLyfNPRvjqfip3Q5tdYlmAa9lA==", + "dev": true, + "requires": { + "domelementtype": "^2.0.1" + } + }, + "domutils": { + "version": "2.4.2", + "resolved": "https://registry.npmjs.org/domutils/-/domutils-2.4.2.tgz", + "integrity": "sha512-NKbgaM8ZJOecTZsIzW5gSuplsX2IWW2mIK7xVr8hTQF2v1CJWTmLZ1HOCh5sH+IzVPAGE5IucooOkvwBRAdowA==", + "dev": true, + "requires": { + "dom-serializer": "^1.0.1", + "domelementtype": "^2.0.1", + "domhandler": "^3.3.0" + } + }, + "dot-case": { + "version": "3.0.3", + "resolved": "https://registry.npmjs.org/dot-case/-/dot-case-3.0.3.tgz", + "integrity": "sha512-7hwEmg6RiSQfm/GwPL4AAWXKy3YNNZA3oFv2Pdiey0mwkRCPZ9x6SZbkLcn8Ma5PYeVokzoD4Twv2n7LKp5WeA==", + "dev": true, + "requires": { + "no-case": "^3.0.3", + "tslib": "^1.10.0" + }, + "dependencies": { + "tslib": { + "version": "1.14.1", + "resolved": "https://registry.npmjs.org/tslib/-/tslib-1.14.1.tgz", + "integrity": "sha512-Xni35NKzjgMrwevysHTCArtLDpPvye8zV/0E4EyYn43P7/7qvQwPh9BGkHewbMulVntbigmcT7rdX3BNo9wRJg==", + "dev": true + } + } + }, + "dot-prop": { + "version": "5.3.0", + "resolved": "https://registry.npmjs.org/dot-prop/-/dot-prop-5.3.0.tgz", + "integrity": "sha512-QM8q3zDe58hqUqjraQOmzZ1LIH9SWQJTlEKCH4kJ2oQvLZk7RbQXvtDM2XEq3fwkV9CCvvH4LA0AV+ogFsBM2Q==", + "dev": true, + "requires": { + "is-obj": "^2.0.0" + } + }, + "download-stats": { + "version": "0.3.4", + "resolved": "https://registry.npmjs.org/download-stats/-/download-stats-0.3.4.tgz", + "integrity": "sha512-ic2BigbyUWx7/CBbsfGjf71zUNZB4edBGC3oRliSzsoNmvyVx3Ycfp1w3vp2Y78Ee0eIIkjIEO5KzW0zThDGaA==", + "dev": true, + "requires": { + "JSONStream": "^1.2.1", + "lazy-cache": "^2.0.1", + "moment": "^2.15.1" + } + }, + "drange": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/drange/-/drange-1.1.1.tgz", + "integrity": "sha512-pYxfDYpued//QpnLIm4Avk7rsNtAtQkUES2cwAYSvD/wd2pKD71gN2Ebj3e7klzXwjocvE8c5vx/1fxwpqmSxA==", + "dev": true + }, + "duplexer": { + "version": "0.1.2", + "resolved": "https://registry.npmjs.org/duplexer/-/duplexer-0.1.2.tgz", + "integrity": "sha512-jtD6YG370ZCIi/9GTaJKQxWTZD045+4R4hTk/x1UyoqadyJ9x9CgSi1RlVDQF8U2sxLLSnFkCaMihqljHIWgMg==", + "dev": true + }, + "duplexer3": { + "version": "0.1.4", + "resolved": "https://registry.npmjs.org/duplexer3/-/duplexer3-0.1.4.tgz", + "integrity": "sha1-7gHdHKwO08vH/b6jfcCo8c4ALOI=", + "dev": true + }, + "duplexify": { + "version": "3.7.1", + "resolved": "https://registry.npmjs.org/duplexify/-/duplexify-3.7.1.tgz", + "integrity": "sha512-07z8uv2wMyS51kKhD1KsdXJg5WQ6t93RneqRxUHnskXVtlYYkLqM0gqStQZ3pj073g687jPCHrqNfCzawLYh5g==", + "dev": true, + "requires": { + "end-of-stream": "^1.0.0", + "inherits": "^2.0.1", + "readable-stream": "^2.0.0", + "stream-shift": "^1.0.0" + } + }, + "easy-extender": { + "version": "2.3.4", + "resolved": "https://registry.npmjs.org/easy-extender/-/easy-extender-2.3.4.tgz", + "integrity": "sha512-8cAwm6md1YTiPpOvDULYJL4ZS6WfM5/cTeVVh4JsvyYZAoqlRVUpHL9Gr5Fy7HA6xcSZicUia3DeAgO3Us8E+Q==", + "dev": true, + "requires": { + "lodash": "^4.17.10" + } + }, + "eazy-logger": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/eazy-logger/-/eazy-logger-3.1.0.tgz", + "integrity": "sha512-/snsn2JqBtUSSstEl4R0RKjkisGHAhvYj89i7r3ytNUKW12y178KDZwXLXIgwDqLW6E/VRMT9qfld7wvFae8bQ==", + "dev": true, + "requires": { + "tfunk": "^4.0.0" + } + }, + "ecc-jsbn": { + "version": "0.1.2", + "resolved": "https://registry.npmjs.org/ecc-jsbn/-/ecc-jsbn-0.1.2.tgz", + "integrity": "sha1-OoOpBOVDUyh4dMVkt1SThoSamMk=", + "dev": true, + "requires": { + "jsbn": "~0.1.0", + "safer-buffer": "^2.1.0" + } + }, + "editions": { + "version": "2.3.1", + "resolved": "https://registry.npmjs.org/editions/-/editions-2.3.1.tgz", + "integrity": "sha512-ptGvkwTvGdGfC0hfhKg0MT+TRLRKGtUiWGBInxOm5pz7ssADezahjCUaYuZ8Dr+C05FW0AECIIPt4WBxVINEhA==", + "dev": true, + "requires": { + "errlop": "^2.0.0", + "semver": "^6.3.0" + }, + "dependencies": { + "semver": { + "version": "6.3.0", + "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.0.tgz", + "integrity": "sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw==", + "dev": true + } + } + }, + "ee-first": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/ee-first/-/ee-first-1.1.1.tgz", + "integrity": "sha1-WQxhFWsK4vTwJVcyoViyZrxWsh0=", + "dev": true + }, + "ejs": { + "version": "3.1.3", + "resolved": "https://registry.npmjs.org/ejs/-/ejs-3.1.3.tgz", + "integrity": "sha512-wmtrUGyfSC23GC/B1SMv2ogAUgbQEtDmTIhfqielrG5ExIM9TP4UoYdi90jLF1aTcsWCJNEO0UrgKzP0y3nTSg==", + "dev": true, + "requires": { + "jake": "^10.6.1" + } + }, + "electron-to-chromium": { + "version": "1.3.606", + "resolved": "https://registry.npmjs.org/electron-to-chromium/-/electron-to-chromium-1.3.606.tgz", + "integrity": "sha512-+/2yPHwtNf6NWKpaYt0KoqdSZ6Qddt6nDfH/pnhcrHq9hSb23e5LFy06Mlf0vF2ykXvj7avJ597psqcbKnG5YQ==", + "dev": true + }, + "elegant-spinner": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/elegant-spinner/-/elegant-spinner-1.0.1.tgz", + "integrity": "sha1-2wQ1IcldfjA/2PNFvtwzSc+wcp4=", + "dev": true + }, + "elliptic": { + "version": "6.5.3", + "resolved": "https://registry.npmjs.org/elliptic/-/elliptic-6.5.3.tgz", + "integrity": "sha512-IMqzv5wNQf+E6aHeIqATs0tOLeOTwj1QKbRcS3jBbYkl5oLAserA8yJTT7/VyHUYG91PRmPyeQDObKLPpeS4dw==", + "dev": true, + "requires": { + "bn.js": "^4.4.0", + "brorand": "^1.0.1", + "hash.js": "^1.0.0", + "hmac-drbg": "^1.0.0", + "inherits": "^2.0.1", + "minimalistic-assert": "^1.0.0", + "minimalistic-crypto-utils": "^1.0.0" + }, + "dependencies": { + "bn.js": { + "version": "4.11.9", + "resolved": "https://registry.npmjs.org/bn.js/-/bn.js-4.11.9.tgz", + "integrity": "sha512-E6QoYqCKZfgatHTdHzs1RRKP7ip4vvm+EyRUeE2RF0NblwVvb0p6jSVeNTOFxPn26QXN2o6SMfNxKp6kU8zQaw==", + "dev": true + } + } + }, + "emittery": { + "version": "0.7.2", + "resolved": "https://registry.npmjs.org/emittery/-/emittery-0.7.2.tgz", + "integrity": "sha512-A8OG5SR/ij3SsJdWDJdkkSYUjQdCUx6APQXem0SaEePBSRg4eymGYwBkKo1Y6DU+af/Jn2dBQqDBvjnr9Vi8nQ==", + "dev": true + }, + "emoji-regex": { + "version": "8.0.0", + "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-8.0.0.tgz", + "integrity": "sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A==" + }, + "emojis-list": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/emojis-list/-/emojis-list-3.0.0.tgz", + "integrity": "sha512-/kyM18EfinwXZbno9FyUGeFh87KC8HRQBQGildHZbEuRyWFOmv1U10o9BBp8XVZDVNNuQKyIGIu5ZYAAXJ0V2Q==", + "dev": true + }, + "enabled": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/enabled/-/enabled-1.0.2.tgz", + "integrity": "sha1-ll9lE9LC0cX0ZStkouM5ZGf8L5M=", + "dev": true, + "requires": { + "env-variable": "0.0.x" + } + }, + "encodeurl": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/encodeurl/-/encodeurl-1.0.2.tgz", + "integrity": "sha1-rT/0yG7C0CkyL1oCw6mmBslbP1k=", + "dev": true + }, + "encoding": { + "version": "0.1.13", + "resolved": "https://registry.npmjs.org/encoding/-/encoding-0.1.13.tgz", + "integrity": "sha512-ETBauow1T35Y/WZMkio9jiM0Z5xjHHmJ4XmjZOq1l/dXz3lr2sRn87nJy20RupqSh1F2m3HHPSp8ShIPQJrJ3A==", + "dev": true, + "requires": { + "iconv-lite": "^0.6.2" + } + }, + "end-of-stream": { + "version": "1.4.4", + "resolved": "https://registry.npmjs.org/end-of-stream/-/end-of-stream-1.4.4.tgz", + "integrity": "sha512-+uw1inIHVPQoaVuHzRyXd21icM+cnt4CzD5rW+NC1wjOUSTOs+Te7FOv7AhN7vS9x/oIyhLP5PR1H+phQAHu5Q==", + "dev": true, + "requires": { + "once": "^1.4.0" + } + }, + "engine.io": { + "version": "3.2.1", + "resolved": "https://registry.npmjs.org/engine.io/-/engine.io-3.2.1.tgz", + "integrity": "sha512-+VlKzHzMhaU+GsCIg4AoXF1UdDFjHHwMmMKqMJNDNLlUlejz58FCy4LBqB2YVJskHGYl06BatYWKP2TVdVXE5w==", + "dev": true, + "requires": { + "accepts": "~1.3.4", + "base64id": "1.0.0", + "cookie": "0.3.1", + "debug": "~3.1.0", + "engine.io-parser": "~2.1.0", + "ws": "~3.3.1" + }, + "dependencies": { + "base64-arraybuffer": { + "version": "0.1.5", + "resolved": "https://registry.npmjs.org/base64-arraybuffer/-/base64-arraybuffer-0.1.5.tgz", + "integrity": "sha1-c5JncZI7Whl0etZmqlzUv5xunOg=", + "dev": true + }, + "debug": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/debug/-/debug-3.1.0.tgz", + "integrity": "sha512-OX8XqP7/1a9cqkxYw2yXss15f26NKWBpDXQd0/uK/KPqdQhxbPa994hnzjcE2VqQpDslf55723cKPUOGSmMY3g==", + "dev": true, + "requires": { + "ms": "2.0.0" + } + }, + "engine.io-parser": { + "version": "2.1.3", + "resolved": "https://registry.npmjs.org/engine.io-parser/-/engine.io-parser-2.1.3.tgz", + "integrity": "sha512-6HXPre2O4Houl7c4g7Ic/XzPnHBvaEmN90vtRO9uLmwtRqQmTOw0QMevL1TOfL2Cpu1VzsaTmMotQgMdkzGkVA==", + "dev": true, + "requires": { + "after": "0.8.2", + "arraybuffer.slice": "~0.0.7", + "base64-arraybuffer": "0.1.5", + "blob": "0.0.5", + "has-binary2": "~1.0.2" + } + }, + "ms": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", + "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=", + "dev": true + }, + "ws": { + "version": "3.3.3", + "resolved": "https://registry.npmjs.org/ws/-/ws-3.3.3.tgz", + "integrity": "sha512-nnWLa/NwZSt4KQJu51MYlCcSQ5g7INpOrOMt4XV8j4dqTXdmlUmSHQ8/oLC069ckre0fRsgfvsKwbTdtKLCDkA==", + "dev": true, + "requires": { + "async-limiter": "~1.0.0", + "safe-buffer": "~5.1.0", + "ultron": "~1.1.0" + } + } + } + }, + "engine.io-client": { + "version": "3.4.4", + "resolved": "https://registry.npmjs.org/engine.io-client/-/engine.io-client-3.4.4.tgz", + "integrity": "sha512-iU4CRr38Fecj8HoZEnFtm2EiKGbYZcPn3cHxqNGl/tmdWRf60KhK+9vE0JeSjgnlS/0oynEfLgKbT9ALpim0sQ==", + "dev": true, + "requires": { + "component-emitter": "~1.3.0", + "component-inherit": "0.0.3", + "debug": "~3.1.0", + "engine.io-parser": "~2.2.0", + "has-cors": "1.1.0", + "indexof": "0.0.1", + "parseqs": "0.0.6", + "parseuri": "0.0.6", + "ws": "~6.1.0", + "xmlhttprequest-ssl": "~1.5.4", + "yeast": "0.1.2" + }, + "dependencies": { + "debug": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/debug/-/debug-3.1.0.tgz", + "integrity": "sha512-OX8XqP7/1a9cqkxYw2yXss15f26NKWBpDXQd0/uK/KPqdQhxbPa994hnzjcE2VqQpDslf55723cKPUOGSmMY3g==", + "dev": true, + "requires": { + "ms": "2.0.0" + } + }, + "ms": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", + "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=", + "dev": true + } + } + }, + "engine.io-parser": { + "version": "2.2.1", + "resolved": "https://registry.npmjs.org/engine.io-parser/-/engine.io-parser-2.2.1.tgz", + "integrity": "sha512-x+dN/fBH8Ro8TFwJ+rkB2AmuVw9Yu2mockR/p3W8f8YtExwFgDvBDi0GWyb4ZLkpahtDGZgtr3zLovanJghPqg==", + "dev": true, + "requires": { + "after": "0.8.2", + "arraybuffer.slice": "~0.0.7", + "base64-arraybuffer": "0.1.4", + "blob": "0.0.5", + "has-binary2": "~1.0.2" + } + }, + "enhanced-resolve": { + "version": "4.1.1", + "resolved": "https://registry.npmjs.org/enhanced-resolve/-/enhanced-resolve-4.1.1.tgz", + "integrity": "sha512-98p2zE+rL7/g/DzMHMTF4zZlCgeVdJ7yr6xzEpJRYwFYrGi9ANdn5DnJURg6RpBkyk60XYDnWIv51VfIhfNGuA==", + "dev": true, + "requires": { + "graceful-fs": "^4.1.2", + "memory-fs": "^0.5.0", + "tapable": "^1.0.0" + } + }, + "entities": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/entities/-/entities-2.1.0.tgz", + "integrity": "sha512-hCx1oky9PFrJ611mf0ifBLBRW8lUUVRlFolb5gWRfIELabBlbp9xZvrqZLZAs+NxFnbfQoeGd8wDkygjg7U85w==", + "dev": true + }, + "env-paths": { + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/env-paths/-/env-paths-2.2.0.tgz", + "integrity": "sha512-6u0VYSCo/OW6IoD5WCLLy9JUGARbamfSavcNXry/eu8aHVFei6CD3Sw+VGX5alea1i9pgPHW0mbu6Xj0uBh7gA==", + "dev": true + }, + "env-variable": { + "version": "0.0.6", + "resolved": "https://registry.npmjs.org/env-variable/-/env-variable-0.0.6.tgz", + "integrity": "sha512-bHz59NlBbtS0NhftmR8+ExBEekE7br0e01jw+kk0NDro7TtZzBYZ5ScGPs3OmwnpyfHTHOtr1Y6uedCdrIldtg==", + "dev": true + }, + "err-code": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/err-code/-/err-code-1.1.2.tgz", + "integrity": "sha1-BuARbTAo9q70gGhJ6w6mp0iuaWA=", + "dev": true + }, + "errlop": { + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/errlop/-/errlop-2.2.0.tgz", + "integrity": "sha512-e64Qj9+4aZzjzzFpZC7p5kmm/ccCrbLhAJplhsDXQFs87XTsXwOpH4s1Io2s90Tau/8r2j9f4l/thhDevRjzxw==", + "dev": true + }, + "errno": { + "version": "0.1.7", + "resolved": "https://registry.npmjs.org/errno/-/errno-0.1.7.tgz", + "integrity": "sha512-MfrRBDWzIWifgq6tJj60gkAwtLNb6sQPlcFrSOflcP1aFmmruKQ2wRnze/8V6kgyz7H3FF8Npzv78mZ7XLLflg==", + "dev": true, + "requires": { + "prr": "~1.0.1" + } + }, + "error": { + "version": "7.2.1", + "resolved": "https://registry.npmjs.org/error/-/error-7.2.1.tgz", + "integrity": "sha512-fo9HBvWnx3NGUKMvMwB/CBCMMrfEJgbDTVDEkPygA3Bdd3lM1OyCd+rbQ8BwnpF6GdVeOLDNmyL4N5Bg80ZvdA==", + "dev": true, + "requires": { + "string-template": "~0.2.1" + } + }, + "error-ex": { + "version": "1.3.2", + "resolved": "https://registry.npmjs.org/error-ex/-/error-ex-1.3.2.tgz", + "integrity": "sha512-7dFHNmqeFSEt2ZBsCriorKnn3Z2pj+fd9kmI6QoWw4//DL+icEBfc0U7qJCisqrTsKTjw4fNFy2pW9OqStD84g==", + "dev": true, + "requires": { + "is-arrayish": "^0.2.1" + } + }, + "error-stack-parser": { + "version": "2.0.6", + "resolved": "https://registry.npmjs.org/error-stack-parser/-/error-stack-parser-2.0.6.tgz", + "integrity": "sha512-d51brTeqC+BHlwF0BhPtcYgF5nlzf9ZZ0ZIUQNZpc9ZB9qw5IJ2diTrBY9jlCJkTLITYPjmiX6OWCwH+fuyNgQ==", + "dev": true, + "requires": { + "stackframe": "^1.1.1" + } + }, + "es-abstract": { + "version": "1.17.7", + "resolved": "https://registry.npmjs.org/es-abstract/-/es-abstract-1.17.7.tgz", + "integrity": "sha512-VBl/gnfcJ7OercKA9MVaegWsBHFjV492syMudcnQZvt/Dw8ezpcOHYZXa/J96O8vx+g4x65YKhxOwDUh63aS5g==", + "dev": true, + "requires": { + "es-to-primitive": "^1.2.1", + "function-bind": "^1.1.1", + "has": "^1.0.3", + "has-symbols": "^1.0.1", + "is-callable": "^1.2.2", + "is-regex": "^1.1.1", + "object-inspect": "^1.8.0", + "object-keys": "^1.1.1", + "object.assign": "^4.1.1", + "string.prototype.trimend": "^1.0.1", + "string.prototype.trimstart": "^1.0.1" + } + }, + "es-to-primitive": { + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/es-to-primitive/-/es-to-primitive-1.2.1.tgz", + "integrity": "sha512-QCOllgZJtaUo9miYBcLChTUaHNjJF3PYs1VidD7AwiEj1kYxKeQTctLAezAOH5ZKRH0g2IgPn6KwB4IT8iRpvA==", + "dev": true, + "requires": { + "is-callable": "^1.1.4", + "is-date-object": "^1.0.1", + "is-symbol": "^1.0.2" + } + }, + "es6-promise": { + "version": "4.2.8", + "resolved": "https://registry.npmjs.org/es6-promise/-/es6-promise-4.2.8.tgz", + "integrity": "sha512-HJDGx5daxeIvxdBxvG2cb9g4tEvwIk3i8+nhX0yGrYmZUzbkdg8QbDevheDB8gd0//uPj4c1EQua8Q+MViT0/w==", + "dev": true + }, + "es6-promisify": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/es6-promisify/-/es6-promisify-5.0.0.tgz", + "integrity": "sha1-UQnWLz5W6pZ8S2NQWu8IKRyKUgM=", + "dev": true, + "requires": { + "es6-promise": "^4.0.3" + } + }, + "escalade": { + "version": "3.1.1", + "resolved": "https://registry.npmjs.org/escalade/-/escalade-3.1.1.tgz", + "integrity": "sha512-k0er2gUkLf8O0zKJiAhmkTnJlTvINGv7ygDNPbeIsX/TJjGJZHuh9B2UxbsaEkmlEo9MfhrSzmhIlhRlI2GXnw==", + "dev": true + }, + "escape-html": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/escape-html/-/escape-html-1.0.3.tgz", + "integrity": "sha1-Aljq5NPQwJdN4cFpGI7wBR0dGYg=", + "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": "sha1-G2HAViGQqN/2rjuyzwIAyhMLhtQ=" + }, + "escodegen": { + "version": "1.14.3", + "resolved": "https://registry.npmjs.org/escodegen/-/escodegen-1.14.3.tgz", + "integrity": "sha512-qFcX0XJkdg+PB3xjZZG/wKSuT1PnQWx57+TVSjIMmILd2yC/6ByYElPwJnslDsuWuSAp4AwJGumarAAmJch5Kw==", + "dev": true, + "requires": { + "esprima": "^4.0.1", + "estraverse": "^4.2.0", + "esutils": "^2.0.2", + "optionator": "^0.8.1", + "source-map": "~0.6.1" + }, + "dependencies": { + "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, + "optional": true + } + } + }, + "eslint": { + "version": "6.8.0", + "resolved": "https://registry.npmjs.org/eslint/-/eslint-6.8.0.tgz", + "integrity": "sha512-K+Iayyo2LtyYhDSYwz5D5QdWw0hCacNzyq1Y821Xna2xSJj7cijoLLYmLxTQgcgZ9mC61nryMy9S7GRbYpI5Ig==", + "dev": true, + "requires": { + "@babel/code-frame": "^7.0.0", + "ajv": "^6.10.0", + "chalk": "^2.1.0", + "cross-spawn": "^6.0.5", + "debug": "^4.0.1", + "doctrine": "^3.0.0", + "eslint-scope": "^5.0.0", + "eslint-utils": "^1.4.3", + "eslint-visitor-keys": "^1.1.0", + "espree": "^6.1.2", + "esquery": "^1.0.1", + "esutils": "^2.0.2", + "file-entry-cache": "^5.0.1", + "functional-red-black-tree": "^1.0.1", + "glob-parent": "^5.0.0", + "globals": "^12.1.0", + "ignore": "^4.0.6", + "import-fresh": "^3.0.0", + "imurmurhash": "^0.1.4", + "inquirer": "^7.0.0", + "is-glob": "^4.0.0", + "js-yaml": "^3.13.1", + "json-stable-stringify-without-jsonify": "^1.0.1", + "levn": "^0.3.0", + "lodash": "^4.17.14", + "minimatch": "^3.0.4", + "mkdirp": "^0.5.1", + "natural-compare": "^1.4.0", + "optionator": "^0.8.3", + "progress": "^2.0.0", + "regexpp": "^2.0.1", + "semver": "^6.1.2", + "strip-ansi": "^5.2.0", + "strip-json-comments": "^3.0.1", + "table": "^5.2.3", + "text-table": "^0.2.0", + "v8-compile-cache": "^2.0.3" + }, + "dependencies": { + "ansi-regex": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-4.1.0.tgz", + "integrity": "sha512-1apePfXM1UOSqw0o9IiFAovVz9M5S1Dg+4TrDwfMewQ6p/rmMueb7tWZjQ1rx4Loy1ArBggoqGpfqqdI4rondg==", + "dev": true + }, + "eslint-utils": { + "version": "1.4.3", + "resolved": "https://registry.npmjs.org/eslint-utils/-/eslint-utils-1.4.3.tgz", + "integrity": "sha512-fbBN5W2xdY45KulGXmLHZ3c3FHfVYmKg0IrAKGOkT/464PQsx2UeIzfz1RmEci+KLm1bBaAzZAh8+/E+XAeZ8Q==", + "dev": true, + "requires": { + "eslint-visitor-keys": "^1.1.0" + } + }, + "globals": { + "version": "12.4.0", + "resolved": "https://registry.npmjs.org/globals/-/globals-12.4.0.tgz", + "integrity": "sha512-BWICuzzDvDoH54NHKCseDanAhE3CeDorgDL5MT6LMXXj2WCnd9UC2szdk4AWLfjdgNBCXLUanXYcpBBKOSWGwg==", + "dev": true, + "requires": { + "type-fest": "^0.8.1" + } + }, + "ignore": { + "version": "4.0.6", + "resolved": "https://registry.npmjs.org/ignore/-/ignore-4.0.6.tgz", + "integrity": "sha512-cyFDKrqc/YdcWFniJhzI42+AzS+gNwmUzOSFcRCQYwySuBBBy/KjuxWLZ/FHEH6Moq1NizMOBWyTcv8O4OZIMg==", + "dev": true + }, + "regexpp": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/regexpp/-/regexpp-2.0.1.tgz", + "integrity": "sha512-lv0M6+TkDVniA3aD1Eg0DVpfU/booSu7Eev3TDO/mZKHBfVjgCGTV4t4buppESEYDtkArYFOxTJWv6S5C+iaNw==", + "dev": true + }, + "semver": { + "version": "6.3.0", + "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.0.tgz", + "integrity": "sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw==", + "dev": true + }, + "strip-ansi": { + "version": "5.2.0", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-5.2.0.tgz", + "integrity": "sha512-DuRs1gKbBqsMKIZlrffwlug8MHkcnpjs5VPmL1PAh+mA30U0DTotfDZ0d2UUsXpPmPmMMJ6W773MaA3J+lbiWA==", + "dev": true, + "requires": { + "ansi-regex": "^4.1.0" + } + }, + "type-fest": { + "version": "0.8.1", + "resolved": "https://registry.npmjs.org/type-fest/-/type-fest-0.8.1.tgz", + "integrity": "sha512-4dbzIzqvjtgiM5rw1k5rEHtBANKmdudhGyBEajN01fEyhaAIhsoKNy6y7+IN93IfpFtwY9iqi7kD+xwKhQsNJA==", + "dev": true + } + } + }, + "eslint-config-jhipster": { + "version": "0.0.1", + "resolved": "https://registry.npmjs.org/eslint-config-jhipster/-/eslint-config-jhipster-0.0.1.tgz", + "integrity": "sha512-E9/05t3+6V2K9YrJwsLmrblGgYnYfw62o/ja+Hr9wz3+AX2nG6gtycMW7LrS4alFuar1SGhbAyT3orTR5ur9MQ==", + "dev": true + }, + "eslint-config-prettier": { + "version": "6.11.0", + "resolved": "https://registry.npmjs.org/eslint-config-prettier/-/eslint-config-prettier-6.11.0.tgz", + "integrity": "sha512-oB8cpLWSAjOVFEJhhyMZh6NOEOtBVziaqdDQ86+qhDHFbZXoRTM7pNSvFRfW/W/L/LrQ38C99J5CGuRBBzBsdA==", + "dev": true, + "requires": { + "get-stdin": "^6.0.0" + } + }, + "eslint-loader": { + "version": "4.0.2", + "resolved": "https://registry.npmjs.org/eslint-loader/-/eslint-loader-4.0.2.tgz", + "integrity": "sha512-EDpXor6lsjtTzZpLUn7KmXs02+nIjGcgees9BYjNkWra3jVq5vVa8IoCKgzT2M7dNNeoMBtaSG83Bd40N3poLw==", + "dev": true, + "requires": { + "find-cache-dir": "^3.3.1", + "fs-extra": "^8.1.0", + "loader-utils": "^2.0.0", + "object-hash": "^2.0.3", + "schema-utils": "^2.6.5" + }, + "dependencies": { + "fs-extra": { + "version": "8.1.0", + "resolved": "https://registry.npmjs.org/fs-extra/-/fs-extra-8.1.0.tgz", + "integrity": "sha512-yhlQgA6mnOJUKOsRUFsgJdQCvkKhcz8tlZG5HBQfReYZy46OwLcY+Zia0mtdHsOo9y/hP+CxMN0TU9QxoOtG4g==", + "dev": true, + "requires": { + "graceful-fs": "^4.2.0", + "jsonfile": "^4.0.0", + "universalify": "^0.1.0" + } + } + } + }, + "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" + } + }, + "eslint-utils": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/eslint-utils/-/eslint-utils-2.1.0.tgz", + "integrity": "sha512-w94dQYoauyvlDc43XnGB8lU3Zt713vNChgt4EWwhXAP2XkBvndfxF0AgIqKOOasjPIPzj9JqgwkwbCYD0/V3Zg==", + "dev": true, + "requires": { + "eslint-visitor-keys": "^1.1.0" + } + }, + "eslint-visitor-keys": { + "version": "1.3.0", + "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-1.3.0.tgz", + "integrity": "sha512-6J72N8UNa462wa/KFODt/PJ3IU60SDpC3QXC1Hjc1BXXpfL2C9R5+AU7jhe0F6GREqVMh4Juu+NY7xn+6dipUQ==", + "dev": true + }, + "espree": { + "version": "6.2.1", + "resolved": "https://registry.npmjs.org/espree/-/espree-6.2.1.tgz", + "integrity": "sha512-ysCxRQY3WaXJz9tdbWOwuWr5Y/XrPTGX9Kiz3yoUXwW0VZ4w30HTkQLaGx/+ttFjF8i+ACbArnB4ce68a9m5hw==", + "dev": true, + "requires": { + "acorn": "^7.1.1", + "acorn-jsx": "^5.2.0", + "eslint-visitor-keys": "^1.1.0" + } + }, + "esprima": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/esprima/-/esprima-4.0.1.tgz", + "integrity": "sha512-eGuFFw7Upda+g4p+QHvnW0RyTX/SVeJBDM/gCtMARO0cLuT2HcEKnTPvhjV6aGeqrCB/sbNop0Kszm0jsaWU4A==", + "dev": true + }, + "esquery": { + "version": "1.3.1", + "resolved": "https://registry.npmjs.org/esquery/-/esquery-1.3.1.tgz", + "integrity": "sha512-olpvt9QG0vniUBZspVRN6lwB7hOZoTRtT+jzR+tS4ffYx2mzbw+z0XCOk44aaLYKApNX5nMm+E+P6o25ip/DHQ==", + "dev": true, + "requires": { + "estraverse": "^5.1.0" + }, + "dependencies": { + "estraverse": { + "version": "5.2.0", + "resolved": "https://registry.npmjs.org/estraverse/-/estraverse-5.2.0.tgz", + "integrity": "sha512-BxbNGGNm0RyRYvUdHpIwv9IWzeM9XClbOxwoATuFdOE7ZE6wHL+HQ5T8hoPM+zHvmKzzsEqhgy0GrQ5X13afiQ==", + "dev": true + } + } + }, + "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" + }, + "dependencies": { + "estraverse": { + "version": "5.2.0", + "resolved": "https://registry.npmjs.org/estraverse/-/estraverse-5.2.0.tgz", + "integrity": "sha512-BxbNGGNm0RyRYvUdHpIwv9IWzeM9XClbOxwoATuFdOE7ZE6wHL+HQ5T8hoPM+zHvmKzzsEqhgy0GrQ5X13afiQ==", + "dev": true + } + } + }, + "estraverse": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/estraverse/-/estraverse-4.3.0.tgz", + "integrity": "sha512-39nnKffWz8xN1BU/2c79n9nB9HDzo0niYUqx6xyqUnyoAnQyyWpOTdZEeiCch8BBu515t4wp9ZmgVfVhn9EBpw==", + "dev": true + }, + "estree-walker": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/estree-walker/-/estree-walker-1.0.1.tgz", + "integrity": "sha512-1fMXF3YP4pZZVozF8j/ZLfvnR8NSIljt56UhbZ5PeeDmmGHpgpdwQt7ITlGvYaQukCvuBRMLEiKiYC+oeIg4cg==", + "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 + }, + "etag": { + "version": "1.8.1", + "resolved": "https://registry.npmjs.org/etag/-/etag-1.8.1.tgz", + "integrity": "sha1-Qa4u62XvpiJorr/qg6x9eSmbCIc=", + "dev": true + }, + "eventemitter3": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/eventemitter3/-/eventemitter3-1.2.0.tgz", + "integrity": "sha1-HIaZHYFq0eUEdQ5zh0Ik7PO+xQg=", + "dev": true + }, + "events": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/events/-/events-1.1.1.tgz", + "integrity": "sha1-nr23Y1rQmccNzEwqH1AEKI6L2SQ=", + "dev": true + }, + "eventsource": { + "version": "1.0.7", + "resolved": "https://registry.npmjs.org/eventsource/-/eventsource-1.0.7.tgz", + "integrity": "sha512-4Ln17+vVT0k8aWq+t/bF5arcS3EpT9gYtW66EPacdj/mAFevznsnyoHLPy2BA8gbIQeIHoPsvwmfBftfcG//BQ==", + "dev": true, + "requires": { + "original": "^1.0.0" + } + }, + "evp_bytestokey": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/evp_bytestokey/-/evp_bytestokey-1.0.3.tgz", + "integrity": "sha512-/f2Go4TognH/KvCISP7OUsHn85hT9nUkxxA9BEWxFn+Oj9o8ZNLm/40hdlgSLyuOimsrTKLUMEorQexp/aPQeA==", + "dev": true, + "requires": { + "md5.js": "^1.3.4", + "safe-buffer": "^5.1.1" + } + }, + "exec-sh": { + "version": "0.3.4", + "resolved": "https://registry.npmjs.org/exec-sh/-/exec-sh-0.3.4.tgz", + "integrity": "sha512-sEFIkc61v75sWeOe72qyrqg2Qg0OuLESziUDk/O/z2qgS15y2gWVFrI6f2Qn/qw/0/NCfCEsmNA4zOjkwEZT1A==", + "dev": true + }, + "execa": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/execa/-/execa-1.0.0.tgz", + "integrity": "sha512-adbxcyWV46qiHyvSp50TKt05tB4tK3HcmF7/nxfAdhnox83seTDbwnaqKO4sXRy7roHAIFqJP/Rw/AuEbX61LA==", + "dev": true, + "requires": { + "cross-spawn": "^6.0.0", + "get-stream": "^4.0.0", + "is-stream": "^1.1.0", + "npm-run-path": "^2.0.0", + "p-finally": "^1.0.0", + "signal-exit": "^3.0.0", + "strip-eof": "^1.0.0" + } + }, + "exit": { + "version": "0.1.2", + "resolved": "https://registry.npmjs.org/exit/-/exit-0.1.2.tgz", + "integrity": "sha1-BjJjj42HfMghB9MKD/8aF8uhzQw=", + "dev": true + }, + "exit-hook": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/exit-hook/-/exit-hook-1.1.1.tgz", + "integrity": "sha1-8FyiM7SMBdVP/wd2XfhQfpXAL/g=", + "dev": true + }, + "expand-brackets": { + "version": "2.1.4", + "resolved": "https://registry.npmjs.org/expand-brackets/-/expand-brackets-2.1.4.tgz", + "integrity": "sha1-t3c14xXOMPa27/D4OwQVGiJEliI=", + "dev": true, + "requires": { + "debug": "^2.3.3", + "define-property": "^0.2.5", + "extend-shallow": "^2.0.1", + "posix-character-classes": "^0.1.0", + "regex-not": "^1.0.0", + "snapdragon": "^0.8.1", + "to-regex": "^3.0.1" + }, + "dependencies": { + "debug": { + "version": "2.6.9", + "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", + "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", + "dev": true, + "requires": { + "ms": "2.0.0" + } + }, + "define-property": { + "version": "0.2.5", + "resolved": "https://registry.npmjs.org/define-property/-/define-property-0.2.5.tgz", + "integrity": "sha1-w1se+RjsPJkPmlvFe+BKrOxcgRY=", + "dev": true, + "requires": { + "is-descriptor": "^0.1.0" + } + }, + "extend-shallow": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/extend-shallow/-/extend-shallow-2.0.1.tgz", + "integrity": "sha1-Ua99YUrZqfYQ6huvu5idaxxWiQ8=", + "dev": true, + "requires": { + "is-extendable": "^0.1.0" + } + }, + "ms": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", + "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=", + "dev": true + } + } + }, + "expand-tilde": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/expand-tilde/-/expand-tilde-2.0.2.tgz", + "integrity": "sha1-l+gBqgUt8CRU3kawK/YhZCzchQI=", + "dev": true, + "requires": { + "homedir-polyfill": "^1.0.1" + } + }, + "expect": { + "version": "26.6.2", + "resolved": "https://registry.npmjs.org/expect/-/expect-26.6.2.tgz", + "integrity": "sha512-9/hlOBkQl2l/PLHJx6JjoDF6xPKcJEsUlWKb23rKE7KzeDqUZKXKNMW27KIue5JMdBV9HgmoJPcc8HtO85t9IA==", + "dev": true, + "requires": { + "@jest/types": "^26.6.2", + "ansi-styles": "^4.0.0", + "jest-get-type": "^26.3.0", + "jest-matcher-utils": "^26.6.2", + "jest-message-util": "^26.6.2", + "jest-regex-util": "^26.0.0" + }, + "dependencies": { + "@jest/types": { + "version": "26.6.2", + "resolved": "https://registry.npmjs.org/@jest/types/-/types-26.6.2.tgz", + "integrity": "sha512-fC6QCp7Sc5sX6g8Tvbmj4XUTbyrik0akgRy03yjXbQaBWWNWGE7SGtJk98m0N8nzegD/7SggrUlivxo5ax4KWQ==", + "dev": true, + "requires": { + "@types/istanbul-lib-coverage": "^2.0.0", + "@types/istanbul-reports": "^3.0.0", + "@types/node": "*", + "@types/yargs": "^15.0.0", + "chalk": "^4.0.0" + } + }, + "@types/istanbul-reports": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/@types/istanbul-reports/-/istanbul-reports-3.0.0.tgz", + "integrity": "sha512-nwKNbvnwJ2/mndE9ItP/zc2TCzw6uuodnF4EHYWD+gCQDVBuRQL5UzbZD0/ezy1iKsFU2ZQiDqg4M9dN4+wZgA==", + "dev": true, + "requires": { + "@types/istanbul-lib-report": "*" + } + }, + "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.0", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.0.tgz", + "integrity": "sha512-qwx12AxXe2Q5xQ43Ac//I6v5aXTipYrSESdOgzrN+9XjgEpyjpKuvSGaN4qE93f7TQTlerQQ8S+EQ0EyDoVL1A==", + "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 + }, + "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-get-type": { + "version": "26.3.0", + "resolved": "https://registry.npmjs.org/jest-get-type/-/jest-get-type-26.3.0.tgz", + "integrity": "sha512-TpfaviN1R2pQWkIihlfEanwOXK0zcxrKEE4MlU6Tn7keoXdN6/3gK/xl0yEh8DOunn5pOVGKf8hB4R9gVh04ig==", + "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" + } + } + } + }, + "express": { + "version": "4.17.1", + "resolved": "https://registry.npmjs.org/express/-/express-4.17.1.tgz", + "integrity": "sha512-mHJ9O79RqluphRrcw2X/GTh3k9tVv8YcoyY4Kkh4WDMUYKRZUq0h1o0w2rrrxBqM7VoeUVqgb27xlEMXTnYt4g==", + "dev": true, + "requires": { + "accepts": "~1.3.7", + "array-flatten": "1.1.1", + "body-parser": "1.19.0", + "content-disposition": "0.5.3", + "content-type": "~1.0.4", + "cookie": "0.4.0", + "cookie-signature": "1.0.6", + "debug": "2.6.9", + "depd": "~1.1.2", + "encodeurl": "~1.0.2", + "escape-html": "~1.0.3", + "etag": "~1.8.1", + "finalhandler": "~1.1.2", + "fresh": "0.5.2", + "merge-descriptors": "1.0.1", + "methods": "~1.1.2", + "on-finished": "~2.3.0", + "parseurl": "~1.3.3", + "path-to-regexp": "0.1.7", + "proxy-addr": "~2.0.5", + "qs": "6.7.0", + "range-parser": "~1.2.1", + "safe-buffer": "5.1.2", + "send": "0.17.1", + "serve-static": "1.14.1", + "setprototypeof": "1.1.1", + "statuses": "~1.5.0", + "type-is": "~1.6.18", + "utils-merge": "1.0.1", + "vary": "~1.1.2" + }, + "dependencies": { + "cookie": { + "version": "0.4.0", + "resolved": "https://registry.npmjs.org/cookie/-/cookie-0.4.0.tgz", + "integrity": "sha512-+Hp8fLp57wnUSt0tY0tHEXh4voZRDnoIrZPqlo3DPiI4y9lwg/jqx+1Om94/W6ZaPDOUbnjOt/99w66zk+l1Xg==", + "dev": true + }, + "debug": { + "version": "2.6.9", + "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", + "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", + "dev": true, + "requires": { + "ms": "2.0.0" + } + }, + "finalhandler": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/finalhandler/-/finalhandler-1.1.2.tgz", + "integrity": "sha512-aAWcW57uxVNrQZqFXjITpW3sIUQmHGG3qSb9mUah9MgMC4NeWhNOlNjXEYq3HjRAvL6arUviZGGJsBg6z0zsWA==", + "dev": true, + "requires": { + "debug": "2.6.9", + "encodeurl": "~1.0.2", + "escape-html": "~1.0.3", + "on-finished": "~2.3.0", + "parseurl": "~1.3.3", + "statuses": "~1.5.0", + "unpipe": "~1.0.0" + } + }, + "mime": { + "version": "1.6.0", + "resolved": "https://registry.npmjs.org/mime/-/mime-1.6.0.tgz", + "integrity": "sha512-x0Vn8spI+wuJ1O6S7gnbaQg8Pxh4NNHb7KSINmEWKiPE4RKOplvijn+NkmYmmRgP68mc70j2EbeTFRsrswaQeg==", + "dev": true + }, + "ms": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", + "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=", + "dev": true + }, + "qs": { + "version": "6.7.0", + "resolved": "https://registry.npmjs.org/qs/-/qs-6.7.0.tgz", + "integrity": "sha512-VCdBRNFTX1fyE7Nb6FYoURo/SPe62QCaAyzJvUjwRaIsc+NePBEniHlvxFmmX56+HZphIGtV0XeCirBtpDrTyQ==", + "dev": true + }, + "send": { + "version": "0.17.1", + "resolved": "https://registry.npmjs.org/send/-/send-0.17.1.tgz", + "integrity": "sha512-BsVKsiGcQMFwT8UxypobUKyv7irCNRHk1T0G680vk88yf6LBByGcZJOTJCrTP2xVN6yI+XjPJcNuE3V4fT9sAg==", + "dev": true, + "requires": { + "debug": "2.6.9", + "depd": "~1.1.2", + "destroy": "~1.0.4", + "encodeurl": "~1.0.2", + "escape-html": "~1.0.3", + "etag": "~1.8.1", + "fresh": "0.5.2", + "http-errors": "~1.7.2", + "mime": "1.6.0", + "ms": "2.1.1", + "on-finished": "~2.3.0", + "range-parser": "~1.2.1", + "statuses": "~1.5.0" + }, + "dependencies": { + "ms": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.1.tgz", + "integrity": "sha512-tgp+dl5cGk28utYktBsrFqA7HKgrhgPsg6Z/EfhWI4gl1Hwq8B/GmY/0oXZ6nF8hDVesS/FpnYaD/kOWhYQvyg==", + "dev": true + } + } + }, + "serve-static": { + "version": "1.14.1", + "resolved": "https://registry.npmjs.org/serve-static/-/serve-static-1.14.1.tgz", + "integrity": "sha512-JMrvUwE54emCYWlTI+hGrGv5I8dEwmco/00EvkzIIsR7MqrHonbD9pO2MOfFnpFntl7ecpZs+3mW+XbQZu9QCg==", + "dev": true, + "requires": { + "encodeurl": "~1.0.2", + "escape-html": "~1.0.3", + "parseurl": "~1.3.3", + "send": "0.17.1" + } + }, + "statuses": { + "version": "1.5.0", + "resolved": "https://registry.npmjs.org/statuses/-/statuses-1.5.0.tgz", + "integrity": "sha1-Fhx9rBd2Wf2YEfQ3cfqZOBR4Yow=", + "dev": true + } + } + }, + "extend": { + "version": "3.0.2", + "resolved": "https://registry.npmjs.org/extend/-/extend-3.0.2.tgz", + "integrity": "sha512-fjquC59cD7CyW6urNXK0FBufkZcoiGG80wTuPujX590cB5Ttln20E2UB4S/WARVqhXffZl2LNgS+gQdPIIim/g==", + "dev": true + }, + "extend-shallow": { + "version": "3.0.2", + "resolved": "https://registry.npmjs.org/extend-shallow/-/extend-shallow-3.0.2.tgz", + "integrity": "sha1-Jqcarwc7OfshJxcnRhMcJwQCjbg=", + "dev": true, + "requires": { + "assign-symbols": "^1.0.0", + "is-extendable": "^1.0.1" + }, + "dependencies": { + "is-extendable": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/is-extendable/-/is-extendable-1.0.1.tgz", + "integrity": "sha512-arnXMxT1hhoKo9k1LZdmlNyJdDDfy2v0fXjFlmok4+i8ul/6WlbVge9bhM74OpNPQPMGUToDtz+KXa1PneJxOA==", + "dev": true, + "requires": { + "is-plain-object": "^2.0.4" + } + } + } + }, + "external-editor": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/external-editor/-/external-editor-3.1.0.tgz", + "integrity": "sha512-hMQ4CX1p1izmuLYyZqLMO/qGNw10wSv9QDCPfzXfyFrOaCSSoRfqE1Kf1s5an66J5JZC62NewG+mK49jOCtQew==", + "dev": true, + "requires": { + "chardet": "^0.7.0", + "iconv-lite": "^0.4.24", + "tmp": "^0.0.33" + }, + "dependencies": { + "iconv-lite": { + "version": "0.4.24", + "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.4.24.tgz", + "integrity": "sha512-v3MXnZAcvnywkTUEZomIActle7RXXeedOR31wwl7VlyoXO4Qi9arvSenNQWne1TcRwhCL1HwLI21bEqdpj8/rA==", + "dev": true, + "requires": { + "safer-buffer": ">= 2.1.2 < 3" + } + } + } + }, + "extglob": { + "version": "2.0.4", + "resolved": "https://registry.npmjs.org/extglob/-/extglob-2.0.4.tgz", + "integrity": "sha512-Nmb6QXkELsuBr24CJSkilo6UHHgbekK5UiZgfE6UHD3Eb27YC6oD+bhcT+tJ6cl8dmsgdQxnWlcry8ksBIBLpw==", + "dev": true, + "requires": { + "array-unique": "^0.3.2", + "define-property": "^1.0.0", + "expand-brackets": "^2.1.4", + "extend-shallow": "^2.0.1", + "fragment-cache": "^0.2.1", + "regex-not": "^1.0.0", + "snapdragon": "^0.8.1", + "to-regex": "^3.0.1" + }, + "dependencies": { + "define-property": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/define-property/-/define-property-1.0.0.tgz", + "integrity": "sha1-dp66rz9KY6rTr56NMEybvnm/sOY=", + "dev": true, + "requires": { + "is-descriptor": "^1.0.0" + } + }, + "extend-shallow": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/extend-shallow/-/extend-shallow-2.0.1.tgz", + "integrity": "sha1-Ua99YUrZqfYQ6huvu5idaxxWiQ8=", + "dev": true, + "requires": { + "is-extendable": "^0.1.0" + } + }, + "is-accessor-descriptor": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/is-accessor-descriptor/-/is-accessor-descriptor-1.0.0.tgz", + "integrity": "sha512-m5hnHTkcVsPfqx3AKlyttIPb7J+XykHvJP2B9bZDjlhLIoEq4XoK64Vg7boZlVWYK6LUY94dYPEE7Lh0ZkZKcQ==", + "dev": true, + "requires": { + "kind-of": "^6.0.0" + } + }, + "is-data-descriptor": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/is-data-descriptor/-/is-data-descriptor-1.0.0.tgz", + "integrity": "sha512-jbRXy1FmtAoCjQkVmIVYwuuqDFUbaOeDjmed1tOGPrsMhtJA4rD9tkgA0F1qJ3gRFRXcHYVkdeaP50Q5rE/jLQ==", + "dev": true, + "requires": { + "kind-of": "^6.0.0" + } + }, + "is-descriptor": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/is-descriptor/-/is-descriptor-1.0.2.tgz", + "integrity": "sha512-2eis5WqQGV7peooDyLmNEPUrps9+SXX5c9pL3xEB+4e9HnGuDa7mB7kHxHw4CbqS9k1T2hOH3miL8n8WtiYVtg==", + "dev": true, + "requires": { + "is-accessor-descriptor": "^1.0.0", + "is-data-descriptor": "^1.0.0", + "kind-of": "^6.0.2" + } + } + } + }, + "extsprintf": { + "version": "1.3.0", + "resolved": "https://registry.npmjs.org/extsprintf/-/extsprintf-1.3.0.tgz", + "integrity": "sha1-lpGEQOMEGnpBT4xS48V06zw+HgU=", + "dev": true + }, + "faker": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/faker/-/faker-4.1.0.tgz", + "integrity": "sha1-HkW7vsxndLPBlfrSg1EJxtdIzD8=", + "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-glob": { + "version": "3.2.4", + "resolved": "https://registry.npmjs.org/fast-glob/-/fast-glob-3.2.4.tgz", + "integrity": "sha512-kr/Oo6PX51265qeuCYsyGypiO5uJFgBS0jksyG7FUeCyQzNwYnzrNIMR1NXfkZXsMYXYLRAHgISHBz8gQcxKHQ==", + "dev": true, + "requires": { + "@nodelib/fs.stat": "^2.0.2", + "@nodelib/fs.walk": "^1.2.3", + "glob-parent": "^5.1.0", + "merge2": "^1.3.0", + "micromatch": "^4.0.2", + "picomatch": "^2.2.1" + }, + "dependencies": { + "micromatch": { + "version": "4.0.2", + "resolved": "https://registry.npmjs.org/micromatch/-/micromatch-4.0.2.tgz", + "integrity": "sha512-y7FpHSbMUMoyPbYUSzO6PaZ6FyRnQOpHuKwbo1G+Knck95XVU4QAiKdGEnj5wwoS7PlOgthX/09u5iFJ+aYf5Q==", + "dev": true, + "requires": { + "braces": "^3.0.1", + "picomatch": "^2.0.5" + } + } + } + }, + "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": "sha1-PYpcZog6FqMMqGQ+hR8Zuqd5eRc=", + "dev": true + }, + "fast-safe-stringify": { + "version": "2.0.7", + "resolved": "https://registry.npmjs.org/fast-safe-stringify/-/fast-safe-stringify-2.0.7.tgz", + "integrity": "sha512-Utm6CdzT+6xsDk2m8S6uL8VHxNwI6Jub+e9NYTcAms28T84pTa25GJQV9j0CY0N1rM8hK4x6grpF2BQf+2qwVA==", + "dev": true + }, + "fastparse": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/fastparse/-/fastparse-1.1.2.tgz", + "integrity": "sha512-483XLLxTVIwWK3QTrMGRqUfUpoOs/0hbQrl2oz4J0pAcm3A3bu84wxTFqGqkJzewCLdME38xJLJAxBABfQT8sQ==", + "dev": true + }, + "fastq": { + "version": "1.9.0", + "resolved": "https://registry.npmjs.org/fastq/-/fastq-1.9.0.tgz", + "integrity": "sha512-i7FVWL8HhVY+CTkwFxkN2mk3h+787ixS5S63eb78diVRc1MCssarHq3W5cj0av7YDSwmaV928RNag+U1etRQ7w==", + "dev": true, + "requires": { + "reusify": "^1.0.4" + } + }, + "faye-websocket": { + "version": "0.10.0", + "resolved": "https://registry.npmjs.org/faye-websocket/-/faye-websocket-0.10.0.tgz", + "integrity": "sha1-TkkvjQTftviQA1B/btvy1QHnxvQ=", + "dev": true, + "requires": { + "websocket-driver": ">=0.5.1" + } + }, + "fb-watchman": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/fb-watchman/-/fb-watchman-2.0.1.tgz", + "integrity": "sha512-DkPJKQeY6kKwmuMretBhr7G6Vodr7bFwDYTXIkfG1gjvNpaxBTQV3PbXg6bR1c1UP4jPOX0jHUbbHANL9vRjVg==", + "dev": true, + "requires": { + "bser": "2.1.1" + } + }, + "fecha": { + "version": "4.2.0", + "resolved": "https://registry.npmjs.org/fecha/-/fecha-4.2.0.tgz", + "integrity": "sha512-aN3pcx/DSmtyoovUudctc8+6Hl4T+hI9GBBHLjA76jdZl7+b1sgh5g4k+u/GL3dTy1/pnYzKp69FpJ0OicE3Wg==", + "dev": true + }, + "figgy-pudding": { + "version": "3.5.2", + "resolved": "https://registry.npmjs.org/figgy-pudding/-/figgy-pudding-3.5.2.tgz", + "integrity": "sha512-0btnI/H8f2pavGMN8w40mlSKOfTK2SVJmBfBeVIj3kNw0swwgzyRq0d5TJVOwodFmtvpPeWPN/MCcfuWF0Ezbw==", + "dev": true + }, + "figures": { + "version": "3.2.0", + "resolved": "https://registry.npmjs.org/figures/-/figures-3.2.0.tgz", + "integrity": "sha512-yaduQFRKLXYOGgEn6AZau90j3ggSOyiqXU0F9JZfeXYhNa+Jk4X+s45A2zg5jns87GAFa34BBm2kXw4XpNcbdg==", + "dev": true, + "requires": { + "escape-string-regexp": "^1.0.5" + } + }, + "file-entry-cache": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/file-entry-cache/-/file-entry-cache-5.0.1.tgz", + "integrity": "sha512-bCg29ictuBaKUwwArK4ouCaqDgLZcysCFLmM/Yn/FDoqndh/9vNuQfXRDvTuXKLxfD/JtZQGKFT8MGcJBK644g==", + "dev": true, + "requires": { + "flat-cache": "^2.0.1" + } + }, + "file-loader": { + "version": "6.0.0", + "resolved": "https://registry.npmjs.org/file-loader/-/file-loader-6.0.0.tgz", + "integrity": "sha512-/aMOAYEFXDdjG0wytpTL5YQLfZnnTmLNjn+AIrJ/6HVnTfDqLsVKUUwkDf4I4kgex36BvjuXEn/TX9B/1ESyqQ==", + "dev": true, + "requires": { + "loader-utils": "^2.0.0", + "schema-utils": "^2.6.5" + } + }, + "filelist": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/filelist/-/filelist-1.0.1.tgz", + "integrity": "sha512-8zSK6Nu0DQIC08mUC46sWGXi+q3GGpKydAG36k+JDba6VRpkevvOWUW5a/PhShij4+vHT9M+ghgG7eM+a9JDUQ==", + "dev": true, + "requires": { + "minimatch": "^3.0.4" + } + }, + "filesize": { + "version": "3.6.1", + "resolved": "https://registry.npmjs.org/filesize/-/filesize-3.6.1.tgz", + "integrity": "sha512-7KjR1vv6qnicaPMi1iiTcI85CyYwRO/PSFCu6SvqL8jN2Wjt/NIYQTFtFs7fSDCYOstUkEWIQGFUg5YZQfjlcg==", + "dev": true + }, + "fill-range": { + "version": "7.0.1", + "resolved": "https://registry.npmjs.org/fill-range/-/fill-range-7.0.1.tgz", + "integrity": "sha512-qOo9F+dMUmC2Lcb4BbVvnKJxTPjCm+RRpe4gDuGrzkL7mEVl/djYSu2OdQ2Pa302N4oqkSg9ir6jaLWJ2USVpQ==", + "dev": true, + "requires": { + "to-regex-range": "^5.0.1" + } + }, + "finalhandler": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/finalhandler/-/finalhandler-1.1.0.tgz", + "integrity": "sha1-zgtoVbRYU+eRsvzGgARtiCU91/U=", + "dev": true, + "requires": { + "debug": "2.6.9", + "encodeurl": "~1.0.1", + "escape-html": "~1.0.3", + "on-finished": "~2.3.0", + "parseurl": "~1.3.2", + "statuses": "~1.3.1", + "unpipe": "~1.0.0" + }, + "dependencies": { + "debug": { + "version": "2.6.9", + "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", + "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", + "dev": true, + "requires": { + "ms": "2.0.0" + } + }, + "ms": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", + "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=", + "dev": true + } + } + }, + "find-cache-dir": { + "version": "3.3.1", + "resolved": "https://registry.npmjs.org/find-cache-dir/-/find-cache-dir-3.3.1.tgz", + "integrity": "sha512-t2GDMt3oGC/v+BMwzmllWDuJF/xcDtE5j/fCGbqDD7OLuJkj0cfh1YSA5VKPvwMeLFLNDBkwOKZ2X85jGLVftQ==", + "dev": true, + "requires": { + "commondir": "^1.0.1", + "make-dir": "^3.0.2", + "pkg-dir": "^4.1.0" + } + }, + "find-up": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/find-up/-/find-up-4.1.0.tgz", + "integrity": "sha512-PpOwAdQ/YlXQ2vj8a3h8IipDuYRi3wceVQQGYWxNINccq40Anw7BlsEXCMbt1Zt+OLA6Fq9suIpIWD0OsnISlw==", + "requires": { + "locate-path": "^5.0.0", + "path-exists": "^4.0.0" + } + }, + "find-versions": { + "version": "3.2.0", + "resolved": "https://registry.npmjs.org/find-versions/-/find-versions-3.2.0.tgz", + "integrity": "sha512-P8WRou2S+oe222TOCHitLy8zj+SIsVJh52VP4lvXkaFVnOFFdoWv1H1Jjvel1aI6NCFOAaeAVm8qrI0odiLcww==", + "dev": true, + "requires": { + "semver-regex": "^2.0.0" + } + }, + "findup-sync": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/findup-sync/-/findup-sync-3.0.0.tgz", + "integrity": "sha512-YbffarhcicEhOrm4CtrwdKBdCuz576RLdhJDsIfvNtxUuhdRet1qZcsMjqbePtAseKdAnDyM/IyXbu7PRPRLYg==", + "dev": true, + "requires": { + "detect-file": "^1.0.0", + "is-glob": "^4.0.0", + "micromatch": "^3.0.4", + "resolve-dir": "^1.0.1" + } + }, + "first-chunk-stream": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/first-chunk-stream/-/first-chunk-stream-2.0.0.tgz", + "integrity": "sha1-G97NuOCDwGZLkZRVgVd6Q6nzHXA=", + "dev": true, + "requires": { + "readable-stream": "^2.0.2" + } + }, + "flat-cache": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/flat-cache/-/flat-cache-2.0.1.tgz", + "integrity": "sha512-LoQe6yDuUMDzQAEH8sgmh4Md6oZnc/7PjtwjNFSzveXqSHt6ka9fPBuso7IGf9Rz4uqnSnWiFH2B/zj24a5ReA==", + "dev": true, + "requires": { + "flatted": "^2.0.0", + "rimraf": "2.6.3", + "write": "1.0.3" + }, + "dependencies": { + "glob": { + "version": "7.1.6", + "resolved": "https://registry.npmjs.org/glob/-/glob-7.1.6.tgz", + "integrity": "sha512-LwaxwyZ72Lk7vZINtNNrywX0ZuLyStrdDtabefZKAY5ZGJhVtgdznluResxNmPitE0SAO+O26sWTHeKSI2wMBA==", + "dev": true, + "requires": { + "fs.realpath": "^1.0.0", + "inflight": "^1.0.4", + "inherits": "2", + "minimatch": "^3.0.4", + "once": "^1.3.0", + "path-is-absolute": "^1.0.0" + } + }, + "rimraf": { + "version": "2.6.3", + "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-2.6.3.tgz", + "integrity": "sha512-mwqeW5XsA2qAejG46gYdENaxXjx9onRNCfn7L0duuP4hCuTIi/QO7PDK07KJfp1d+izWPrzEJDcSqBa0OZQriA==", + "dev": true, + "requires": { + "glob": "^7.1.3" + } + } + } + }, + "flatted": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/flatted/-/flatted-2.0.2.tgz", + "integrity": "sha512-r5wGx7YeOwNWNlCA0wQ86zKyDLMQr+/RB8xy74M4hTphfmjlijTSSXGuH8rnvKZnfT9i+75zmd8jcKdMR4O6jA==", + "dev": true + }, + "flush-write-stream": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/flush-write-stream/-/flush-write-stream-1.1.1.tgz", + "integrity": "sha512-3Z4XhFZ3992uIq0XOqb9AreonueSYphE6oYbpt5+3u06JWklbsPkNv3ZKkP9Bz/r+1MWCaMoSQ28P85+1Yc77w==", + "dev": true, + "requires": { + "inherits": "^2.0.3", + "readable-stream": "^2.3.6" + } + }, + "fn-name": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/fn-name/-/fn-name-2.0.1.tgz", + "integrity": "sha1-UhTXU3pNBqSjAcDMJi/rhBiAAuc=", + "dev": true + }, + "follow-redirects": { + "version": "1.5.10", + "resolved": "https://registry.npmjs.org/follow-redirects/-/follow-redirects-1.5.10.tgz", + "integrity": "sha512-0V5l4Cizzvqt5D44aTXbFZz+FtyXV1vrDN6qrelxtfYQKW0KO0W2T/hkE8xvGa/540LkZlkaUjO4ailYTFtHVQ==", + "dev": true, + "requires": { + "debug": "=3.1.0" + }, + "dependencies": { + "debug": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/debug/-/debug-3.1.0.tgz", + "integrity": "sha512-OX8XqP7/1a9cqkxYw2yXss15f26NKWBpDXQd0/uK/KPqdQhxbPa994hnzjcE2VqQpDslf55723cKPUOGSmMY3g==", + "dev": true, + "requires": { + "ms": "2.0.0" + } + }, + "ms": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", + "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=", + "dev": true + } + } + }, + "for-in": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/for-in/-/for-in-1.0.2.tgz", + "integrity": "sha1-gQaNKVqBQuwKxybG4iAMMPttXoA=", + "dev": true + }, + "forever-agent": { + "version": "0.6.1", + "resolved": "https://registry.npmjs.org/forever-agent/-/forever-agent-0.6.1.tgz", + "integrity": "sha1-+8cfDEGt6zf5bFd60e1C2P2sypE=", + "dev": true + }, + "form-data": { + "version": "2.3.3", + "resolved": "https://registry.npmjs.org/form-data/-/form-data-2.3.3.tgz", + "integrity": "sha512-1lLKB2Mu3aGP1Q/2eCOx0fNbRMe7XdwktwOruhfqqd0rIJWwN4Dh+E3hrPSlDCXnSR7UtZ1N38rVXm+6+MEhJQ==", + "dev": true, + "requires": { + "asynckit": "^0.4.0", + "combined-stream": "^1.0.6", + "mime-types": "^2.1.12" + } + }, + "forwarded": { + "version": "0.1.2", + "resolved": "https://registry.npmjs.org/forwarded/-/forwarded-0.1.2.tgz", + "integrity": "sha1-mMI9qxF1ZXuMBXPozszZGw/xjIQ=", + "dev": true + }, + "fragment-cache": { + "version": "0.2.1", + "resolved": "https://registry.npmjs.org/fragment-cache/-/fragment-cache-0.2.1.tgz", + "integrity": "sha1-QpD60n8T6Jvn8zeZxrxaCr//DRk=", + "dev": true, + "requires": { + "map-cache": "^0.2.2" + } + }, + "fresh": { + "version": "0.5.2", + "resolved": "https://registry.npmjs.org/fresh/-/fresh-0.5.2.tgz", + "integrity": "sha1-PYyt2Q2XZWn6g1qx+OSyOhBWBac=", + "dev": true + }, + "friendly-errors-webpack-plugin": { + "version": "1.7.0", + "resolved": "https://registry.npmjs.org/friendly-errors-webpack-plugin/-/friendly-errors-webpack-plugin-1.7.0.tgz", + "integrity": "sha512-K27M3VK30wVoOarP651zDmb93R9zF28usW4ocaK3mfQeIEI5BPht/EzZs5E8QLLwbLRJQMwscAjDxYPb1FuNiw==", + "dev": true, + "requires": { + "chalk": "^1.1.3", + "error-stack-parser": "^2.0.0", + "string-width": "^2.0.0" + }, + "dependencies": { + "ansi-regex": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-2.1.1.tgz", + "integrity": "sha1-w7M6te42DYbg5ijwRorn7yfWVN8=", + "dev": true + }, + "ansi-styles": { + "version": "2.2.1", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-2.2.1.tgz", + "integrity": "sha1-tDLdM1i2NM914eRmQ2gkBTPB3b4=", + "dev": true + }, + "chalk": { + "version": "1.1.3", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-1.1.3.tgz", + "integrity": "sha1-qBFcVeSnAv5NFQq9OHKCKn4J/Jg=", + "dev": true, + "requires": { + "ansi-styles": "^2.2.1", + "escape-string-regexp": "^1.0.2", + "has-ansi": "^2.0.0", + "strip-ansi": "^3.0.0", + "supports-color": "^2.0.0" + } + }, + "is-fullwidth-code-point": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-2.0.0.tgz", + "integrity": "sha1-o7MKXE8ZkYMWeqq5O+764937ZU8=", + "dev": true + }, + "string-width": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/string-width/-/string-width-2.1.1.tgz", + "integrity": "sha512-nOqH59deCq9SRHlxq1Aw85Jnt4w6KvLKqWVik6oA9ZklXLNIOlqg4F2yrT1MVaTjAqvVwdfeZ7w7aCvJD7ugkw==", + "dev": true, + "requires": { + "is-fullwidth-code-point": "^2.0.0", + "strip-ansi": "^4.0.0" + }, + "dependencies": { + "ansi-regex": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-3.0.0.tgz", + "integrity": "sha1-7QMXwyIGT3lGbAKWa922Bas32Zg=", + "dev": true + }, + "strip-ansi": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-4.0.0.tgz", + "integrity": "sha1-qEeQIusaw2iocTibY1JixQXuNo8=", + "dev": true, + "requires": { + "ansi-regex": "^3.0.0" + } + } + } + }, + "strip-ansi": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-3.0.1.tgz", + "integrity": "sha1-ajhfuIU9lS1f8F0Oiq+UJ43GPc8=", + "dev": true, + "requires": { + "ansi-regex": "^2.0.0" + } + }, + "supports-color": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-2.0.0.tgz", + "integrity": "sha1-U10EXOa2Nj+kARcIRimZXp3zJMc=", + "dev": true + } + } + }, + "from2": { + "version": "2.3.0", + "resolved": "https://registry.npmjs.org/from2/-/from2-2.3.0.tgz", + "integrity": "sha1-i/tVAr3kpNNs/e6gB/zKIdfjgq8=", + "dev": true, + "requires": { + "inherits": "^2.0.1", + "readable-stream": "^2.0.0" + } + }, + "fs-extra": { + "version": "4.0.2", + "resolved": "https://registry.npmjs.org/fs-extra/-/fs-extra-4.0.2.tgz", + "integrity": "sha1-+RcExT0bRh+JNFKwwwfZmXZHq2s=", + "dev": true, + "requires": { + "graceful-fs": "^4.1.2", + "jsonfile": "^4.0.0", + "universalify": "^0.1.0" + } + }, + "fs-minipass": { + "version": "1.2.7", + "resolved": "https://registry.npmjs.org/fs-minipass/-/fs-minipass-1.2.7.tgz", + "integrity": "sha512-GWSSJGFy4e9GUeCcbIkED+bgAoFyj7XF1mV8rma3QW4NIqX9Kyx79N/PF61H5udOV3aY1IaMLs6pGbH71nlCTA==", + "dev": true, + "requires": { + "minipass": "^2.6.0" + } + }, + "fs-write-stream-atomic": { + "version": "1.0.10", + "resolved": "https://registry.npmjs.org/fs-write-stream-atomic/-/fs-write-stream-atomic-1.0.10.tgz", + "integrity": "sha1-tH31NJPvkR33VzHnCp3tAYnbQMk=", + "dev": true, + "requires": { + "graceful-fs": "^4.1.2", + "iferr": "^0.1.5", + "imurmurhash": "^0.1.4", + "readable-stream": "1 || 2" + } + }, + "fs.realpath": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/fs.realpath/-/fs.realpath-1.0.0.tgz", + "integrity": "sha1-FQStJSMVjKpA20onh8sBQRmU6k8=" + }, + "fsevents": { + "version": "2.1.3", + "resolved": "https://registry.npmjs.org/fsevents/-/fsevents-2.1.3.tgz", + "integrity": "sha512-Auw9a4AxqWpa9GUfj370BMPzzyncfBABW8Mab7BGWBYDj4Isgq+cDKtx0i6u9jcX9pQDnswsaaOTgTmA5pEjuQ==", + "dev": true, + "optional": true + }, + "function-bind": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/function-bind/-/function-bind-1.1.1.tgz", + "integrity": "sha512-yIovAzMX49sF8Yl58fSCWJ5svSLuaibPxXQJFLmBObTuCr0Mf1KiPopGM9NiFjiYBCbfaa2Fh6breQ6ANVTI0A==" + }, + "functional-red-black-tree": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/functional-red-black-tree/-/functional-red-black-tree-1.0.1.tgz", + "integrity": "sha1-GwqzvVU7Kg1jmdKcDj6gslIHgyc=", + "dev": true + }, + "g-status": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/g-status/-/g-status-2.0.2.tgz", + "integrity": "sha512-kQoE9qH+T1AHKgSSD0Hkv98bobE90ILQcXAF4wvGgsr7uFqNvwmh8j+Lq3l0RVt3E3HjSbv2B9biEGcEtpHLCA==", + "dev": true, + "requires": { + "arrify": "^1.0.1", + "matcher": "^1.0.0", + "simple-git": "^1.85.0" + }, + "dependencies": { + "arrify": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/arrify/-/arrify-1.0.1.tgz", + "integrity": "sha1-iYUI2iIm84DfkEcoRWhJwVAaSw0=", + "dev": true + } + } + }, + "gauge": { + "version": "1.2.7", + "resolved": "https://registry.npmjs.org/gauge/-/gauge-1.2.7.tgz", + "integrity": "sha1-6c7FSD09TuDvRLYKfZnkk14TbZM=", + "dev": true, + "requires": { + "ansi": "^0.3.0", + "has-unicode": "^2.0.0", + "lodash.pad": "^4.1.0", + "lodash.padend": "^4.1.0", + "lodash.padstart": "^4.1.0" + } + }, + "generator-jhipster": { + "version": "6.10.4", + "resolved": "https://registry.npmjs.org/generator-jhipster/-/generator-jhipster-6.10.4.tgz", + "integrity": "sha512-AuPNTXucH8ekryRtVbiX+EC64jIHozxq7A+hTC/rhEgQyaMMqish+CpPGYaNLgzT2oqAvXFLWFndsiBJ4YdjVw==", + "dev": true, + "requires": { + "aws-sdk": "2.706.0", + "axios": "0.19.2", + "chalk": "4.1.0", + "commander": "5.1.0", + "conf": "6.2.4", + "didyoumean": "1.2.1", + "ejs": "3.1.3", + "faker": "4.1.0", + "glob": "7.1.6", + "gulp-filter": "6.0.0", + "insight": "0.10.3", + "jhipster-core": "7.3.4", + "js-object-pretty-print": "0.3.0", + "js-yaml": "3.14.0", + "lodash": "4.17.15", + "meow": "6.1.0", + "mkdirp": "1.0.4", + "normalize-path": "3.0.0", + "ora": "4.0.4", + "os-locale": "5.0.0", + "parse-gitignore": "1.0.1", + "pluralize": "8.0.0", + "prettier": "2.1.2", + "prettier-plugin-java": "0.8.2", + "progress": "2.0.3", + "randexp": "0.5.3", + "semver": "7.3.2", + "shelljs": "0.8.4", + "tabtab": "2.2.2", + "test": "^0.6.0", + "then-request": "6.0.2", + "through2": "3.0.2", + "uuid": "7.0.3", + "yeoman-environment": "2.10.3", + "yeoman-generator": "4.11.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" + } + }, + "axios": { + "version": "0.19.2", + "resolved": "https://registry.npmjs.org/axios/-/axios-0.19.2.tgz", + "integrity": "sha512-fjgm5MvRHLhx+osE2xoekY70AhARk3a6hkN+3Io1jc00jtquGvxYlKlsFUhmUET0V5te6CcZI7lcv2Ym61mjHA==", + "dev": true, + "requires": { + "follow-redirects": "1.5.10" + } + }, + "chalk": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.0.tgz", + "integrity": "sha512-qwx12AxXe2Q5xQ43Ac//I6v5aXTipYrSESdOgzrN+9XjgEpyjpKuvSGaN4qE93f7TQTlerQQ8S+EQ0EyDoVL1A==", + "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 + }, + "commander": { + "version": "5.1.0", + "resolved": "https://registry.npmjs.org/commander/-/commander-5.1.0.tgz", + "integrity": "sha512-P0CysNDQ7rtVw4QIQtm+MRxV66vKFSvlsQvGYXZWR3qFU0jlMKHZZZgw8e+8DSah4UDKMqnknRDQz+xuQXQ/Zg==", + "dev": 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" + } + }, + "execa": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/execa/-/execa-4.1.0.tgz", + "integrity": "sha512-j5W0//W7f8UxAn8hXVnwG8tLwdiUy4FJLcSupCg6maBYZDpyBvTApK7KyuI4bKj8KOh1r2YH+6ucuYtJv1bTZA==", + "dev": true, + "requires": { + "cross-spawn": "^7.0.0", + "get-stream": "^5.0.0", + "human-signals": "^1.1.1", + "is-stream": "^2.0.0", + "merge-stream": "^2.0.0", + "npm-run-path": "^4.0.0", + "onetime": "^5.1.0", + "signal-exit": "^3.0.2", + "strip-final-newline": "^2.0.0" + } + }, + "get-stream": { + "version": "5.2.0", + "resolved": "https://registry.npmjs.org/get-stream/-/get-stream-5.2.0.tgz", + "integrity": "sha512-nBF+F1rAZVCu/p7rjzgA+Yb4lfYXrpl7a6VmJrU8wF9I1CKvP/QwPNZHnOlwbTkY6dvtFIzFMSyQXbLoTQPRpA==", + "dev": true, + "requires": { + "pump": "^3.0.0" + } + }, + "glob": { + "version": "7.1.6", + "resolved": "https://registry.npmjs.org/glob/-/glob-7.1.6.tgz", + "integrity": "sha512-LwaxwyZ72Lk7vZINtNNrywX0ZuLyStrdDtabefZKAY5ZGJhVtgdznluResxNmPitE0SAO+O26sWTHeKSI2wMBA==", + "dev": true, + "requires": { + "fs.realpath": "^1.0.0", + "inflight": "^1.0.4", + "inherits": "2", + "minimatch": "^3.0.4", + "once": "^1.3.0", + "path-is-absolute": "^1.0.0" + } + }, + "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 + }, + "invert-kv": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/invert-kv/-/invert-kv-3.0.1.tgz", + "integrity": "sha512-CYdFeFexxhv/Bcny+Q0BfOV+ltRlJcd4BBZBYFX/O0u4npJrgZtIcjokegtiSMAvlMTJ+Koq0GBCc//3bueQxw==", + "dev": true + }, + "is-stream": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/is-stream/-/is-stream-2.0.0.tgz", + "integrity": "sha512-XCoy+WlUr7d1+Z8GgSuXmpuUFC9fOhRXglJMx+dwLKTkL44Cjd4W1Z5P+BQZpr+cR93aGP4S/s7Ftw6Nd/kiEw==", + "dev": true + }, + "lcid": { + "version": "3.1.1", + "resolved": "https://registry.npmjs.org/lcid/-/lcid-3.1.1.tgz", + "integrity": "sha512-M6T051+5QCGLBQb8id3hdvIW8+zeFV2FyBGFS9IEK5H9Wt4MueD4bW1eWikpHgZp+5xR3l5c8pZUkQsIA0BFZg==", + "dev": true, + "requires": { + "invert-kv": "^3.0.0" + } + }, + "lodash": { + "version": "4.17.15", + "resolved": "https://registry.npmjs.org/lodash/-/lodash-4.17.15.tgz", + "integrity": "sha512-8xOcRHvCjnocdS5cpwXQXVzmmh5e5+saE2QGoeQmbKmRS6J3VQppPOIt0MnmE+4xlZoumy0GPG0D0MVIQbNA1A==", + "dev": true + }, + "mkdirp": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/mkdirp/-/mkdirp-1.0.4.tgz", + "integrity": "sha512-vVqVZQyf3WLx2Shd0qJ9xuvqgAyKPLAiqITEtqW0oIUjzo3PePDd6fW9iFz30ef7Ysp/oiWqbhszeGWW2T6Gzw==", + "dev": true + }, + "npm-run-path": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/npm-run-path/-/npm-run-path-4.0.1.tgz", + "integrity": "sha512-S48WzZW777zhNIrn7gxOlISNAqi9ZC/uQFnRdbeIHhZhCA6UqpkOT8T1G7BvfdgP4Er8gF4sUbaS0i7QvIfCWw==", + "dev": true, + "requires": { + "path-key": "^3.0.0" + } + }, + "os-locale": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/os-locale/-/os-locale-5.0.0.tgz", + "integrity": "sha512-tqZcNEDAIZKBEPnHPlVDvKrp7NzgLi7jRmhKiUoa2NUmhl13FtkAGLUVR+ZsYvApBQdBfYm43A4tXXQ4IrYLBA==", + "dev": true, + "requires": { + "execa": "^4.0.0", + "lcid": "^3.0.0", + "mem": "^5.0.0" + } + }, + "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 + }, + "semver": { + "version": "7.3.2", + "resolved": "https://registry.npmjs.org/semver/-/semver-7.3.2.tgz", + "integrity": "sha512-OrOb32TeeambH6UrhtShmF7CRDqhL6/5XpPNp2DuRH6+9QLw/orhp72j87v8Qa1ScDkvrrBNpZcDejAirJmfXQ==", + "dev": true + }, + "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 + }, + "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" + } + }, + "through2": { + "version": "3.0.2", + "resolved": "https://registry.npmjs.org/through2/-/through2-3.0.2.tgz", + "integrity": "sha512-enaDQ4MUyP2W6ZyT6EsMzqBPZaM/avg8iuo+l2d3QCs0J+6RaqkHV/2/lOwDTueBHeJ/2LG9lrLW3d5rWPucuQ==", + "dev": true, + "requires": { + "inherits": "^2.0.4", + "readable-stream": "2 || 3" + } + }, + "uuid": { + "version": "7.0.3", + "resolved": "https://registry.npmjs.org/uuid/-/uuid-7.0.3.tgz", + "integrity": "sha512-DPSke0pXhTZgoF/d+WSt2QaKMCFSfx7QegxEWT+JOuHF5aWrKEn0G+ztjuJg/gG8/ItK+rbPCD/yNv8yyih6Cg==", + "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" + } + } + } + }, + "genfun": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/genfun/-/genfun-5.0.0.tgz", + "integrity": "sha512-KGDOARWVga7+rnB3z9Sd2Letx515owfk0hSxHGuqjANb1M+x2bGZGqHLiozPsYMdM2OubeMni/Hpwmjq6qIUhA==", + "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==" + }, + "get-caller-file": { + "version": "2.0.5", + "resolved": "https://registry.npmjs.org/get-caller-file/-/get-caller-file-2.0.5.tgz", + "integrity": "sha512-DyFP3BM/3YHTQOCUL/w0OZHR0lpKeGrxotcHWcqNEdnltqFwXVfhEBQ94eIo34AfQpo0rGki4cyIiftY06h2Fg==" + }, + "get-intrinsic": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/get-intrinsic/-/get-intrinsic-1.0.1.tgz", + "integrity": "sha512-ZnWP+AmS1VUaLgTRy47+zKtjTxz+0xMpx3I52i+aalBK1QP19ggLF3Db89KJX7kjfOfP2eoa01qc++GwPgufPg==", + "dev": true, + "requires": { + "function-bind": "^1.1.1", + "has": "^1.0.3", + "has-symbols": "^1.0.1" + } + }, + "get-own-enumerable-property-symbols": { + "version": "3.0.2", + "resolved": "https://registry.npmjs.org/get-own-enumerable-property-symbols/-/get-own-enumerable-property-symbols-3.0.2.tgz", + "integrity": "sha512-I0UBV/XOz1XkIJHEUDMZAbzCThU/H8DxmSfmdGcKPnVhu2VfFqr34jr9777IyaTYvxjedWhqVIilEDsCdP5G6g==", + "dev": true + }, + "get-package-type": { + "version": "0.1.0", + "resolved": "https://registry.npmjs.org/get-package-type/-/get-package-type-0.1.0.tgz", + "integrity": "sha512-pjzuKtY64GYfWizNAJ0fr9VqttZkNiK2iS430LtIHzjBEr6bX8Am2zm4sW4Ro5wjWW5cAlRL1qAMTcXbjNAO2Q==", + "dev": true + }, + "get-stdin": { + "version": "6.0.0", + "resolved": "https://registry.npmjs.org/get-stdin/-/get-stdin-6.0.0.tgz", + "integrity": "sha512-jp4tHawyV7+fkkSKyvjuLZswblUtz+SQKzSWnBbii16BuZksJlU1wuBYXY75r+duh/llF1ur6oNwi+2ZzjKZ7g==", + "dev": true + }, + "get-stream": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/get-stream/-/get-stream-4.1.0.tgz", + "integrity": "sha512-GMat4EJ5161kIy2HevLlr4luNjBgvmj413KaQA7jt4V8B4RDsfpHk7WQ9GVqfYyyx8OS/L66Kox+rJRNklLK7w==", + "dev": true, + "requires": { + "pump": "^3.0.0" + } + }, + "get-value": { + "version": "2.0.6", + "resolved": "https://registry.npmjs.org/get-value/-/get-value-2.0.6.tgz", + "integrity": "sha1-3BXKHGcjh8p2vTesCjlbogQqLCg=", + "dev": true + }, + "getpass": { + "version": "0.1.7", + "resolved": "https://registry.npmjs.org/getpass/-/getpass-0.1.7.tgz", + "integrity": "sha1-Xv+OPmhNVprkyysSgmBOi6YhSfo=", + "dev": true, + "requires": { + "assert-plus": "^1.0.0" + } + }, + "gh-got": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/gh-got/-/gh-got-5.0.0.tgz", + "integrity": "sha1-7pW+NxBv2HSKlvjR20uuqJ4b+oo=", + "dev": true, + "requires": { + "got": "^6.2.0", + "is-plain-obj": "^1.1.0" + } + }, + "github-username": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/github-username/-/github-username-3.0.0.tgz", + "integrity": "sha1-CnciGbMTB0NCnyRW0L3T21Xc57E=", + "dev": true, + "requires": { + "gh-got": "^5.0.0" + } + }, + "glob": { + "version": "7.1.2", + "resolved": "https://registry.npmjs.org/glob/-/glob-7.1.2.tgz", + "integrity": "sha512-MJTUg1kjuLeQCJ+ccE4Vpa6kKVXkPYJ2mOCQyUuKLcLQsdrMCpBPUi8qVE6+YuaJkozeA9NusTAw3hLr8Xe5EQ==", + "requires": { + "fs.realpath": "^1.0.0", + "inflight": "^1.0.4", + "inherits": "2", + "minimatch": "^3.0.4", + "once": "^1.3.0", + "path-is-absolute": "^1.0.0" + } + }, + "glob-parent": { + "version": "5.1.1", + "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-5.1.1.tgz", + "integrity": "sha512-FnI+VGOpnlGHWZxthPGR+QhR78fuiK0sNLkHQv+bL9fQi57lNNdquIbna/WrfROrolq8GK5Ek6BiMwqL/voRYQ==", + "dev": true, + "requires": { + "is-glob": "^4.0.1" + } + }, + "glob-to-regexp": { + "version": "0.3.0", + "resolved": "https://registry.npmjs.org/glob-to-regexp/-/glob-to-regexp-0.3.0.tgz", + "integrity": "sha1-jFoUlNIGbFcMw7/kSWF1rMTVAqs=", + "dev": true + }, + "global-modules": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/global-modules/-/global-modules-2.0.0.tgz", + "integrity": "sha512-NGbfmJBp9x8IxyJSd1P+otYK8vonoJactOogrVfFRIAEY1ukil8RSKDz2Yo7wh1oihl51l/r6W4epkeKJHqL8A==", + "dev": true, + "requires": { + "global-prefix": "^3.0.0" + }, + "dependencies": { + "global-prefix": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/global-prefix/-/global-prefix-3.0.0.tgz", + "integrity": "sha512-awConJSVCHVGND6x3tmMaKcQvwXLhjdkmomy2W+Goaui8YPgYgXJZewhg3fWC+DlfqqQuWg8AwqjGTD2nAPVWg==", + "dev": true, + "requires": { + "ini": "^1.3.5", + "kind-of": "^6.0.2", + "which": "^1.3.1" + } + } + } + }, + "global-prefix": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/global-prefix/-/global-prefix-1.0.2.tgz", + "integrity": "sha1-2/dDxsFJklk8ZVVoy2btMsASLr4=", + "dev": true, + "requires": { + "expand-tilde": "^2.0.2", + "homedir-polyfill": "^1.0.1", + "ini": "^1.3.4", + "is-windows": "^1.0.1", + "which": "^1.2.14" + } + }, + "globals": { + "version": "11.12.0", + "resolved": "https://registry.npmjs.org/globals/-/globals-11.12.0.tgz", + "integrity": "sha512-WOBp/EEGUiIsJSp7wcv/y6MO+lV9UoncWqxuFfm8eBwzWNgyfBd6Gz+IeKQ9jCmyhoH99g15M3T+QaVHFjizVA==" + }, + "globby": { + "version": "11.0.1", + "resolved": "https://registry.npmjs.org/globby/-/globby-11.0.1.tgz", + "integrity": "sha512-iH9RmgwCmUJHi2z5o2l3eTtGBtXek1OYlHrbcxOYugyHLmAsZrPj43OtHThd62Buh/Vv6VyCBD2bdyWcGNQqoQ==", + "dev": true, + "requires": { + "array-union": "^2.1.0", + "dir-glob": "^3.0.1", + "fast-glob": "^3.1.1", + "ignore": "^5.1.4", + "merge2": "^1.3.0", + "slash": "^3.0.0" + } + }, + "got": { + "version": "6.7.1", + "resolved": "https://registry.npmjs.org/got/-/got-6.7.1.tgz", + "integrity": "sha1-JAzQV4WpoY5WHcG0S0HHY+8ejbA=", + "dev": true, + "requires": { + "create-error-class": "^3.0.0", + "duplexer3": "^0.1.4", + "get-stream": "^3.0.0", + "is-redirect": "^1.0.0", + "is-retry-allowed": "^1.0.0", + "is-stream": "^1.0.0", + "lowercase-keys": "^1.0.0", + "safe-buffer": "^5.0.1", + "timed-out": "^4.0.0", + "unzip-response": "^2.0.1", + "url-parse-lax": "^1.0.0" + }, + "dependencies": { + "get-stream": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/get-stream/-/get-stream-3.0.0.tgz", + "integrity": "sha1-jpQ9E1jcN1VQVOy+LtsFqhdO3hQ=", + "dev": true + } + } + }, + "graceful-fs": { + "version": "4.2.4", + "resolved": "https://registry.npmjs.org/graceful-fs/-/graceful-fs-4.2.4.tgz", + "integrity": "sha512-WjKPNJF79dtJAVniUlGGWHYGz2jWxT6VhN/4m1NdkbZ2nOsEF+cI1Edgql5zCRhs/VsQYRvrXctxktVXZUkixw==", + "dev": true + }, + "grouped-queue": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/grouped-queue/-/grouped-queue-1.1.0.tgz", + "integrity": "sha512-rZOFKfCqLhsu5VqjBjEWiwrYqJR07KxIkH4mLZlNlGDfntbb4FbMyGFP14TlvRPrU9S3Hnn/sgxbC5ZeN0no3Q==", + "dev": true, + "requires": { + "lodash": "^4.17.15" + } + }, + "growly": { + "version": "1.3.0", + "resolved": "https://registry.npmjs.org/growly/-/growly-1.3.0.tgz", + "integrity": "sha1-8QdIy+dq+WS3yWyTxrzCivEgwIE=", + "dev": true + }, + "gulp-filter": { + "version": "6.0.0", + "resolved": "https://registry.npmjs.org/gulp-filter/-/gulp-filter-6.0.0.tgz", + "integrity": "sha512-veQFW93kf6jBdWdF/RxMEIlDK2mkjHyPftM381DID2C9ImTVngwYpyyThxm4/EpgcNOT37BLefzMOjEKbyYg0Q==", + "dev": true, + "requires": { + "multimatch": "^4.0.0", + "plugin-error": "^1.0.1", + "streamfilter": "^3.0.0" + } + }, + "gzip-size": { + "version": "5.1.1", + "resolved": "https://registry.npmjs.org/gzip-size/-/gzip-size-5.1.1.tgz", + "integrity": "sha512-FNHi6mmoHvs1mxZAds4PpdCS6QG8B4C1krxJsMutgxl5t3+GlRTzzI3NEkifXx2pVsOvJdOGSmIgDhQ55FwdPA==", + "dev": true, + "requires": { + "duplexer": "^0.1.1", + "pify": "^4.0.1" + }, + "dependencies": { + "pify": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/pify/-/pify-4.0.1.tgz", + "integrity": "sha512-uB80kBFb/tfd68bVleG9T5GGsGPjJrLAUpR5PZIrhBnIaRTQRjqdJSsIKkOP6OAIFbj7GOrcudc5pNjZ+geV2g==", + "dev": true + } + } + }, + "handle-thing": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/handle-thing/-/handle-thing-2.0.1.tgz", + "integrity": "sha512-9Qn4yBxelxoh2Ow62nP+Ka/kMnOXRi8BXnRaUwezLNhqelnN49xKz4F/dPP8OYLxLxq6JDtZb2i9XznUQbNPTg==", + "dev": true + }, + "har-schema": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/har-schema/-/har-schema-2.0.0.tgz", + "integrity": "sha1-qUwiJOvKwEeCoNkDVSHyRzW37JI=", + "dev": true + }, + "har-validator": { + "version": "5.1.5", + "resolved": "https://registry.npmjs.org/har-validator/-/har-validator-5.1.5.tgz", + "integrity": "sha512-nmT2T0lljbxdQZfspsno9hgrG3Uir6Ks5afism62poxqBM6sDnMEuPmzTq8XN0OEwqKLLdh1jQI3qyE66Nzb3w==", + "dev": true, + "requires": { + "ajv": "^6.12.3", + "har-schema": "^2.0.0" + }, + "dependencies": { + "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" + } + } + } + }, + "hard-rejection": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/hard-rejection/-/hard-rejection-2.1.0.tgz", + "integrity": "sha512-VIZB+ibDhx7ObhAe7OVtoEbuP4h/MuOTHJ+J8h/eBXotJYl0fBgR72xDFCKgIh22OJZIOVNxBMWuhAr10r8HdA==", + "dev": true + }, + "has": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/has/-/has-1.0.3.tgz", + "integrity": "sha512-f2dvO0VU6Oej7RkWJGrehjbzMAjFp5/VKPp5tTpWIV4JHHZK1/BxbFRtf/siA2SWTe09caDmVtYYzWEIbBS4zw==", + "requires": { + "function-bind": "^1.1.1" + } + }, + "has-ansi": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/has-ansi/-/has-ansi-2.0.0.tgz", + "integrity": "sha1-NPUEnOHs3ysGSa8+8k5F7TVBbZE=", + "dev": true, + "requires": { + "ansi-regex": "^2.0.0" + }, + "dependencies": { + "ansi-regex": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-2.1.1.tgz", + "integrity": "sha1-w7M6te42DYbg5ijwRorn7yfWVN8=", + "dev": true + } + } + }, + "has-binary2": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/has-binary2/-/has-binary2-1.0.3.tgz", + "integrity": "sha512-G1LWKhDSvhGeAQ8mPVQlqNcOB2sJdwATtZKl2pDKKHfpf/rYj24lkinxf69blJbnsvtqqNU+L3SL50vzZhXOnw==", + "dev": true, + "requires": { + "isarray": "2.0.1" + }, + "dependencies": { + "isarray": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/isarray/-/isarray-2.0.1.tgz", + "integrity": "sha1-o32U7ZzaLVmGXJ92/llu4fM4dB4=", + "dev": true + } + } + }, + "has-cors": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/has-cors/-/has-cors-1.1.0.tgz", + "integrity": "sha1-XkdHk/fqmEPRu5nCPu9J/xJv/zk=", + "dev": true + }, + "has-flag": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-3.0.0.tgz", + "integrity": "sha1-tdRU3CGZriJWmfNGfloH87lVuv0=" + }, + "has-symbols": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/has-symbols/-/has-symbols-1.0.1.tgz", + "integrity": "sha512-PLcsoqu++dmEIZB+6totNFKq/7Do+Z0u4oT0zKOJNl3lYK6vGwwu2hjHs+68OEZbTjiUE9bgOABXbP/GvrS0Kg==", + "dev": true + }, + "has-unicode": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/has-unicode/-/has-unicode-2.0.1.tgz", + "integrity": "sha1-4Ob+aijPUROIVeCG0Wkedx3iqLk=", + "dev": true + }, + "has-value": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/has-value/-/has-value-1.0.0.tgz", + "integrity": "sha1-GLKB2lhbHFxR3vJMkw7SmgvmsXc=", + "dev": true, + "requires": { + "get-value": "^2.0.6", + "has-values": "^1.0.0", + "isobject": "^3.0.0" + } + }, + "has-values": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/has-values/-/has-values-1.0.0.tgz", + "integrity": "sha1-lbC2P+whRmGab+V/51Yo1aOe/k8=", + "dev": true, + "requires": { + "is-number": "^3.0.0", + "kind-of": "^4.0.0" + }, + "dependencies": { + "is-number": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/is-number/-/is-number-3.0.0.tgz", + "integrity": "sha1-JP1iAaR4LPUFYcgQJ2r8fRLXEZU=", + "dev": true, + "requires": { + "kind-of": "^3.0.2" + }, + "dependencies": { + "kind-of": { + "version": "3.2.2", + "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-3.2.2.tgz", + "integrity": "sha1-MeohpzS6ubuw8yRm2JOupR5KPGQ=", + "dev": true, + "requires": { + "is-buffer": "^1.1.5" + } + } + } + }, + "kind-of": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-4.0.0.tgz", + "integrity": "sha1-IIE989cSkosgc3hpGkUGb65y3Vc=", + "dev": true, + "requires": { + "is-buffer": "^1.1.5" + } + } + } + }, + "hash-base": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/hash-base/-/hash-base-3.1.0.tgz", + "integrity": "sha512-1nmYp/rhMDiE7AYkDw+lLwlAzz0AntGIe51F3RfFfEqyQ3feY2eI/NcwC6umIQVOASPMsWJLJScWKSSvzL9IVA==", + "dev": true, + "requires": { + "inherits": "^2.0.4", + "readable-stream": "^3.6.0", + "safe-buffer": "^5.2.0" + }, + "dependencies": { + "readable-stream": { + "version": "3.6.0", + "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-3.6.0.tgz", + "integrity": "sha512-BViHy7LKeTz4oNnkcLJ+lVSL6vpiFeX6/d3oSH8zCW7UxP2onchk+vTGB143xuFjHS3deTgkKoXXymXqymiIdA==", + "dev": true, + "requires": { + "inherits": "^2.0.3", + "string_decoder": "^1.1.1", + "util-deprecate": "^1.0.1" + } + }, + "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 + } + } + }, + "hash.js": { + "version": "1.1.7", + "resolved": "https://registry.npmjs.org/hash.js/-/hash.js-1.1.7.tgz", + "integrity": "sha512-taOaskGt4z4SOANNseOviYDvjEJinIkRgmp7LbKP2YTTmVxWBl87s/uzK9r+44BclBSp2X7K1hqeNfz9JbBeXA==", + "dev": true, + "requires": { + "inherits": "^2.0.3", + "minimalistic-assert": "^1.0.1" + } + }, + "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 + }, + "hex-color-regex": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/hex-color-regex/-/hex-color-regex-1.1.0.tgz", + "integrity": "sha512-l9sfDFsuqtOqKDsQdqrMRk0U85RZc0RtOR9yPI7mRVOa4FsR/BVnZ0shmQRM96Ji99kYZP/7hn1cedc1+ApsTQ==", + "dev": true + }, + "hmac-drbg": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/hmac-drbg/-/hmac-drbg-1.0.1.tgz", + "integrity": "sha1-0nRXAQJabHdabFRXk+1QL8DGSaE=", + "dev": true, + "requires": { + "hash.js": "^1.0.3", + "minimalistic-assert": "^1.0.0", + "minimalistic-crypto-utils": "^1.0.1" + } + }, + "homedir-polyfill": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/homedir-polyfill/-/homedir-polyfill-1.0.3.tgz", + "integrity": "sha512-eSmmWE5bZTK2Nou4g0AI3zZ9rswp7GRKoKXS1BLUkvPviOqs4YTN1djQIqrXy9k5gEtdLPy86JjRwsNM9tnDcA==", + "dev": true, + "requires": { + "parse-passwd": "^1.0.0" + } + }, + "hoopy": { + "version": "0.1.4", + "resolved": "https://registry.npmjs.org/hoopy/-/hoopy-0.1.4.tgz", + "integrity": "sha512-HRcs+2mr52W0K+x8RzcLzuPPmVIKMSv97RGHy0Ea9y/mpcaK+xTrjICA04KAHi4GRzxliNqNJEFYWHghy3rSfQ==", + "dev": true + }, + "hosted-git-info": { + "version": "3.0.7", + "resolved": "https://registry.npmjs.org/hosted-git-info/-/hosted-git-info-3.0.7.tgz", + "integrity": "sha512-fWqc0IcuXs+BmE9orLDyVykAG9GJtGLGuZAAqgcckPgv5xad4AcXGIv8galtQvlwutxSlaMcdw7BUtq2EIvqCQ==", + "dev": true, + "requires": { + "lru-cache": "^6.0.0" + } + }, + "hpack.js": { + "version": "2.1.6", + "resolved": "https://registry.npmjs.org/hpack.js/-/hpack.js-2.1.6.tgz", + "integrity": "sha1-h3dMCUnlE/QuhFdbPEVoH63ioLI=", + "dev": true, + "requires": { + "inherits": "^2.0.1", + "obuf": "^1.0.0", + "readable-stream": "^2.0.1", + "wbuf": "^1.1.0" + } + }, + "hsl-regex": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/hsl-regex/-/hsl-regex-1.0.0.tgz", + "integrity": "sha1-1JMwx4ntgZ4nakwNJy3/owsY/m4=", + "dev": true + }, + "hsla-regex": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/hsla-regex/-/hsla-regex-1.0.0.tgz", + "integrity": "sha1-wc56MWjIxmFAM6S194d/OyJfnDg=", + "dev": true + }, + "html-comment-regex": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/html-comment-regex/-/html-comment-regex-1.1.2.tgz", + "integrity": "sha512-P+M65QY2JQ5Y0G9KKdlDpo0zK+/OHptU5AaBwUfAIDJZk1MYf32Frm84EcOytfJE0t5JvkAnKlmjsXDnWzCJmQ==", + "dev": true + }, + "html-encoding-sniffer": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/html-encoding-sniffer/-/html-encoding-sniffer-2.0.1.tgz", + "integrity": "sha512-D5JbOMBIR/TVZkubHT+OyT2705QvogUW4IBn6nHd756OwieSF9aDYFj4dv6HHEVGYbHaLETa3WggZYWWMyy3ZQ==", + "dev": true, + "requires": { + "whatwg-encoding": "^1.0.5" + } + }, + "html-entities": { + "version": "1.3.1", + "resolved": "https://registry.npmjs.org/html-entities/-/html-entities-1.3.1.tgz", + "integrity": "sha512-rhE/4Z3hIhzHAUKbW8jVcCyuT5oJCXXqhN/6mXXVCpzTmvJnoH2HL/bt3EZ6p55jbFJBeAe1ZNpL5BugLujxNA==", + "dev": true + }, + "html-escaper": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/html-escaper/-/html-escaper-2.0.2.tgz", + "integrity": "sha512-H2iMtd0I4Mt5eYiapRdIDjp+XzelXQ0tFE4JS7YFwFevXXMmOp9myNrUvCg0D6ws8iqkRPBfKHgbwig1SmlLfg==", + "dev": true + }, + "html-loader": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/html-loader/-/html-loader-1.1.0.tgz", + "integrity": "sha512-zwLbEgy+i7sgIYTlxI9M7jwkn29IvdsV6f1y7a2aLv/w8l1RigVk0PFijBZLLFsdi2gvL8sf2VJhTjLlfnK8sA==", + "dev": true, + "requires": { + "html-minifier-terser": "^5.0.5", + "htmlparser2": "^4.1.0", + "loader-utils": "^2.0.0", + "parse-srcset": "^1.0.2", + "schema-utils": "^2.6.5" + } + }, + "html-minifier-terser": { + "version": "5.1.1", + "resolved": "https://registry.npmjs.org/html-minifier-terser/-/html-minifier-terser-5.1.1.tgz", + "integrity": "sha512-ZPr5MNObqnV/T9akshPKbVgyOqLmy+Bxo7juKCfTfnjNniTAMdy4hz21YQqoofMBJD2kdREaqPPdThoR78Tgxg==", + "dev": true, + "requires": { + "camel-case": "^4.1.1", + "clean-css": "^4.2.3", + "commander": "^4.1.1", + "he": "^1.2.0", + "param-case": "^3.0.3", + "relateurl": "^0.2.7", + "terser": "^4.6.3" + }, + "dependencies": { + "commander": { + "version": "4.1.1", + "resolved": "https://registry.npmjs.org/commander/-/commander-4.1.1.tgz", + "integrity": "sha512-NOKm8xhkzAjzFx8B2v5OAHT+u5pRQc2UCa2Vq9jYL/31o2wi9mxBA7LIFs3sV5VSC49z6pEhfbMULvShKj26WA==", + "dev": true + } + } + }, + "html-webpack-plugin": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/html-webpack-plugin/-/html-webpack-plugin-4.3.0.tgz", + "integrity": "sha512-C0fzKN8yQoVLTelcJxZfJCE+aAvQiY2VUf3UuKrR4a9k5UMWYOtpDLsaXwATbcVCnI05hUS7L9ULQHWLZhyi3w==", + "dev": true, + "requires": { + "@types/html-minifier-terser": "^5.0.0", + "@types/tapable": "^1.0.5", + "@types/webpack": "^4.41.8", + "html-minifier-terser": "^5.0.1", + "loader-utils": "^1.2.3", + "lodash": "^4.17.15", + "pretty-error": "^2.1.1", + "tapable": "^1.1.3", + "util.promisify": "1.0.0" + }, + "dependencies": { + "json5": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/json5/-/json5-1.0.1.tgz", + "integrity": "sha512-aKS4WQjPenRxiQsC93MNfjx+nbF4PAdYzmd/1JIj8HYzqfbu86beTuNgXDzPknWk0n0uARlyewZo4s++ES36Ow==", + "dev": true, + "requires": { + "minimist": "^1.2.0" + } + }, + "loader-utils": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/loader-utils/-/loader-utils-1.4.0.tgz", + "integrity": "sha512-qH0WSMBtn/oHuwjy/NucEgbx5dbxxnxup9s4PVXJUDHZBQY+s0NWA9rJf53RBnQZxfch7euUui7hpoAPvALZdA==", + "dev": true, + "requires": { + "big.js": "^5.2.2", + "emojis-list": "^3.0.0", + "json5": "^1.0.1" + } + } + } + }, + "htmlparser2": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/htmlparser2/-/htmlparser2-4.1.0.tgz", + "integrity": "sha512-4zDq1a1zhE4gQso/c5LP1OtrhYTncXNSpvJYtWJBtXAETPlMfi3IFNjGuQbYLuVY4ZR0QMqRVvo4Pdy9KLyP8Q==", + "dev": true, + "requires": { + "domelementtype": "^2.0.1", + "domhandler": "^3.0.0", + "domutils": "^2.0.0", + "entities": "^2.0.0" + } + }, + "http-basic": { + "version": "8.1.3", + "resolved": "https://registry.npmjs.org/http-basic/-/http-basic-8.1.3.tgz", + "integrity": "sha512-/EcDMwJZh3mABI2NhGfHOGOeOZITqfkEO4p/xK+l3NpyncIHUQBoMvCSF/b5GqvKtySC2srL/GGG3+EtlqlmCw==", + "dev": true, + "requires": { + "caseless": "^0.12.0", + "concat-stream": "^1.6.2", + "http-response-object": "^3.0.1", + "parse-cache-control": "^1.0.1" + } + }, + "http-cache-semantics": { + "version": "3.8.1", + "resolved": "https://registry.npmjs.org/http-cache-semantics/-/http-cache-semantics-3.8.1.tgz", + "integrity": "sha512-5ai2iksyV8ZXmnZhHH4rWPoxxistEexSi5936zIQ1bnNTW5VnA85B6P/VpXiRM017IgRvb2kKo1a//y+0wSp3w==", + "dev": true + }, + "http-deceiver": { + "version": "1.2.7", + "resolved": "https://registry.npmjs.org/http-deceiver/-/http-deceiver-1.2.7.tgz", + "integrity": "sha1-+nFolEq5pRnTN8sL7HKE3D5yPYc=", + "dev": true + }, + "http-errors": { + "version": "1.7.3", + "resolved": "https://registry.npmjs.org/http-errors/-/http-errors-1.7.3.tgz", + "integrity": "sha512-ZTTX0MWrsQ2ZAhA1cejAwDLycFsd7I7nVtnkT3Ol0aqodaKW+0CTZDQ1uBv5whptCnc8e8HeRRJxRs0kmm/Qfw==", + "dev": true, + "requires": { + "depd": "~1.1.2", + "inherits": "2.0.4", + "setprototypeof": "1.1.1", + "statuses": ">= 1.5.0 < 2", + "toidentifier": "1.0.0" + }, + "dependencies": { + "statuses": { + "version": "1.5.0", + "resolved": "https://registry.npmjs.org/statuses/-/statuses-1.5.0.tgz", + "integrity": "sha1-Fhx9rBd2Wf2YEfQ3cfqZOBR4Yow=", + "dev": true + } + } + }, + "http-proxy": { + "version": "1.15.2", + "resolved": "https://registry.npmjs.org/http-proxy/-/http-proxy-1.15.2.tgz", + "integrity": "sha1-ZC/cr/5S00SNK9o7AHnpQJBk2jE=", + "dev": true, + "requires": { + "eventemitter3": "1.x.x", + "requires-port": "1.x.x" + } + }, + "http-proxy-agent": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/http-proxy-agent/-/http-proxy-agent-2.1.0.tgz", + "integrity": "sha512-qwHbBLV7WviBl0rQsOzH6o5lwyOIvwp/BdFnvVxXORldu5TmjFfjzBcWUWS5kWAZhmv+JtiDhSuQCp4sBfbIgg==", + "dev": true, + "requires": { + "agent-base": "4", + "debug": "3.1.0" + }, + "dependencies": { + "debug": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/debug/-/debug-3.1.0.tgz", + "integrity": "sha512-OX8XqP7/1a9cqkxYw2yXss15f26NKWBpDXQd0/uK/KPqdQhxbPa994hnzjcE2VqQpDslf55723cKPUOGSmMY3g==", + "dev": true, + "requires": { + "ms": "2.0.0" + } + }, + "ms": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", + "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=", + "dev": true + } + } + }, + "http-proxy-middleware": { + "version": "0.19.1", + "resolved": "https://registry.npmjs.org/http-proxy-middleware/-/http-proxy-middleware-0.19.1.tgz", + "integrity": "sha512-yHYTgWMQO8VvwNS22eLLloAkvungsKdKTLO8AJlftYIKNfJr3GK3zK0ZCfzDDGUBttdGc8xFy1mCitvNKQtC3Q==", + "dev": true, + "requires": { + "http-proxy": "^1.17.0", + "is-glob": "^4.0.0", + "lodash": "^4.17.11", + "micromatch": "^3.1.10" + }, + "dependencies": { + "eventemitter3": { + "version": "4.0.7", + "resolved": "https://registry.npmjs.org/eventemitter3/-/eventemitter3-4.0.7.tgz", + "integrity": "sha512-8guHBZCwKnFhYdHr2ysuRWErTwhoN2X8XELRlrRwpmfeY2jjuUN4taQMsULKUVo1K4DvZl+0pgfyoysHxvmvEw==", + "dev": true + }, + "http-proxy": { + "version": "1.18.1", + "resolved": "https://registry.npmjs.org/http-proxy/-/http-proxy-1.18.1.tgz", + "integrity": "sha512-7mz/721AbnJwIVbnaSv1Cz3Am0ZLT/UBwkC92VlxhXv/k/BBQfM2fXElQNC27BVGr0uwUpplYPQM9LnaBMR5NQ==", + "dev": true, + "requires": { + "eventemitter3": "^4.0.0", + "follow-redirects": "^1.0.0", + "requires-port": "^1.0.0" + } + } + } + }, + "http-response-object": { + "version": "3.0.2", + "resolved": "https://registry.npmjs.org/http-response-object/-/http-response-object-3.0.2.tgz", + "integrity": "sha512-bqX0XTF6fnXSQcEJ2Iuyr75yVakyjIDCqroJQ/aHfSdlM743Cwqoi2nDYMzLGWUcuTWGWy8AAvOKXTfiv6q9RA==", + "dev": true, + "requires": { + "@types/node": "^10.0.3" + }, + "dependencies": { + "@types/node": { + "version": "10.17.46", + "resolved": "https://registry.npmjs.org/@types/node/-/node-10.17.46.tgz", + "integrity": "sha512-Tice8a+sJtlP9C1EUo0DYyjq52T37b3LexVu3p871+kfIBIN+OQ7PKPei1oF3MgF39olEpUfxaLtD+QFc1k69Q==", + "dev": true + } + } + }, + "http-signature": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/http-signature/-/http-signature-1.2.0.tgz", + "integrity": "sha1-muzZJRFHcvPZW2WmCruPfBj7rOE=", + "dev": true, + "requires": { + "assert-plus": "^1.0.0", + "jsprim": "^1.2.2", + "sshpk": "^1.7.0" + } + }, + "https-browserify": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/https-browserify/-/https-browserify-1.0.0.tgz", + "integrity": "sha1-7AbBDgo0wPL68Zn3/X/Hj//QPHM=", + "dev": true + }, + "https-proxy-agent": { + "version": "2.2.4", + "resolved": "https://registry.npmjs.org/https-proxy-agent/-/https-proxy-agent-2.2.4.tgz", + "integrity": "sha512-OmvfoQ53WLjtA9HeYP9RNrWMJzzAz1JGaSFr1nijg0PVR1JaD/xbJq1mdEIIlxGpXp9eSe/O2LgU9DJmTPd0Eg==", + "dev": true, + "requires": { + "agent-base": "^4.3.0", + "debug": "^3.1.0" + }, + "dependencies": { + "debug": { + "version": "3.2.7", + "resolved": "https://registry.npmjs.org/debug/-/debug-3.2.7.tgz", + "integrity": "sha512-CFjzYYAi4ThfiQvizrFQevTTXHtnCqWfe7x1AhgEscTz6ZbLbfoLRLPugTQyBth6f8ZERVUSyWHFD/7Wu4t1XQ==", + "dev": true, + "requires": { + "ms": "^2.1.1" + } + } + } + }, + "human-signals": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/human-signals/-/human-signals-1.1.1.tgz", + "integrity": "sha512-SEQu7vl8KjNL2eoGBLF3+wAjpsNfA9XMlXAYj/3EdaNfAlxKthD1xjEQfGOUhllCGGJVNY34bRr6lPINhNjyZw==", + "dev": true + }, + "humanize-ms": { + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/humanize-ms/-/humanize-ms-1.2.1.tgz", + "integrity": "sha1-xG4xWaKT9riW2ikxbYtv6Lt5u+0=", + "dev": true, + "requires": { + "ms": "^2.0.0" + } + }, + "husky": { + "version": "4.2.5", + "resolved": "https://registry.npmjs.org/husky/-/husky-4.2.5.tgz", + "integrity": "sha512-SYZ95AjKcX7goYVZtVZF2i6XiZcHknw50iXvY7b0MiGoj5RwdgRQNEHdb+gPDPCXKlzwrybjFjkL6FOj8uRhZQ==", + "dev": true, + "requires": { + "chalk": "^4.0.0", + "ci-info": "^2.0.0", + "compare-versions": "^3.6.0", + "cosmiconfig": "^6.0.0", + "find-versions": "^3.2.0", + "opencollective-postinstall": "^2.0.2", + "pkg-dir": "^4.2.0", + "please-upgrade-node": "^3.2.0", + "slash": "^3.0.0", + "which-pm-runs": "^1.0.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.0", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.0.tgz", + "integrity": "sha512-qwx12AxXe2Q5xQ43Ac//I6v5aXTipYrSESdOgzrN+9XjgEpyjpKuvSGaN4qE93f7TQTlerQQ8S+EQ0EyDoVL1A==", + "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 + }, + "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" + } + } + } + }, + "iconv-lite": { + "version": "0.6.2", + "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.6.2.tgz", + "integrity": "sha512-2y91h5OpQlolefMPmUlivelittSWy0rP+oYVpn6A7GwVHNE8AWzoYOBNmlwks3LobaJxgHCYZAnyNo2GgpNRNQ==", + "dev": true, + "requires": { + "safer-buffer": ">= 2.1.2 < 3.0.0" + } + }, + "icss-utils": { + "version": "4.1.1", + "resolved": "https://registry.npmjs.org/icss-utils/-/icss-utils-4.1.1.tgz", + "integrity": "sha512-4aFq7wvWyMHKgxsH8QQtGpvbASCf+eM3wPRLI6R+MgAnTCZ6STYsRvttLvRWK0Nfif5piF394St3HeJDaljGPA==", + "dev": true, + "requires": { + "postcss": "^7.0.14" + } + }, + "ieee754": { + "version": "1.1.13", + "resolved": "https://registry.npmjs.org/ieee754/-/ieee754-1.1.13.tgz", + "integrity": "sha512-4vf7I2LYV/HaWerSo3XmlMkp5eZ83i+/CDluXi/IGTs/O1sejBNhTtnxzmRZfvOUqj7lZjqHkeTvpgSFDlWZTg==", + "dev": true + }, + "iferr": { + "version": "0.1.5", + "resolved": "https://registry.npmjs.org/iferr/-/iferr-0.1.5.tgz", + "integrity": "sha1-xg7taebY/bazEEofy8ocGS3FtQE=", + "dev": true + }, + "ignore": { + "version": "5.1.8", + "resolved": "https://registry.npmjs.org/ignore/-/ignore-5.1.8.tgz", + "integrity": "sha512-BMpfD7PpiETpBl/A6S498BaIJ6Y/ABT93ETbby2fP00v4EbvPBXWEoaR1UBPKs3iR53pJY7EtZk5KACI57i1Uw==", + "dev": true + }, + "ignore-walk": { + "version": "3.0.3", + "resolved": "https://registry.npmjs.org/ignore-walk/-/ignore-walk-3.0.3.tgz", + "integrity": "sha512-m7o6xuOaT1aqheYHKf8W6J5pYH85ZI9w077erOzLje3JsB1gkafkAhHHY19dqjulgIZHFm32Cp5uNZgcQqdJKw==", + "dev": true, + "requires": { + "minimatch": "^3.0.4" + } + }, + "immutable": { + "version": "3.8.2", + "resolved": "https://registry.npmjs.org/immutable/-/immutable-3.8.2.tgz", + "integrity": "sha1-wkOZUUVbs5kT2vKBN28VMOEErfM=", + "dev": true + }, + "import-cwd": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/import-cwd/-/import-cwd-2.1.0.tgz", + "integrity": "sha1-qmzzbnInYShcs3HsZRn1PiQ1sKk=", + "dev": true, + "requires": { + "import-from": "^2.1.0" + } + }, + "import-fresh": { + "version": "3.2.2", + "resolved": "https://registry.npmjs.org/import-fresh/-/import-fresh-3.2.2.tgz", + "integrity": "sha512-cTPNrlvJT6twpYy+YmKUKrTSjWFs3bjYjAhCwm+z4EOCubZxAuO+hHpRN64TqjEaYSHs7tJAE0w1CKMGmsG/lw==", + "dev": true, + "requires": { + "parent-module": "^1.0.0", + "resolve-from": "^4.0.0" + } + }, + "import-from": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/import-from/-/import-from-2.1.0.tgz", + "integrity": "sha1-M1238qev/VOqpHHUuAId7ja387E=", + "dev": true, + "requires": { + "resolve-from": "^3.0.0" + }, + "dependencies": { + "resolve-from": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/resolve-from/-/resolve-from-3.0.0.tgz", + "integrity": "sha1-six699nWiBvItuZTM17rywoYh0g=", + "dev": true + } + } + }, + "import-local": { + "version": "3.0.2", + "resolved": "https://registry.npmjs.org/import-local/-/import-local-3.0.2.tgz", + "integrity": "sha512-vjL3+w0oulAVZ0hBHnxa/Nm5TAurf9YLQJDhqRZyqb+VKGOB6LU8t9H1Nr5CIo16vh9XfJTOoHwU0B71S557gA==", + "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": "sha1-khi5srkoojixPcT7a21XbyMUU+o=", + "dev": true + }, + "indent-string": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/indent-string/-/indent-string-4.0.0.tgz", + "integrity": "sha512-EdDDZu4A2OyIK7Lr/2zG+w5jmbuk1DVBnEwREQvBzspBJkCEbRa8GxU1lghYcaGJCnRWibjDXlq779X1/y5xwg==", + "dev": true + }, + "indexes-of": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/indexes-of/-/indexes-of-1.0.1.tgz", + "integrity": "sha1-8w9xbI4r00bHtn0985FVZqfAVgc=", + "dev": true + }, + "indexof": { + "version": "0.0.1", + "resolved": "https://registry.npmjs.org/indexof/-/indexof-0.0.1.tgz", + "integrity": "sha1-gtwzbSMrkGIXnQWrMpOmYFn9Q10=", + "dev": true + }, + "infer-owner": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/infer-owner/-/infer-owner-1.0.4.tgz", + "integrity": "sha512-IClj+Xz94+d7irH5qRyfJonOdfTzuDaifE6ZPWfx0N0+/ATZCbuTPq2prFl526urkQd90WyUKIh1DfBQ2hMz9A==", + "dev": true + }, + "inflight": { + "version": "1.0.6", + "resolved": "https://registry.npmjs.org/inflight/-/inflight-1.0.6.tgz", + "integrity": "sha1-Sb1jMdfQLQwJvJEKEHW6gWW1bfk=", + "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==" + }, + "ini": { + "version": "1.3.5", + "resolved": "https://registry.npmjs.org/ini/-/ini-1.3.5.tgz", + "integrity": "sha512-RZY5huIKCMRWDUqZlEi72f/lmXKMvuszcMBduliQ3nnWbx9X/ZBQO7DijMEYS9EhHBb2qacRUMtC7svLwe0lcw==", + "dev": true + }, + "inquirer": { + "version": "7.1.0", + "resolved": "https://registry.npmjs.org/inquirer/-/inquirer-7.1.0.tgz", + "integrity": "sha512-5fJMWEmikSYu0nv/flMc475MhGbB7TSPd/2IpFV4I4rMklboCH2rQjYY5kKiYGHqUF9gvaambupcJFFG9dvReg==", + "dev": true, + "requires": { + "ansi-escapes": "^4.2.1", + "chalk": "^3.0.0", + "cli-cursor": "^3.1.0", + "cli-width": "^2.0.0", + "external-editor": "^3.0.3", + "figures": "^3.0.0", + "lodash": "^4.17.15", + "mute-stream": "0.0.8", + "run-async": "^2.4.0", + "rxjs": "^6.5.3", + "string-width": "^4.1.0", + "strip-ansi": "^6.0.0", + "through": "^2.3.6" + }, + "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": "3.0.0", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-3.0.0.tgz", + "integrity": "sha512-4D3B6Wf41KOYRFdszmDqMCGq5VV/uMAB273JILmO+3jAlh8X4qDtdtgCR3fxtbLEMzSx22QdhnDcJvu2u1fVwg==", + "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 + }, + "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" + } + } + } + }, + "insight": { + "version": "0.10.3", + "resolved": "https://registry.npmjs.org/insight/-/insight-0.10.3.tgz", + "integrity": "sha512-YOncxSN6Omh+1Oqxt+OJAvJVMDKw7l6IEG0wT2cTMGxjsTcroOGW4IR926QDzxg/uZHcFZ2cZbckDWdZhc2pZw==", + "dev": true, + "requires": { + "async": "^2.6.2", + "chalk": "^2.4.2", + "conf": "^1.4.0", + "inquirer": "^6.3.1", + "lodash.debounce": "^4.0.8", + "os-name": "^3.1.0", + "request": "^2.88.0", + "tough-cookie": "^3.0.1", + "uuid": "^3.3.2" + }, + "dependencies": { + "ansi-escapes": { + "version": "3.2.0", + "resolved": "https://registry.npmjs.org/ansi-escapes/-/ansi-escapes-3.2.0.tgz", + "integrity": "sha512-cBhpre4ma+U0T1oM5fXg7Dy1Jw7zzwv7lt/GoCpr+hDQJoYnKVPLL4dCvSEFMmQurOQvSrwT7SL/DAlhBI97RQ==", + "dev": true + }, + "ansi-regex": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-3.0.0.tgz", + "integrity": "sha1-7QMXwyIGT3lGbAKWa922Bas32Zg=", + "dev": true + }, + "async": { + "version": "2.6.3", + "resolved": "https://registry.npmjs.org/async/-/async-2.6.3.tgz", + "integrity": "sha512-zflvls11DCy+dQWzTW2dzuilv8Z5X/pjfmZOWba6TNIVDm+2UDaJmXSOXlasHKfNBs8oo3M0aT50fDEWfKZjXg==", + "dev": true, + "requires": { + "lodash": "^4.17.14" + } + }, + "cli-cursor": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/cli-cursor/-/cli-cursor-2.1.0.tgz", + "integrity": "sha1-s12sN2R5+sw+lHR9QdDQ9SOP/LU=", + "dev": true, + "requires": { + "restore-cursor": "^2.0.0" + } + }, + "conf": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/conf/-/conf-1.4.0.tgz", + "integrity": "sha512-bzlVWS2THbMetHqXKB8ypsXN4DQ/1qopGwNJi1eYbpwesJcd86FBjFciCQX/YwAhp9bM7NVnPFqZ5LpV7gP0Dg==", + "dev": true, + "requires": { + "dot-prop": "^4.1.0", + "env-paths": "^1.0.0", + "make-dir": "^1.0.0", + "pkg-up": "^2.0.0", + "write-file-atomic": "^2.3.0" + } + }, + "dot-prop": { + "version": "4.2.1", + "resolved": "https://registry.npmjs.org/dot-prop/-/dot-prop-4.2.1.tgz", + "integrity": "sha512-l0p4+mIuJIua0mhxGoh4a+iNL9bmeK5DvnSVQa6T0OhrVmaEa1XScX5Etc673FePCJOArq/4Pa2cLGODUWTPOQ==", + "dev": true, + "requires": { + "is-obj": "^1.0.0" + } + }, + "env-paths": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/env-paths/-/env-paths-1.0.0.tgz", + "integrity": "sha1-QWgTO0K7BcOKNbGuQ5fIKYqzaeA=", + "dev": true + }, + "figures": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/figures/-/figures-2.0.0.tgz", + "integrity": "sha1-OrGi0qYsi/tDGgyUy3l6L84nyWI=", + "dev": true, + "requires": { + "escape-string-regexp": "^1.0.5" + } + }, + "find-up": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/find-up/-/find-up-2.1.0.tgz", + "integrity": "sha1-RdG35QbHF93UgndaK3eSCjwMV6c=", + "dev": true, + "requires": { + "locate-path": "^2.0.0" + } + }, + "inquirer": { + "version": "6.5.2", + "resolved": "https://registry.npmjs.org/inquirer/-/inquirer-6.5.2.tgz", + "integrity": "sha512-cntlB5ghuB0iuO65Ovoi8ogLHiWGs/5yNrtUcKjFhSSiVeAIVpD7koaSU9RM8mpXw5YDi9RdYXGQMaOURB7ycQ==", + "dev": true, + "requires": { + "ansi-escapes": "^3.2.0", + "chalk": "^2.4.2", + "cli-cursor": "^2.1.0", + "cli-width": "^2.0.0", + "external-editor": "^3.0.3", + "figures": "^2.0.0", + "lodash": "^4.17.12", + "mute-stream": "0.0.7", + "run-async": "^2.2.0", + "rxjs": "^6.4.0", + "string-width": "^2.1.0", + "strip-ansi": "^5.1.0", + "through": "^2.3.6" + } + }, + "is-fullwidth-code-point": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-2.0.0.tgz", + "integrity": "sha1-o7MKXE8ZkYMWeqq5O+764937ZU8=", + "dev": true + }, + "is-obj": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/is-obj/-/is-obj-1.0.1.tgz", + "integrity": "sha1-PkcprB9f3gJc19g6iW2rn09n2w8=", + "dev": true + }, + "locate-path": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-2.0.0.tgz", + "integrity": "sha1-K1aLJl7slExtnA3pw9u7ygNUzY4=", + "dev": true, + "requires": { + "p-locate": "^2.0.0", + "path-exists": "^3.0.0" + } + }, + "make-dir": { + "version": "1.3.0", + "resolved": "https://registry.npmjs.org/make-dir/-/make-dir-1.3.0.tgz", + "integrity": "sha512-2w31R7SJtieJJnQtGc7RVL2StM2vGYVfqUOvUDxH6bC6aJTxPxTF0GnIgCyu7tjockiUWAYQRbxa7vKn34s5sQ==", + "dev": true, + "requires": { + "pify": "^3.0.0" + } + }, + "mimic-fn": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/mimic-fn/-/mimic-fn-1.2.0.tgz", + "integrity": "sha512-jf84uxzwiuiIVKiOLpfYk7N46TSy8ubTonmneY9vrpHNAnp0QBt2BxWV9dO3/j+BoVAb+a5G6YDPW3M5HOdMWQ==", + "dev": true + }, + "mute-stream": { + "version": "0.0.7", + "resolved": "https://registry.npmjs.org/mute-stream/-/mute-stream-0.0.7.tgz", + "integrity": "sha1-MHXOk7whuPq0PhvE2n6BFe0ee6s=", + "dev": true + }, + "onetime": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/onetime/-/onetime-2.0.1.tgz", + "integrity": "sha1-BnQoIw/WdEOyeUsiu6UotoZ5YtQ=", + "dev": true, + "requires": { + "mimic-fn": "^1.0.0" + } + }, + "p-limit": { + "version": "1.3.0", + "resolved": "https://registry.npmjs.org/p-limit/-/p-limit-1.3.0.tgz", + "integrity": "sha512-vvcXsLAJ9Dr5rQOPk7toZQZJApBl2K4J6dANSsEuh6QI41JYcsS/qhTGa9ErIUUgK3WNQoJYvylxvjqmiqEA9Q==", + "dev": true, + "requires": { + "p-try": "^1.0.0" + } + }, + "p-locate": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-2.0.0.tgz", + "integrity": "sha1-IKAQOyIqcMj9OcwuWAaA893l7EM=", + "dev": true, + "requires": { + "p-limit": "^1.1.0" + } + }, + "p-try": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/p-try/-/p-try-1.0.0.tgz", + "integrity": "sha1-y8ec26+P1CKOE/Yh8rGiN8GyB7M=", + "dev": true + }, + "path-exists": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/path-exists/-/path-exists-3.0.0.tgz", + "integrity": "sha1-zg6+ql94yxiSXqfYENe1mwEP1RU=", + "dev": true + }, + "pify": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/pify/-/pify-3.0.0.tgz", + "integrity": "sha1-5aSs0sEB/fPZpNB/DbxNtJ3SgXY=", + "dev": true + }, + "pkg-up": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/pkg-up/-/pkg-up-2.0.0.tgz", + "integrity": "sha1-yBmscoBZpGHKscOImivjxJoATX8=", + "dev": true, + "requires": { + "find-up": "^2.1.0" + } + }, + "restore-cursor": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/restore-cursor/-/restore-cursor-2.0.0.tgz", + "integrity": "sha1-n37ih/gv0ybU/RYpI9YhKe7g368=", + "dev": true, + "requires": { + "onetime": "^2.0.0", + "signal-exit": "^3.0.2" + } + }, + "string-width": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/string-width/-/string-width-2.1.1.tgz", + "integrity": "sha512-nOqH59deCq9SRHlxq1Aw85Jnt4w6KvLKqWVik6oA9ZklXLNIOlqg4F2yrT1MVaTjAqvVwdfeZ7w7aCvJD7ugkw==", + "dev": true, + "requires": { + "is-fullwidth-code-point": "^2.0.0", + "strip-ansi": "^4.0.0" + }, + "dependencies": { + "strip-ansi": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-4.0.0.tgz", + "integrity": "sha1-qEeQIusaw2iocTibY1JixQXuNo8=", + "dev": true, + "requires": { + "ansi-regex": "^3.0.0" + } + } + } + }, + "strip-ansi": { + "version": "5.2.0", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-5.2.0.tgz", + "integrity": "sha512-DuRs1gKbBqsMKIZlrffwlug8MHkcnpjs5VPmL1PAh+mA30U0DTotfDZ0d2UUsXpPmPmMMJ6W773MaA3J+lbiWA==", + "dev": true, + "requires": { + "ansi-regex": "^4.1.0" + }, + "dependencies": { + "ansi-regex": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-4.1.0.tgz", + "integrity": "sha512-1apePfXM1UOSqw0o9IiFAovVz9M5S1Dg+4TrDwfMewQ6p/rmMueb7tWZjQ1rx4Loy1ArBggoqGpfqqdI4rondg==", + "dev": true + } + } + }, + "tough-cookie": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/tough-cookie/-/tough-cookie-3.0.1.tgz", + "integrity": "sha512-yQyJ0u4pZsv9D4clxO69OEjLWYw+jbgspjTue4lTQZLfV0c5l1VmK2y1JK8E9ahdpltPOaAThPcp5nKPUgSnsg==", + "dev": true, + "requires": { + "ip-regex": "^2.1.0", + "psl": "^1.1.28", + "punycode": "^2.1.1" + } + }, + "uuid": { + "version": "3.4.0", + "resolved": "https://registry.npmjs.org/uuid/-/uuid-3.4.0.tgz", + "integrity": "sha512-HjSDRw6gZE5JMggctHBcjVak08+KEVhSIiDzFnT9S9aegmp85S/bReBVTb4QTFaRNptJ9kuYaNhnbNEOkbKb/A==", + "dev": true + }, + "write-file-atomic": { + "version": "2.4.3", + "resolved": "https://registry.npmjs.org/write-file-atomic/-/write-file-atomic-2.4.3.tgz", + "integrity": "sha512-GaETH5wwsX+GcnzhPgKcKjJ6M2Cq3/iZp1WyY/X1CSqrW+jVNM9Y7D8EC2sM4ZG/V8wZlSniJnCKWPmBYAucRQ==", + "dev": true, + "requires": { + "graceful-fs": "^4.1.11", + "imurmurhash": "^0.1.4", + "signal-exit": "^3.0.2" + } + } + } + }, + "internal-ip": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/internal-ip/-/internal-ip-4.3.0.tgz", + "integrity": "sha512-S1zBo1D6zcsyuC6PMmY5+55YMILQ9av8lotMx447Bq6SAgo/sDK6y6uUKmuYhW7eacnIhFfsPmCNYdDzsnnDCg==", + "dev": true, + "requires": { + "default-gateway": "^4.2.0", + "ipaddr.js": "^1.9.0" + } + }, + "interpret": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/interpret/-/interpret-1.4.0.tgz", + "integrity": "sha512-agE4QfB2Lkp9uICn7BAqoscw4SZP9kTE2hxiFI3jBPmXJfdqiahTbUuKGsMoN2GtqL9AxhYioAcVvgsb1HvRbA==", + "dev": true + }, + "invert-kv": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/invert-kv/-/invert-kv-1.0.0.tgz", + "integrity": "sha1-EEqOSqym09jNFXqO+L+rLXo//bY=", + "dev": true + }, + "ip": { + "version": "1.1.5", + "resolved": "https://registry.npmjs.org/ip/-/ip-1.1.5.tgz", + "integrity": "sha1-vd7XARQpCCjAoDnnLvJfWq7ENUo=", + "dev": true + }, + "ip-regex": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/ip-regex/-/ip-regex-2.1.0.tgz", + "integrity": "sha1-+ni/XS5pE8kRzp+BnuUUa7bYROk=", + "dev": true + }, + "ipaddr.js": { + "version": "1.9.1", + "resolved": "https://registry.npmjs.org/ipaddr.js/-/ipaddr.js-1.9.1.tgz", + "integrity": "sha512-0KI/607xoxSToH7GjN1FfSbLoU0+btTicjsQSWQlh/hZykN8KpmMf7uYwPW3R+akZ6R/w18ZlXSHBYXiYUPO3g==", + "dev": true + }, + "is-absolute-url": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/is-absolute-url/-/is-absolute-url-2.1.0.tgz", + "integrity": "sha1-UFMN+4T8yap9vnhS6Do3uTufKqY=", + "dev": true + }, + "is-accessor-descriptor": { + "version": "0.1.6", + "resolved": "https://registry.npmjs.org/is-accessor-descriptor/-/is-accessor-descriptor-0.1.6.tgz", + "integrity": "sha1-qeEss66Nh2cn7u84Q/igiXtcmNY=", + "dev": true, + "requires": { + "kind-of": "^3.0.2" + }, + "dependencies": { + "kind-of": { + "version": "3.2.2", + "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-3.2.2.tgz", + "integrity": "sha1-MeohpzS6ubuw8yRm2JOupR5KPGQ=", + "dev": true, + "requires": { + "is-buffer": "^1.1.5" + } + } + } + }, + "is-arguments": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/is-arguments/-/is-arguments-1.0.4.tgz", + "integrity": "sha512-xPh0Rmt8NE65sNzvyUmWgI1tz3mKq74lGA0mL8LYZcoIzKOzDh6HmrYm3d18k60nHerC8A9Km8kYu87zfSFnLA==", + "dev": true + }, + "is-arrayish": { + "version": "0.2.1", + "resolved": "https://registry.npmjs.org/is-arrayish/-/is-arrayish-0.2.1.tgz", + "integrity": "sha1-d8mYQFJ6qOyxqLppe4BkWnqSap0=", + "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-buffer": { + "version": "1.1.6", + "resolved": "https://registry.npmjs.org/is-buffer/-/is-buffer-1.1.6.tgz", + "integrity": "sha512-NcdALwpXkTm5Zvvbk7owOUSvVvBKDgKP5/ewfXEznmQFfs4ZRmanOeKBTjRVjka3QFoN6XJ+9F3USqfHqTaU5w==", + "dev": true + }, + "is-callable": { + "version": "1.2.2", + "resolved": "https://registry.npmjs.org/is-callable/-/is-callable-1.2.2.tgz", + "integrity": "sha512-dnMqspv5nU3LoewK2N/y7KLtxtakvTuaCsU9FU50/QDmdbHNy/4/JuRtMHqRU22o3q+W89YQndQEeCVwK+3qrA==", + "dev": true + }, + "is-ci": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/is-ci/-/is-ci-2.0.0.tgz", + "integrity": "sha512-YfJT7rkpQB0updsdHLGWrvhBJfcfzNNawYDNIyQXJz0IViGf75O8EBPKSdvw2rF+LGCsX4FZ8tcr3b19LcZq4w==", + "dev": true, + "requires": { + "ci-info": "^2.0.0" + } + }, + "is-color-stop": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/is-color-stop/-/is-color-stop-1.1.0.tgz", + "integrity": "sha1-z/9HGu5N1cnhWFmPvhKWe1za00U=", + "dev": true, + "requires": { + "css-color-names": "^0.0.4", + "hex-color-regex": "^1.1.0", + "hsl-regex": "^1.0.0", + "hsla-regex": "^1.0.0", + "rgb-regex": "^1.0.1", + "rgba-regex": "^1.0.0" + } + }, + "is-core-module": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/is-core-module/-/is-core-module-2.1.0.tgz", + "integrity": "sha512-YcV7BgVMRFRua2FqQzKtTDMz8iCuLEyGKjr70q8Zm1yy2qKcurbFEd79PAdHV77oL3NrAaOVQIbMmiHQCHB7ZA==", + "requires": { + "has": "^1.0.3" + } + }, + "is-data-descriptor": { + "version": "0.1.4", + "resolved": "https://registry.npmjs.org/is-data-descriptor/-/is-data-descriptor-0.1.4.tgz", + "integrity": "sha1-C17mSDiOLIYCgueT8YVv7D8wG1Y=", + "dev": true, + "requires": { + "kind-of": "^3.0.2" + }, + "dependencies": { + "kind-of": { + "version": "3.2.2", + "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-3.2.2.tgz", + "integrity": "sha1-MeohpzS6ubuw8yRm2JOupR5KPGQ=", + "dev": true, + "requires": { + "is-buffer": "^1.1.5" + } + } + } + }, + "is-date-object": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/is-date-object/-/is-date-object-1.0.2.tgz", + "integrity": "sha512-USlDT524woQ08aoZFzh3/Z6ch9Y/EWXEHQ/AaRN0SkKq4t2Jw2R2339tSXmwuVoY7LLlBCbOIlx2myP/L5zk0g==", + "dev": true + }, + "is-descriptor": { + "version": "0.1.6", + "resolved": "https://registry.npmjs.org/is-descriptor/-/is-descriptor-0.1.6.tgz", + "integrity": "sha512-avDYr0SB3DwO9zsMov0gKCESFYqCnE4hq/4z3TdUlukEy5t9C0YRq7HLrsN52NAcqXKaepeCD0n+B0arnVG3Hg==", + "dev": true, + "requires": { + "is-accessor-descriptor": "^0.1.6", + "is-data-descriptor": "^0.1.4", + "kind-of": "^5.0.0" + }, + "dependencies": { + "kind-of": { + "version": "5.1.0", + "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-5.1.0.tgz", + "integrity": "sha512-NGEErnH6F2vUuXDh+OlbcKW7/wOcfdRHaZ7VWtqCztfHri/++YKmP51OdWeGPuqCOba6kk2OTe5d02VmTB80Pw==", + "dev": true + } + } + }, + "is-directory": { + "version": "0.3.1", + "resolved": "https://registry.npmjs.org/is-directory/-/is-directory-0.3.1.tgz", + "integrity": "sha1-YTObbyR1/Hcv2cnYP1yFddwVSuE=", + "dev": true + }, + "is-docker": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/is-docker/-/is-docker-2.1.1.tgz", + "integrity": "sha512-ZOoqiXfEwtGknTiuDEy8pN2CfE3TxMHprvNer1mXiqwkOT77Rw3YVrUQ52EqAOU3QAWDQ+bQdx7HJzrv7LS2Hw==", + "dev": true + }, + "is-extendable": { + "version": "0.1.1", + "resolved": "https://registry.npmjs.org/is-extendable/-/is-extendable-0.1.1.tgz", + "integrity": "sha1-YrEQ4omkcUGOPsNqYX1HLjAd/Ik=", + "dev": true + }, + "is-extglob": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/is-extglob/-/is-extglob-2.1.1.tgz", + "integrity": "sha1-qIwCU1eR8C7TfHahueqXc8gz+MI=", + "dev": true + }, + "is-fullwidth-code-point": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-3.0.0.tgz", + "integrity": "sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg==" + }, + "is-generator-fn": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/is-generator-fn/-/is-generator-fn-2.1.0.tgz", + "integrity": "sha512-cTIB4yPYL/Grw0EaSzASzg6bBy9gqCofvWN8okThAYIxKJZC+udlRAmGbM0XLeniEJSs8uEgHPGuHSe1XsOLSQ==", + "dev": true + }, + "is-glob": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-4.0.1.tgz", + "integrity": "sha512-5G0tKtBTFImOqDnLB2hG6Bp2qcKEFduo4tZu9MT/H6NQv/ghhy30o55ufafxJ/LdH79LLs2Kfrn85TLKyA7BUg==", + "dev": true, + "requires": { + "is-extglob": "^2.1.1" + } + }, + "is-interactive": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/is-interactive/-/is-interactive-1.0.0.tgz", + "integrity": "sha512-2HvIEKRoqS62guEC+qBjpvRubdX910WCMuJTZ+I9yvqKU2/12eSL549HMwtabb4oupdj2sMP50k+XJfB/8JE6w==", + "dev": true + }, + "is-module": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/is-module/-/is-module-1.0.0.tgz", + "integrity": "sha1-Mlj7afeMFNW4FdZkM2tM/7ZEFZE=", + "dev": true + }, + "is-negative-zero": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/is-negative-zero/-/is-negative-zero-2.0.0.tgz", + "integrity": "sha1-lVOxIbD6wohp2p7UWeIMdUN4hGE=", + "dev": true + }, + "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-number-like": { + "version": "1.0.8", + "resolved": "https://registry.npmjs.org/is-number-like/-/is-number-like-1.0.8.tgz", + "integrity": "sha512-6rZi3ezCyFcn5L71ywzz2bS5b2Igl1En3eTlZlvKjpz1n3IZLAYMbKYAIQgFmEu0GENg92ziU/faEOA/aixjbA==", + "dev": true, + "requires": { + "lodash.isfinite": "^3.3.2" + } + }, + "is-obj": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/is-obj/-/is-obj-2.0.0.tgz", + "integrity": "sha512-drqDG3cbczxxEJRoOXcOjtdp1J/lyp1mNn0xaznRs8+muBhgQcrnbspox5X5fOw0HnMnbfDzvnEMEtqDEJEo8w==", + "dev": true + }, + "is-observable": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/is-observable/-/is-observable-1.1.0.tgz", + "integrity": "sha512-NqCa4Sa2d+u7BWc6CukaObG3Fh+CU9bvixbpcXYhy2VvYS7vVGIdAgnIS5Ks3A/cqk4rebLJ9s8zBstT2aKnIA==", + "dev": true, + "requires": { + "symbol-observable": "^1.1.0" + } + }, + "is-path-cwd": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/is-path-cwd/-/is-path-cwd-1.0.0.tgz", + "integrity": "sha1-0iXsIxMuie3Tj9p2dHLmLmXxEG0=", + "dev": true + }, + "is-path-in-cwd": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/is-path-in-cwd/-/is-path-in-cwd-1.0.1.tgz", + "integrity": "sha512-FjV1RTW48E7CWM7eE/J2NJvAEEVektecDBVBE5Hh3nM1Jd0kvhHtX68Pr3xsDf857xt3Y4AkwVULK1Vku62aaQ==", + "dev": true, + "requires": { + "is-path-inside": "^1.0.0" + } + }, + "is-path-inside": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/is-path-inside/-/is-path-inside-1.0.1.tgz", + "integrity": "sha1-jvW33lBDej/cprToZe96pVy0gDY=", + "dev": true, + "requires": { + "path-is-inside": "^1.0.1" + } + }, + "is-plain-obj": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/is-plain-obj/-/is-plain-obj-1.1.0.tgz", + "integrity": "sha1-caUMhCnfync8kqOQpKA7OfzVHT4=", + "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" + } + }, + "is-potential-custom-element-name": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/is-potential-custom-element-name/-/is-potential-custom-element-name-1.0.0.tgz", + "integrity": "sha1-DFLlS8yjkbssSUsh6GJtczbG45c=", + "dev": true + }, + "is-promise": { + "version": "2.2.2", + "resolved": "https://registry.npmjs.org/is-promise/-/is-promise-2.2.2.tgz", + "integrity": "sha512-+lP4/6lKUBfQjZ2pdxThZvLUAafmZb8OAxFb8XXtiQmS35INgr85hdOGoEs124ez1FCnZJt6jau/T+alh58QFQ==", + "dev": true + }, + "is-redirect": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/is-redirect/-/is-redirect-1.0.0.tgz", + "integrity": "sha1-HQPd7VO9jbDzDCbk+V02/HyH3CQ=", + "dev": true + }, + "is-regex": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/is-regex/-/is-regex-1.1.1.tgz", + "integrity": "sha512-1+QkEcxiLlB7VEyFtyBg94e08OAsvq7FUBgApTq/w2ymCLyKJgDPsybBENVtA7XCQEgEXxKPonG+mvYRxh/LIg==", + "dev": true, + "requires": { + "has-symbols": "^1.0.1" + } + }, + "is-regexp": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/is-regexp/-/is-regexp-1.0.0.tgz", + "integrity": "sha1-/S2INUXEa6xaYz57mgnof6LLUGk=", + "dev": true + }, + "is-resolvable": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/is-resolvable/-/is-resolvable-1.1.0.tgz", + "integrity": "sha512-qgDYXFSR5WvEfuS5dMj6oTMEbrrSaM0CrFk2Yiq/gXnBvD9pMa2jGXxyhGLfvhZpuMZe18CJpFxAt3CRs42NMg==", + "dev": true + }, + "is-retry-allowed": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/is-retry-allowed/-/is-retry-allowed-1.2.0.tgz", + "integrity": "sha512-RUbUeKwvm3XG2VYamhJL1xFktgjvPzL0Hq8C+6yrWIswDy3BIXGqCxhxkc30N9jqK311gVU137K8Ei55/zVJRg==", + "dev": true + }, + "is-scoped": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/is-scoped/-/is-scoped-1.0.0.tgz", + "integrity": "sha1-RJypgpnnEwOCViieyytUDcQ3yzA=", + "dev": true, + "requires": { + "scoped-regex": "^1.0.0" + } + }, + "is-stream": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/is-stream/-/is-stream-1.1.0.tgz", + "integrity": "sha1-EtSj3U5o4Lec6428hBc66A2RykQ=", + "dev": true + }, + "is-svg": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/is-svg/-/is-svg-3.0.0.tgz", + "integrity": "sha512-gi4iHK53LR2ujhLVVj+37Ykh9GLqYHX6JOVXbLAucaG/Cqw9xwdFOjDM2qeifLs1sF1npXXFvDu0r5HNgCMrzQ==", + "dev": true, + "requires": { + "html-comment-regex": "^1.1.0" + } + }, + "is-symbol": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/is-symbol/-/is-symbol-1.0.3.tgz", + "integrity": "sha512-OwijhaRSgqvhm/0ZdAcXNZt9lYdKFpcRDT5ULUuYXPoT794UNOdU+gpT6Rzo7b4V2HUl/op6GqY894AZwv9faQ==", + "dev": true, + "requires": { + "has-symbols": "^1.0.1" + } + }, + "is-typedarray": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/is-typedarray/-/is-typedarray-1.0.0.tgz", + "integrity": "sha1-5HnICFjfDBsR3dppQPlgEfzaSpo=", + "dev": true + }, + "is-utf8": { + "version": "0.2.1", + "resolved": "https://registry.npmjs.org/is-utf8/-/is-utf8-0.2.1.tgz", + "integrity": "sha1-Sw2hRCEE0bM2NA6AeX6GXPOffXI=", + "dev": true + }, + "is-windows": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/is-windows/-/is-windows-1.0.2.tgz", + "integrity": "sha512-eXK1UInq2bPmjyX6e3VHIzMLobc4J94i4AWn+Hpq3OU5KkrRC96OAcR3PRJ/pGu6m8TRnBHP9dkXQVsT/COVIA==", + "dev": true + }, + "is-wsl": { + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/is-wsl/-/is-wsl-2.2.0.tgz", + "integrity": "sha512-fKzAra0rGJUUBwGBgNkHZuToZcn+TtXHpeCgmkMJMMYx1sQDYaCSyjJBSCa2nH1DGm7s3n1oBnohoVTBaN7Lww==", + "dev": true, + "requires": { + "is-docker": "^2.0.0" + } + }, + "isarray": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/isarray/-/isarray-1.0.0.tgz", + "integrity": "sha1-u5NdSFgsuhaMBoNJV6VKPgcSTxE=", + "dev": true + }, + "isbinaryfile": { + "version": "4.0.6", + "resolved": "https://registry.npmjs.org/isbinaryfile/-/isbinaryfile-4.0.6.tgz", + "integrity": "sha512-ORrEy+SNVqUhrCaal4hA4fBzhggQQ+BaLntyPOdoEiwlKZW9BZiJXjg3RMiruE4tPEI3pyVPpySHQF/dKWperg==", + "dev": true + }, + "isexe": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/isexe/-/isexe-2.0.0.tgz", + "integrity": "sha1-6PvzdNxVb/iUehDcsFctYz8s+hA=", + "dev": true + }, + "isobject": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/isobject/-/isobject-3.0.1.tgz", + "integrity": "sha1-TkMekrEalzFjaqH5yNHMvP2reN8=", + "dev": true + }, + "isstream": { + "version": "0.1.2", + "resolved": "https://registry.npmjs.org/isstream/-/isstream-0.1.2.tgz", + "integrity": "sha1-R+Y/evVa+m+S4VAOaQ64uFKcCZo=", + "dev": true + }, + "istanbul-lib-coverage": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/istanbul-lib-coverage/-/istanbul-lib-coverage-3.0.0.tgz", + "integrity": "sha512-UiUIqxMgRDET6eR+o5HbfRYP1l0hqkWOs7vNxC/mggutCMUIhWMm8gAHb8tHlyfD3/l6rlgNA5cKdDzEAf6hEg==", + "dev": true + }, + "istanbul-lib-instrument": { + "version": "4.0.3", + "resolved": "https://registry.npmjs.org/istanbul-lib-instrument/-/istanbul-lib-instrument-4.0.3.tgz", + "integrity": "sha512-BXgQl9kf4WTCPCCpmFGoJkz/+uhvm7h7PFKUYxh7qarQd3ER33vHG//qaE8eN25l07YqZPpHXU9I09l/RD5aGQ==", + "dev": true, + "requires": { + "@babel/core": "^7.7.5", + "@istanbuljs/schema": "^0.1.2", + "istanbul-lib-coverage": "^3.0.0", + "semver": "^6.3.0" + }, + "dependencies": { + "semver": { + "version": "6.3.0", + "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.0.tgz", + "integrity": "sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw==", + "dev": true + } + } + }, + "istanbul-lib-report": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/istanbul-lib-report/-/istanbul-lib-report-3.0.0.tgz", + "integrity": "sha512-wcdi+uAKzfiGT2abPpKZ0hSU1rGQjUQnLvtY5MpQ7QCTahD3VODhcu4wcfY1YtkGaDD5yuydOLINXsfbus9ROw==", + "dev": true, + "requires": { + "istanbul-lib-coverage": "^3.0.0", + "make-dir": "^3.0.0", + "supports-color": "^7.1.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": "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" + } + } + } + }, + "istanbul-lib-source-maps": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/istanbul-lib-source-maps/-/istanbul-lib-source-maps-4.0.0.tgz", + "integrity": "sha512-c16LpFRkR8vQXyHZ5nLpY35JZtzj1PQY1iZmesUbf1FZHbIupcWfjgOXBY9YHkLEQ6puz1u4Dgj6qmU/DisrZg==", + "dev": true, + "requires": { + "debug": "^4.1.1", + "istanbul-lib-coverage": "^3.0.0", + "source-map": "^0.6.1" + }, + "dependencies": { + "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 + } + } + }, + "istanbul-reports": { + "version": "3.0.2", + "resolved": "https://registry.npmjs.org/istanbul-reports/-/istanbul-reports-3.0.2.tgz", + "integrity": "sha512-9tZvz7AiR3PEDNGiV9vIouQ/EAcqMXFmkcA1CDFTwOB98OZVDL0PH9glHotf5Ugp6GCOTypfzGWI/OqjWNCRUw==", + "dev": true, + "requires": { + "html-escaper": "^2.0.0", + "istanbul-lib-report": "^3.0.0" + } + }, + "istextorbinary": { + "version": "2.6.0", + "resolved": "https://registry.npmjs.org/istextorbinary/-/istextorbinary-2.6.0.tgz", + "integrity": "sha512-+XRlFseT8B3L9KyjxxLjfXSLMuErKDsd8DBNrsaxoViABMEZlOSCstwmw0qpoFX3+U6yWU1yhLudAe6/lETGGA==", + "dev": true, + "requires": { + "binaryextensions": "^2.1.2", + "editions": "^2.2.0", + "textextensions": "^2.5.0" + } + }, + "jake": { + "version": "10.8.2", + "resolved": "https://registry.npmjs.org/jake/-/jake-10.8.2.tgz", + "integrity": "sha512-eLpKyrfG3mzvGE2Du8VoPbeSkRry093+tyNjdYaBbJS9v17knImYGNXQCUV0gLxQtF82m3E8iRb/wdSQZLoq7A==", + "dev": true, + "requires": { + "async": "0.9.x", + "chalk": "^2.4.2", + "filelist": "^1.0.1", + "minimatch": "^3.0.4" + }, + "dependencies": { + "async": { + "version": "0.9.2", + "resolved": "https://registry.npmjs.org/async/-/async-0.9.2.tgz", + "integrity": "sha1-rqdNXmHB+JlhO/ZL2mbUx48v0X0=", + "dev": true + } + } + }, + "java-parser": { + "version": "0.8.2", + "resolved": "https://registry.npmjs.org/java-parser/-/java-parser-0.8.2.tgz", + "integrity": "sha512-YphTEk4zIpWAHqBriAdjlNnTEcEYQ2xBA8KOH5V7QxggNbxjkeEcKJjNYnQvqEue8+O1TLj7vfL0NVG6x5LGMw==", + "dev": true, + "requires": { + "chevrotain": "6.5.0", + "lodash": "4.17.20" + }, + "dependencies": { + "chevrotain": { + "version": "6.5.0", + "resolved": "https://registry.npmjs.org/chevrotain/-/chevrotain-6.5.0.tgz", + "integrity": "sha512-BwqQ/AgmKJ8jcMEjaSnfMybnKMgGTrtDKowfTP3pX4jwVy0kNjRsT/AP6h+wC3+3NC+X8X15VWBnTCQlX+wQFg==", + "dev": true, + "requires": { + "regexp-to-ast": "0.4.0" + } + }, + "regexp-to-ast": { + "version": "0.4.0", + "resolved": "https://registry.npmjs.org/regexp-to-ast/-/regexp-to-ast-0.4.0.tgz", + "integrity": "sha512-4qf/7IsIKfSNHQXSwial1IFmfM1Cc/whNBQqRwe0V2stPe7KmN1U0tWQiIx6JiirgSrisjE0eECdNf7Tav1Ntw==", + "dev": true + } + } + }, + "jest": { + "version": "26.1.0", + "resolved": "https://registry.npmjs.org/jest/-/jest-26.1.0.tgz", + "integrity": "sha512-LIti8jppw5BcQvmNJe4w2g1N/3V68HUfAv9zDVm7v+VAtQulGhH0LnmmiVkbNE4M4I43Bj2fXPiBGKt26k9tHw==", + "dev": true, + "requires": { + "@jest/core": "^26.1.0", + "import-local": "^3.0.2", + "jest-cli": "^26.1.0" + }, + "dependencies": { + "@jest/types": { + "version": "26.6.2", + "resolved": "https://registry.npmjs.org/@jest/types/-/types-26.6.2.tgz", + "integrity": "sha512-fC6QCp7Sc5sX6g8Tvbmj4XUTbyrik0akgRy03yjXbQaBWWNWGE7SGtJk98m0N8nzegD/7SggrUlivxo5ax4KWQ==", + "dev": true, + "requires": { + "@types/istanbul-lib-coverage": "^2.0.0", + "@types/istanbul-reports": "^3.0.0", + "@types/node": "*", + "@types/yargs": "^15.0.0", + "chalk": "^4.0.0" + } + }, + "@types/istanbul-reports": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/@types/istanbul-reports/-/istanbul-reports-3.0.0.tgz", + "integrity": "sha512-nwKNbvnwJ2/mndE9ItP/zc2TCzw6uuodnF4EHYWD+gCQDVBuRQL5UzbZD0/ezy1iKsFU2ZQiDqg4M9dN4+wZgA==", + "dev": true, + "requires": { + "@types/istanbul-lib-report": "*" + } + }, + "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.0", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.0.tgz", + "integrity": "sha512-qwx12AxXe2Q5xQ43Ac//I6v5aXTipYrSESdOgzrN+9XjgEpyjpKuvSGaN4qE93f7TQTlerQQ8S+EQ0EyDoVL1A==", + "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 + }, + "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-cli": { + "version": "26.6.3", + "resolved": "https://registry.npmjs.org/jest-cli/-/jest-cli-26.6.3.tgz", + "integrity": "sha512-GF9noBSa9t08pSyl3CY4frMrqp+aQXFGFkf5hEPbh/pIUFYWMK6ZLTfbmadxJVcJrdRoChlWQsA2VkJcDFK8hg==", + "dev": true, + "requires": { + "@jest/core": "^26.6.3", + "@jest/test-result": "^26.6.2", + "@jest/types": "^26.6.2", + "chalk": "^4.0.0", + "exit": "^0.1.2", + "graceful-fs": "^4.2.4", + "import-local": "^3.0.2", + "is-ci": "^2.0.0", + "jest-config": "^26.6.3", + "jest-util": "^26.6.2", + "jest-validate": "^26.6.2", + "prompts": "^2.0.1", + "yargs": "^15.4.1" + } + }, + "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" + } + }, + "yargs": { + "version": "15.4.1", + "resolved": "https://registry.npmjs.org/yargs/-/yargs-15.4.1.tgz", + "integrity": "sha512-aePbxDmcYW++PaqBsJ+HYUFwCdv4LVvdnhBy78E57PIor8/OVvhMrADFFEDh8DHDFRv/O9i3lPhsENjO7QX0+A==", + "dev": true, + "requires": { + "cliui": "^6.0.0", + "decamelize": "^1.2.0", + "find-up": "^4.1.0", + "get-caller-file": "^2.0.1", + "require-directory": "^2.1.1", + "require-main-filename": "^2.0.0", + "set-blocking": "^2.0.0", + "string-width": "^4.2.0", + "which-module": "^2.0.0", + "y18n": "^4.0.0", + "yargs-parser": "^18.1.2" + } + } + } + }, + "jest-changed-files": { + "version": "26.6.2", + "resolved": "https://registry.npmjs.org/jest-changed-files/-/jest-changed-files-26.6.2.tgz", + "integrity": "sha512-fDS7szLcY9sCtIip8Fjry9oGf3I2ht/QT21bAHm5Dmf0mD4X3ReNUf17y+bO6fR8WgbIZTlbyG1ak/53cbRzKQ==", + "dev": true, + "requires": { + "@jest/types": "^26.6.2", + "execa": "^4.0.0", + "throat": "^5.0.0" + }, + "dependencies": { + "@jest/types": { + "version": "26.6.2", + "resolved": "https://registry.npmjs.org/@jest/types/-/types-26.6.2.tgz", + "integrity": "sha512-fC6QCp7Sc5sX6g8Tvbmj4XUTbyrik0akgRy03yjXbQaBWWNWGE7SGtJk98m0N8nzegD/7SggrUlivxo5ax4KWQ==", + "dev": true, + "requires": { + "@types/istanbul-lib-coverage": "^2.0.0", + "@types/istanbul-reports": "^3.0.0", + "@types/node": "*", + "@types/yargs": "^15.0.0", + "chalk": "^4.0.0" + } + }, + "@types/istanbul-reports": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/@types/istanbul-reports/-/istanbul-reports-3.0.0.tgz", + "integrity": "sha512-nwKNbvnwJ2/mndE9ItP/zc2TCzw6uuodnF4EHYWD+gCQDVBuRQL5UzbZD0/ezy1iKsFU2ZQiDqg4M9dN4+wZgA==", + "dev": true, + "requires": { + "@types/istanbul-lib-report": "*" + } + }, + "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.0", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.0.tgz", + "integrity": "sha512-qwx12AxXe2Q5xQ43Ac//I6v5aXTipYrSESdOgzrN+9XjgEpyjpKuvSGaN4qE93f7TQTlerQQ8S+EQ0EyDoVL1A==", + "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 + }, + "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" + } + }, + "execa": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/execa/-/execa-4.1.0.tgz", + "integrity": "sha512-j5W0//W7f8UxAn8hXVnwG8tLwdiUy4FJLcSupCg6maBYZDpyBvTApK7KyuI4bKj8KOh1r2YH+6ucuYtJv1bTZA==", + "dev": true, + "requires": { + "cross-spawn": "^7.0.0", + "get-stream": "^5.0.0", + "human-signals": "^1.1.1", + "is-stream": "^2.0.0", + "merge-stream": "^2.0.0", + "npm-run-path": "^4.0.0", + "onetime": "^5.1.0", + "signal-exit": "^3.0.2", + "strip-final-newline": "^2.0.0" + } + }, + "get-stream": { + "version": "5.2.0", + "resolved": "https://registry.npmjs.org/get-stream/-/get-stream-5.2.0.tgz", + "integrity": "sha512-nBF+F1rAZVCu/p7rjzgA+Yb4lfYXrpl7a6VmJrU8wF9I1CKvP/QwPNZHnOlwbTkY6dvtFIzFMSyQXbLoTQPRpA==", + "dev": true, + "requires": { + "pump": "^3.0.0" + } + }, + "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 + }, + "is-stream": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/is-stream/-/is-stream-2.0.0.tgz", + "integrity": "sha512-XCoy+WlUr7d1+Z8GgSuXmpuUFC9fOhRXglJMx+dwLKTkL44Cjd4W1Z5P+BQZpr+cR93aGP4S/s7Ftw6Nd/kiEw==", + "dev": true + }, + "npm-run-path": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/npm-run-path/-/npm-run-path-4.0.1.tgz", + "integrity": "sha512-S48WzZW777zhNIrn7gxOlISNAqi9ZC/uQFnRdbeIHhZhCA6UqpkOT8T1G7BvfdgP4Er8gF4sUbaS0i7QvIfCWw==", + "dev": true, + "requires": { + "path-key": "^3.0.0" + } + }, + "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 + }, + "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 + }, + "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" + } + }, + "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" + } + } + } + }, + "jest-config": { + "version": "26.6.3", + "resolved": "https://registry.npmjs.org/jest-config/-/jest-config-26.6.3.tgz", + "integrity": "sha512-t5qdIj/bCj2j7NFVHb2nFB4aUdfucDn3JRKgrZnplb8nieAirAzRSHP8uDEd+qV6ygzg9Pz4YG7UTJf94LPSyg==", + "dev": true, + "requires": { + "@babel/core": "^7.1.0", + "@jest/test-sequencer": "^26.6.3", + "@jest/types": "^26.6.2", + "babel-jest": "^26.6.3", + "chalk": "^4.0.0", + "deepmerge": "^4.2.2", + "glob": "^7.1.1", + "graceful-fs": "^4.2.4", + "jest-environment-jsdom": "^26.6.2", + "jest-environment-node": "^26.6.2", + "jest-get-type": "^26.3.0", + "jest-jasmine2": "^26.6.3", + "jest-regex-util": "^26.0.0", + "jest-resolve": "^26.6.2", + "jest-util": "^26.6.2", + "jest-validate": "^26.6.2", + "micromatch": "^4.0.2", + "pretty-format": "^26.6.2" + }, + "dependencies": { + "@jest/types": { + "version": "26.6.2", + "resolved": "https://registry.npmjs.org/@jest/types/-/types-26.6.2.tgz", + "integrity": "sha512-fC6QCp7Sc5sX6g8Tvbmj4XUTbyrik0akgRy03yjXbQaBWWNWGE7SGtJk98m0N8nzegD/7SggrUlivxo5ax4KWQ==", + "dev": true, + "requires": { + "@types/istanbul-lib-coverage": "^2.0.0", + "@types/istanbul-reports": "^3.0.0", + "@types/node": "*", + "@types/yargs": "^15.0.0", + "chalk": "^4.0.0" + } + }, + "@types/istanbul-reports": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/@types/istanbul-reports/-/istanbul-reports-3.0.0.tgz", + "integrity": "sha512-nwKNbvnwJ2/mndE9ItP/zc2TCzw6uuodnF4EHYWD+gCQDVBuRQL5UzbZD0/ezy1iKsFU2ZQiDqg4M9dN4+wZgA==", + "dev": true, + "requires": { + "@types/istanbul-lib-report": "*" + } + }, + "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.0", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.0.tgz", + "integrity": "sha512-qwx12AxXe2Q5xQ43Ac//I6v5aXTipYrSESdOgzrN+9XjgEpyjpKuvSGaN4qE93f7TQTlerQQ8S+EQ0EyDoVL1A==", + "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 + }, + "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-get-type": { + "version": "26.3.0", + "resolved": "https://registry.npmjs.org/jest-get-type/-/jest-get-type-26.3.0.tgz", + "integrity": "sha512-TpfaviN1R2pQWkIihlfEanwOXK0zcxrKEE4MlU6Tn7keoXdN6/3gK/xl0yEh8DOunn5pOVGKf8hB4R9gVh04ig==", + "dev": true + }, + "micromatch": { + "version": "4.0.2", + "resolved": "https://registry.npmjs.org/micromatch/-/micromatch-4.0.2.tgz", + "integrity": "sha512-y7FpHSbMUMoyPbYUSzO6PaZ6FyRnQOpHuKwbo1G+Knck95XVU4QAiKdGEnj5wwoS7PlOgthX/09u5iFJ+aYf5Q==", + "dev": true, + "requires": { + "braces": "^3.0.1", + "picomatch": "^2.0.5" + } + }, + "pretty-format": { + "version": "26.6.2", + "resolved": "https://registry.npmjs.org/pretty-format/-/pretty-format-26.6.2.tgz", + "integrity": "sha512-7AeGuCYNGmycyQbCqd/3PWH4eOoX/OiCa0uphp57NVTeAGdJGaAliecxwBDHYQCIvrW7aDBZCYeNTP/WX69mkg==", + "dev": true, + "requires": { + "@jest/types": "^26.6.2", + "ansi-regex": "^5.0.0", + "ansi-styles": "^4.0.0", + "react-is": "^17.0.1" + } + }, + "react-is": { + "version": "17.0.1", + "resolved": "https://registry.npmjs.org/react-is/-/react-is-17.0.1.tgz", + "integrity": "sha512-NAnt2iGDXohE5LI7uBnLnqvLQMtzhkiAOLXTmv+qnF9Ky7xAPcX8Up/xWIhxvLVGJvuLiNc4xQLtuqDRzb4fSA==", + "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" + } + } + } + }, + "jest-date-mock": { + "version": "1.0.8", + "resolved": "https://registry.npmjs.org/jest-date-mock/-/jest-date-mock-1.0.8.tgz", + "integrity": "sha512-0Lyp+z9xvuNmLbK+5N6FOhSiBeux05Lp5bbveFBmYo40Aggl2wwxFoIrZ+rOWC8nDNcLeBoDd2miQdEDSf3iQw==", + "dev": true + }, + "jest-diff": { + "version": "25.5.0", + "resolved": "https://registry.npmjs.org/jest-diff/-/jest-diff-25.5.0.tgz", + "integrity": "sha512-z1kygetuPiREYdNIumRpAHY6RXiGmp70YHptjdaxTWGmA085W3iCnXNx0DhflK3vwrKmrRWyY1wUpkPMVxMK7A==", + "dev": true, + "requires": { + "chalk": "^3.0.0", + "diff-sequences": "^25.2.6", + "jest-get-type": "^25.2.6", + "pretty-format": "^25.5.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": "3.0.0", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-3.0.0.tgz", + "integrity": "sha512-4D3B6Wf41KOYRFdszmDqMCGq5VV/uMAB273JILmO+3jAlh8X4qDtdtgCR3fxtbLEMzSx22QdhnDcJvu2u1fVwg==", + "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 + }, + "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" + } + } + } + }, + "jest-docblock": { + "version": "26.0.0", + "resolved": "https://registry.npmjs.org/jest-docblock/-/jest-docblock-26.0.0.tgz", + "integrity": "sha512-RDZ4Iz3QbtRWycd8bUEPxQsTlYazfYn/h5R65Fc6gOfwozFhoImx+affzky/FFBuqISPTqjXomoIGJVKBWoo0w==", + "dev": true, + "requires": { + "detect-newline": "^3.0.0" + } + }, + "jest-each": { + "version": "26.6.2", + "resolved": "https://registry.npmjs.org/jest-each/-/jest-each-26.6.2.tgz", + "integrity": "sha512-Mer/f0KaATbjl8MCJ+0GEpNdqmnVmDYqCTJYTvoo7rqmRiDllmp2AYN+06F93nXcY3ur9ShIjS+CO/uD+BbH4A==", + "dev": true, + "requires": { + "@jest/types": "^26.6.2", + "chalk": "^4.0.0", + "jest-get-type": "^26.3.0", + "jest-util": "^26.6.2", + "pretty-format": "^26.6.2" + }, + "dependencies": { + "@jest/types": { + "version": "26.6.2", + "resolved": "https://registry.npmjs.org/@jest/types/-/types-26.6.2.tgz", + "integrity": "sha512-fC6QCp7Sc5sX6g8Tvbmj4XUTbyrik0akgRy03yjXbQaBWWNWGE7SGtJk98m0N8nzegD/7SggrUlivxo5ax4KWQ==", + "dev": true, + "requires": { + "@types/istanbul-lib-coverage": "^2.0.0", + "@types/istanbul-reports": "^3.0.0", + "@types/node": "*", + "@types/yargs": "^15.0.0", + "chalk": "^4.0.0" + } + }, + "@types/istanbul-reports": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/@types/istanbul-reports/-/istanbul-reports-3.0.0.tgz", + "integrity": "sha512-nwKNbvnwJ2/mndE9ItP/zc2TCzw6uuodnF4EHYWD+gCQDVBuRQL5UzbZD0/ezy1iKsFU2ZQiDqg4M9dN4+wZgA==", + "dev": true, + "requires": { + "@types/istanbul-lib-report": "*" + } + }, + "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.0", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.0.tgz", + "integrity": "sha512-qwx12AxXe2Q5xQ43Ac//I6v5aXTipYrSESdOgzrN+9XjgEpyjpKuvSGaN4qE93f7TQTlerQQ8S+EQ0EyDoVL1A==", + "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 + }, + "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-get-type": { + "version": "26.3.0", + "resolved": "https://registry.npmjs.org/jest-get-type/-/jest-get-type-26.3.0.tgz", + "integrity": "sha512-TpfaviN1R2pQWkIihlfEanwOXK0zcxrKEE4MlU6Tn7keoXdN6/3gK/xl0yEh8DOunn5pOVGKf8hB4R9gVh04ig==", + "dev": true + }, + "pretty-format": { + "version": "26.6.2", + "resolved": "https://registry.npmjs.org/pretty-format/-/pretty-format-26.6.2.tgz", + "integrity": "sha512-7AeGuCYNGmycyQbCqd/3PWH4eOoX/OiCa0uphp57NVTeAGdJGaAliecxwBDHYQCIvrW7aDBZCYeNTP/WX69mkg==", + "dev": true, + "requires": { + "@jest/types": "^26.6.2", + "ansi-regex": "^5.0.0", + "ansi-styles": "^4.0.0", + "react-is": "^17.0.1" + } + }, + "react-is": { + "version": "17.0.1", + "resolved": "https://registry.npmjs.org/react-is/-/react-is-17.0.1.tgz", + "integrity": "sha512-NAnt2iGDXohE5LI7uBnLnqvLQMtzhkiAOLXTmv+qnF9Ky7xAPcX8Up/xWIhxvLVGJvuLiNc4xQLtuqDRzb4fSA==", + "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" + } + } + } + }, + "jest-environment-jsdom": { + "version": "26.6.2", + "resolved": "https://registry.npmjs.org/jest-environment-jsdom/-/jest-environment-jsdom-26.6.2.tgz", + "integrity": "sha512-jgPqCruTlt3Kwqg5/WVFyHIOJHsiAvhcp2qiR2QQstuG9yWox5+iHpU3ZrcBxW14T4fe5Z68jAfLRh7joCSP2Q==", + "dev": true, + "requires": { + "@jest/environment": "^26.6.2", + "@jest/fake-timers": "^26.6.2", + "@jest/types": "^26.6.2", + "@types/node": "*", + "jest-mock": "^26.6.2", + "jest-util": "^26.6.2", + "jsdom": "^16.4.0" + }, + "dependencies": { + "@jest/types": { + "version": "26.6.2", + "resolved": "https://registry.npmjs.org/@jest/types/-/types-26.6.2.tgz", + "integrity": "sha512-fC6QCp7Sc5sX6g8Tvbmj4XUTbyrik0akgRy03yjXbQaBWWNWGE7SGtJk98m0N8nzegD/7SggrUlivxo5ax4KWQ==", + "dev": true, + "requires": { + "@types/istanbul-lib-coverage": "^2.0.0", + "@types/istanbul-reports": "^3.0.0", + "@types/node": "*", + "@types/yargs": "^15.0.0", + "chalk": "^4.0.0" + } + }, + "@types/istanbul-reports": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/@types/istanbul-reports/-/istanbul-reports-3.0.0.tgz", + "integrity": "sha512-nwKNbvnwJ2/mndE9ItP/zc2TCzw6uuodnF4EHYWD+gCQDVBuRQL5UzbZD0/ezy1iKsFU2ZQiDqg4M9dN4+wZgA==", + "dev": true, + "requires": { + "@types/istanbul-lib-report": "*" + } + }, + "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.0", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.0.tgz", + "integrity": "sha512-qwx12AxXe2Q5xQ43Ac//I6v5aXTipYrSESdOgzrN+9XjgEpyjpKuvSGaN4qE93f7TQTlerQQ8S+EQ0EyDoVL1A==", + "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 + }, + "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" + } + } + } + }, + "jest-environment-node": { + "version": "26.6.2", + "resolved": "https://registry.npmjs.org/jest-environment-node/-/jest-environment-node-26.6.2.tgz", + "integrity": "sha512-zhtMio3Exty18dy8ee8eJ9kjnRyZC1N4C1Nt/VShN1apyXc8rWGtJ9lI7vqiWcyyXS4BVSEn9lxAM2D+07/Tag==", + "dev": true, + "requires": { + "@jest/environment": "^26.6.2", + "@jest/fake-timers": "^26.6.2", + "@jest/types": "^26.6.2", + "@types/node": "*", + "jest-mock": "^26.6.2", + "jest-util": "^26.6.2" + }, + "dependencies": { + "@jest/types": { + "version": "26.6.2", + "resolved": "https://registry.npmjs.org/@jest/types/-/types-26.6.2.tgz", + "integrity": "sha512-fC6QCp7Sc5sX6g8Tvbmj4XUTbyrik0akgRy03yjXbQaBWWNWGE7SGtJk98m0N8nzegD/7SggrUlivxo5ax4KWQ==", + "dev": true, + "requires": { + "@types/istanbul-lib-coverage": "^2.0.0", + "@types/istanbul-reports": "^3.0.0", + "@types/node": "*", + "@types/yargs": "^15.0.0", + "chalk": "^4.0.0" + } + }, + "@types/istanbul-reports": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/@types/istanbul-reports/-/istanbul-reports-3.0.0.tgz", + "integrity": "sha512-nwKNbvnwJ2/mndE9ItP/zc2TCzw6uuodnF4EHYWD+gCQDVBuRQL5UzbZD0/ezy1iKsFU2ZQiDqg4M9dN4+wZgA==", + "dev": true, + "requires": { + "@types/istanbul-lib-report": "*" + } + }, + "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.0", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.0.tgz", + "integrity": "sha512-qwx12AxXe2Q5xQ43Ac//I6v5aXTipYrSESdOgzrN+9XjgEpyjpKuvSGaN4qE93f7TQTlerQQ8S+EQ0EyDoVL1A==", + "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 + }, + "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" + } + } + } + }, + "jest-get-type": { + "version": "25.2.6", + "resolved": "https://registry.npmjs.org/jest-get-type/-/jest-get-type-25.2.6.tgz", + "integrity": "sha512-DxjtyzOHjObRM+sM1knti6or+eOgcGU4xVSb2HNP1TqO4ahsT+rqZg+nyqHWJSvWgKC5cG3QjGFBqxLghiF/Ig==", + "dev": true + }, + "jest-haste-map": { + "version": "26.6.2", + "resolved": "https://registry.npmjs.org/jest-haste-map/-/jest-haste-map-26.6.2.tgz", + "integrity": "sha512-easWIJXIw71B2RdR8kgqpjQrbMRWQBgiBwXYEhtGUTaX+doCjBheluShdDMeR8IMfJiTqH4+zfhtg29apJf/8w==", + "dev": true, + "requires": { + "@jest/types": "^26.6.2", + "@types/graceful-fs": "^4.1.2", + "@types/node": "*", + "anymatch": "^3.0.3", + "fb-watchman": "^2.0.0", + "fsevents": "^2.1.2", + "graceful-fs": "^4.2.4", + "jest-regex-util": "^26.0.0", + "jest-serializer": "^26.6.2", + "jest-util": "^26.6.2", + "jest-worker": "^26.6.2", + "micromatch": "^4.0.2", + "sane": "^4.0.3", + "walker": "^1.0.7" + }, + "dependencies": { + "@jest/types": { + "version": "26.6.2", + "resolved": "https://registry.npmjs.org/@jest/types/-/types-26.6.2.tgz", + "integrity": "sha512-fC6QCp7Sc5sX6g8Tvbmj4XUTbyrik0akgRy03yjXbQaBWWNWGE7SGtJk98m0N8nzegD/7SggrUlivxo5ax4KWQ==", + "dev": true, + "requires": { + "@types/istanbul-lib-coverage": "^2.0.0", + "@types/istanbul-reports": "^3.0.0", + "@types/node": "*", + "@types/yargs": "^15.0.0", + "chalk": "^4.0.0" + } + }, + "@types/istanbul-reports": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/@types/istanbul-reports/-/istanbul-reports-3.0.0.tgz", + "integrity": "sha512-nwKNbvnwJ2/mndE9ItP/zc2TCzw6uuodnF4EHYWD+gCQDVBuRQL5UzbZD0/ezy1iKsFU2ZQiDqg4M9dN4+wZgA==", + "dev": true, + "requires": { + "@types/istanbul-lib-report": "*" + } + }, + "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.0", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.0.tgz", + "integrity": "sha512-qwx12AxXe2Q5xQ43Ac//I6v5aXTipYrSESdOgzrN+9XjgEpyjpKuvSGaN4qE93f7TQTlerQQ8S+EQ0EyDoVL1A==", + "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 + }, + "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 + }, + "micromatch": { + "version": "4.0.2", + "resolved": "https://registry.npmjs.org/micromatch/-/micromatch-4.0.2.tgz", + "integrity": "sha512-y7FpHSbMUMoyPbYUSzO6PaZ6FyRnQOpHuKwbo1G+Knck95XVU4QAiKdGEnj5wwoS7PlOgthX/09u5iFJ+aYf5Q==", + "dev": true, + "requires": { + "braces": "^3.0.1", + "picomatch": "^2.0.5" + } + }, + "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" + } + } + } + }, + "jest-jasmine2": { + "version": "26.6.3", + "resolved": "https://registry.npmjs.org/jest-jasmine2/-/jest-jasmine2-26.6.3.tgz", + "integrity": "sha512-kPKUrQtc8aYwBV7CqBg5pu+tmYXlvFlSFYn18ev4gPFtrRzB15N2gW/Roew3187q2w2eHuu0MU9TJz6w0/nPEg==", + "dev": true, + "requires": { + "@babel/traverse": "^7.1.0", + "@jest/environment": "^26.6.2", + "@jest/source-map": "^26.6.2", + "@jest/test-result": "^26.6.2", + "@jest/types": "^26.6.2", + "@types/node": "*", + "chalk": "^4.0.0", + "co": "^4.6.0", + "expect": "^26.6.2", + "is-generator-fn": "^2.0.0", + "jest-each": "^26.6.2", + "jest-matcher-utils": "^26.6.2", + "jest-message-util": "^26.6.2", + "jest-runtime": "^26.6.3", + "jest-snapshot": "^26.6.2", + "jest-util": "^26.6.2", + "pretty-format": "^26.6.2", + "throat": "^5.0.0" + }, + "dependencies": { + "@jest/types": { + "version": "26.6.2", + "resolved": "https://registry.npmjs.org/@jest/types/-/types-26.6.2.tgz", + "integrity": "sha512-fC6QCp7Sc5sX6g8Tvbmj4XUTbyrik0akgRy03yjXbQaBWWNWGE7SGtJk98m0N8nzegD/7SggrUlivxo5ax4KWQ==", + "dev": true, + "requires": { + "@types/istanbul-lib-coverage": "^2.0.0", + "@types/istanbul-reports": "^3.0.0", + "@types/node": "*", + "@types/yargs": "^15.0.0", + "chalk": "^4.0.0" + } + }, + "@types/istanbul-reports": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/@types/istanbul-reports/-/istanbul-reports-3.0.0.tgz", + "integrity": "sha512-nwKNbvnwJ2/mndE9ItP/zc2TCzw6uuodnF4EHYWD+gCQDVBuRQL5UzbZD0/ezy1iKsFU2ZQiDqg4M9dN4+wZgA==", + "dev": true, + "requires": { + "@types/istanbul-lib-report": "*" + } + }, + "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.0", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.0.tgz", + "integrity": "sha512-qwx12AxXe2Q5xQ43Ac//I6v5aXTipYrSESdOgzrN+9XjgEpyjpKuvSGaN4qE93f7TQTlerQQ8S+EQ0EyDoVL1A==", + "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 + }, + "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 + }, + "pretty-format": { + "version": "26.6.2", + "resolved": "https://registry.npmjs.org/pretty-format/-/pretty-format-26.6.2.tgz", + "integrity": "sha512-7AeGuCYNGmycyQbCqd/3PWH4eOoX/OiCa0uphp57NVTeAGdJGaAliecxwBDHYQCIvrW7aDBZCYeNTP/WX69mkg==", + "dev": true, + "requires": { + "@jest/types": "^26.6.2", + "ansi-regex": "^5.0.0", + "ansi-styles": "^4.0.0", + "react-is": "^17.0.1" + } + }, + "react-is": { + "version": "17.0.1", + "resolved": "https://registry.npmjs.org/react-is/-/react-is-17.0.1.tgz", + "integrity": "sha512-NAnt2iGDXohE5LI7uBnLnqvLQMtzhkiAOLXTmv+qnF9Ky7xAPcX8Up/xWIhxvLVGJvuLiNc4xQLtuqDRzb4fSA==", + "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" + } + } + } + }, + "jest-junit": { + "version": "11.0.1", + "resolved": "https://registry.npmjs.org/jest-junit/-/jest-junit-11.0.1.tgz", + "integrity": "sha512-stgc0mBoiSg/F9qWd4KkmR3K7Nk2u+M/dc1oup7gxz9mrzGcEaU2YL9/0QscVqqg3IOaA1P5ZXtozG/XR6j6nw==", + "dev": true, + "requires": { + "mkdirp": "^1.0.4", + "strip-ansi": "^5.2.0", + "uuid": "^3.3.3", + "xml": "^1.0.1" + }, + "dependencies": { + "ansi-regex": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-4.1.0.tgz", + "integrity": "sha512-1apePfXM1UOSqw0o9IiFAovVz9M5S1Dg+4TrDwfMewQ6p/rmMueb7tWZjQ1rx4Loy1ArBggoqGpfqqdI4rondg==", + "dev": true + }, + "mkdirp": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/mkdirp/-/mkdirp-1.0.4.tgz", + "integrity": "sha512-vVqVZQyf3WLx2Shd0qJ9xuvqgAyKPLAiqITEtqW0oIUjzo3PePDd6fW9iFz30ef7Ysp/oiWqbhszeGWW2T6Gzw==", + "dev": true + }, + "strip-ansi": { + "version": "5.2.0", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-5.2.0.tgz", + "integrity": "sha512-DuRs1gKbBqsMKIZlrffwlug8MHkcnpjs5VPmL1PAh+mA30U0DTotfDZ0d2UUsXpPmPmMMJ6W773MaA3J+lbiWA==", + "dev": true, + "requires": { + "ansi-regex": "^4.1.0" + } + }, + "uuid": { + "version": "3.4.0", + "resolved": "https://registry.npmjs.org/uuid/-/uuid-3.4.0.tgz", + "integrity": "sha512-HjSDRw6gZE5JMggctHBcjVak08+KEVhSIiDzFnT9S9aegmp85S/bReBVTb4QTFaRNptJ9kuYaNhnbNEOkbKb/A==", + "dev": true + } + } + }, + "jest-leak-detector": { + "version": "26.6.2", + "resolved": "https://registry.npmjs.org/jest-leak-detector/-/jest-leak-detector-26.6.2.tgz", + "integrity": "sha512-i4xlXpsVSMeKvg2cEKdfhh0H39qlJlP5Ex1yQxwF9ubahboQYMgTtz5oML35AVA3B4Eu+YsmwaiKVev9KCvLxg==", + "dev": true, + "requires": { + "jest-get-type": "^26.3.0", + "pretty-format": "^26.6.2" + }, + "dependencies": { + "@jest/types": { + "version": "26.6.2", + "resolved": "https://registry.npmjs.org/@jest/types/-/types-26.6.2.tgz", + "integrity": "sha512-fC6QCp7Sc5sX6g8Tvbmj4XUTbyrik0akgRy03yjXbQaBWWNWGE7SGtJk98m0N8nzegD/7SggrUlivxo5ax4KWQ==", + "dev": true, + "requires": { + "@types/istanbul-lib-coverage": "^2.0.0", + "@types/istanbul-reports": "^3.0.0", + "@types/node": "*", + "@types/yargs": "^15.0.0", + "chalk": "^4.0.0" + } + }, + "@types/istanbul-reports": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/@types/istanbul-reports/-/istanbul-reports-3.0.0.tgz", + "integrity": "sha512-nwKNbvnwJ2/mndE9ItP/zc2TCzw6uuodnF4EHYWD+gCQDVBuRQL5UzbZD0/ezy1iKsFU2ZQiDqg4M9dN4+wZgA==", + "dev": true, + "requires": { + "@types/istanbul-lib-report": "*" + } + }, + "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.0", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.0.tgz", + "integrity": "sha512-qwx12AxXe2Q5xQ43Ac//I6v5aXTipYrSESdOgzrN+9XjgEpyjpKuvSGaN4qE93f7TQTlerQQ8S+EQ0EyDoVL1A==", + "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 + }, + "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-get-type": { + "version": "26.3.0", + "resolved": "https://registry.npmjs.org/jest-get-type/-/jest-get-type-26.3.0.tgz", + "integrity": "sha512-TpfaviN1R2pQWkIihlfEanwOXK0zcxrKEE4MlU6Tn7keoXdN6/3gK/xl0yEh8DOunn5pOVGKf8hB4R9gVh04ig==", + "dev": true + }, + "pretty-format": { + "version": "26.6.2", + "resolved": "https://registry.npmjs.org/pretty-format/-/pretty-format-26.6.2.tgz", + "integrity": "sha512-7AeGuCYNGmycyQbCqd/3PWH4eOoX/OiCa0uphp57NVTeAGdJGaAliecxwBDHYQCIvrW7aDBZCYeNTP/WX69mkg==", + "dev": true, + "requires": { + "@jest/types": "^26.6.2", + "ansi-regex": "^5.0.0", + "ansi-styles": "^4.0.0", + "react-is": "^17.0.1" + } + }, + "react-is": { + "version": "17.0.1", + "resolved": "https://registry.npmjs.org/react-is/-/react-is-17.0.1.tgz", + "integrity": "sha512-NAnt2iGDXohE5LI7uBnLnqvLQMtzhkiAOLXTmv+qnF9Ky7xAPcX8Up/xWIhxvLVGJvuLiNc4xQLtuqDRzb4fSA==", + "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" + } + } + } + }, + "jest-matcher-utils": { + "version": "26.6.2", + "resolved": "https://registry.npmjs.org/jest-matcher-utils/-/jest-matcher-utils-26.6.2.tgz", + "integrity": "sha512-llnc8vQgYcNqDrqRDXWwMr9i7rS5XFiCwvh6DTP7Jqa2mqpcCBBlpCbn+trkG0KNhPu/h8rzyBkriOtBstvWhw==", + "dev": true, + "requires": { + "chalk": "^4.0.0", + "jest-diff": "^26.6.2", + "jest-get-type": "^26.3.0", + "pretty-format": "^26.6.2" + }, + "dependencies": { + "@jest/types": { + "version": "26.6.2", + "resolved": "https://registry.npmjs.org/@jest/types/-/types-26.6.2.tgz", + "integrity": "sha512-fC6QCp7Sc5sX6g8Tvbmj4XUTbyrik0akgRy03yjXbQaBWWNWGE7SGtJk98m0N8nzegD/7SggrUlivxo5ax4KWQ==", + "dev": true, + "requires": { + "@types/istanbul-lib-coverage": "^2.0.0", + "@types/istanbul-reports": "^3.0.0", + "@types/node": "*", + "@types/yargs": "^15.0.0", + "chalk": "^4.0.0" + } + }, + "@types/istanbul-reports": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/@types/istanbul-reports/-/istanbul-reports-3.0.0.tgz", + "integrity": "sha512-nwKNbvnwJ2/mndE9ItP/zc2TCzw6uuodnF4EHYWD+gCQDVBuRQL5UzbZD0/ezy1iKsFU2ZQiDqg4M9dN4+wZgA==", + "dev": true, + "requires": { + "@types/istanbul-lib-report": "*" + } + }, + "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.0", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.0.tgz", + "integrity": "sha512-qwx12AxXe2Q5xQ43Ac//I6v5aXTipYrSESdOgzrN+9XjgEpyjpKuvSGaN4qE93f7TQTlerQQ8S+EQ0EyDoVL1A==", + "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 + }, + "diff-sequences": { + "version": "26.6.2", + "resolved": "https://registry.npmjs.org/diff-sequences/-/diff-sequences-26.6.2.tgz", + "integrity": "sha512-Mv/TDa3nZ9sbc5soK+OoA74BsS3mL37yixCvUAQkiuA4Wz6YtwP/K47n2rv2ovzHZvoiQeA5FTQOschKkEwB0Q==", + "dev": true + }, + "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-diff": { + "version": "26.6.2", + "resolved": "https://registry.npmjs.org/jest-diff/-/jest-diff-26.6.2.tgz", + "integrity": "sha512-6m+9Z3Gv9wN0WFVasqjCL/06+EFCMTqDEUl/b87HYK2rAPTyfz4ZIuSlPhY51PIQRWx5TaxeF1qmXKe9gfN3sA==", + "dev": true, + "requires": { + "chalk": "^4.0.0", + "diff-sequences": "^26.6.2", + "jest-get-type": "^26.3.0", + "pretty-format": "^26.6.2" + } + }, + "jest-get-type": { + "version": "26.3.0", + "resolved": "https://registry.npmjs.org/jest-get-type/-/jest-get-type-26.3.0.tgz", + "integrity": "sha512-TpfaviN1R2pQWkIihlfEanwOXK0zcxrKEE4MlU6Tn7keoXdN6/3gK/xl0yEh8DOunn5pOVGKf8hB4R9gVh04ig==", + "dev": true + }, + "pretty-format": { + "version": "26.6.2", + "resolved": "https://registry.npmjs.org/pretty-format/-/pretty-format-26.6.2.tgz", + "integrity": "sha512-7AeGuCYNGmycyQbCqd/3PWH4eOoX/OiCa0uphp57NVTeAGdJGaAliecxwBDHYQCIvrW7aDBZCYeNTP/WX69mkg==", + "dev": true, + "requires": { + "@jest/types": "^26.6.2", + "ansi-regex": "^5.0.0", + "ansi-styles": "^4.0.0", + "react-is": "^17.0.1" + } + }, + "react-is": { + "version": "17.0.1", + "resolved": "https://registry.npmjs.org/react-is/-/react-is-17.0.1.tgz", + "integrity": "sha512-NAnt2iGDXohE5LI7uBnLnqvLQMtzhkiAOLXTmv+qnF9Ky7xAPcX8Up/xWIhxvLVGJvuLiNc4xQLtuqDRzb4fSA==", + "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" + } + } + } + }, + "jest-message-util": { + "version": "26.6.2", + "resolved": "https://registry.npmjs.org/jest-message-util/-/jest-message-util-26.6.2.tgz", + "integrity": "sha512-rGiLePzQ3AzwUshu2+Rn+UMFk0pHN58sOG+IaJbk5Jxuqo3NYO1U2/MIR4S1sKgsoYSXSzdtSa0TgrmtUwEbmA==", + "dev": true, + "requires": { + "@babel/code-frame": "^7.0.0", + "@jest/types": "^26.6.2", + "@types/stack-utils": "^2.0.0", + "chalk": "^4.0.0", + "graceful-fs": "^4.2.4", + "micromatch": "^4.0.2", + "pretty-format": "^26.6.2", + "slash": "^3.0.0", + "stack-utils": "^2.0.2" + }, + "dependencies": { + "@jest/types": { + "version": "26.6.2", + "resolved": "https://registry.npmjs.org/@jest/types/-/types-26.6.2.tgz", + "integrity": "sha512-fC6QCp7Sc5sX6g8Tvbmj4XUTbyrik0akgRy03yjXbQaBWWNWGE7SGtJk98m0N8nzegD/7SggrUlivxo5ax4KWQ==", + "dev": true, + "requires": { + "@types/istanbul-lib-coverage": "^2.0.0", + "@types/istanbul-reports": "^3.0.0", + "@types/node": "*", + "@types/yargs": "^15.0.0", + "chalk": "^4.0.0" + } + }, + "@types/istanbul-reports": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/@types/istanbul-reports/-/istanbul-reports-3.0.0.tgz", + "integrity": "sha512-nwKNbvnwJ2/mndE9ItP/zc2TCzw6uuodnF4EHYWD+gCQDVBuRQL5UzbZD0/ezy1iKsFU2ZQiDqg4M9dN4+wZgA==", + "dev": true, + "requires": { + "@types/istanbul-lib-report": "*" + } + }, + "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.0", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.0.tgz", + "integrity": "sha512-qwx12AxXe2Q5xQ43Ac//I6v5aXTipYrSESdOgzrN+9XjgEpyjpKuvSGaN4qE93f7TQTlerQQ8S+EQ0EyDoVL1A==", + "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 + }, + "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 + }, + "micromatch": { + "version": "4.0.2", + "resolved": "https://registry.npmjs.org/micromatch/-/micromatch-4.0.2.tgz", + "integrity": "sha512-y7FpHSbMUMoyPbYUSzO6PaZ6FyRnQOpHuKwbo1G+Knck95XVU4QAiKdGEnj5wwoS7PlOgthX/09u5iFJ+aYf5Q==", + "dev": true, + "requires": { + "braces": "^3.0.1", + "picomatch": "^2.0.5" + } + }, + "pretty-format": { + "version": "26.6.2", + "resolved": "https://registry.npmjs.org/pretty-format/-/pretty-format-26.6.2.tgz", + "integrity": "sha512-7AeGuCYNGmycyQbCqd/3PWH4eOoX/OiCa0uphp57NVTeAGdJGaAliecxwBDHYQCIvrW7aDBZCYeNTP/WX69mkg==", + "dev": true, + "requires": { + "@jest/types": "^26.6.2", + "ansi-regex": "^5.0.0", + "ansi-styles": "^4.0.0", + "react-is": "^17.0.1" + } + }, + "react-is": { + "version": "17.0.1", + "resolved": "https://registry.npmjs.org/react-is/-/react-is-17.0.1.tgz", + "integrity": "sha512-NAnt2iGDXohE5LI7uBnLnqvLQMtzhkiAOLXTmv+qnF9Ky7xAPcX8Up/xWIhxvLVGJvuLiNc4xQLtuqDRzb4fSA==", + "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" + } + } + } + }, + "jest-mock": { + "version": "26.6.2", + "resolved": "https://registry.npmjs.org/jest-mock/-/jest-mock-26.6.2.tgz", + "integrity": "sha512-YyFjePHHp1LzpzYcmgqkJ0nm0gg/lJx2aZFzFy1S6eUqNjXsOqTK10zNRff2dNfssgokjkG65OlWNcIlgd3zew==", + "dev": true, + "requires": { + "@jest/types": "^26.6.2", + "@types/node": "*" + }, + "dependencies": { + "@jest/types": { + "version": "26.6.2", + "resolved": "https://registry.npmjs.org/@jest/types/-/types-26.6.2.tgz", + "integrity": "sha512-fC6QCp7Sc5sX6g8Tvbmj4XUTbyrik0akgRy03yjXbQaBWWNWGE7SGtJk98m0N8nzegD/7SggrUlivxo5ax4KWQ==", + "dev": true, + "requires": { + "@types/istanbul-lib-coverage": "^2.0.0", + "@types/istanbul-reports": "^3.0.0", + "@types/node": "*", + "@types/yargs": "^15.0.0", + "chalk": "^4.0.0" + } + }, + "@types/istanbul-reports": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/@types/istanbul-reports/-/istanbul-reports-3.0.0.tgz", + "integrity": "sha512-nwKNbvnwJ2/mndE9ItP/zc2TCzw6uuodnF4EHYWD+gCQDVBuRQL5UzbZD0/ezy1iKsFU2ZQiDqg4M9dN4+wZgA==", + "dev": true, + "requires": { + "@types/istanbul-lib-report": "*" + } + }, + "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.0", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.0.tgz", + "integrity": "sha512-qwx12AxXe2Q5xQ43Ac//I6v5aXTipYrSESdOgzrN+9XjgEpyjpKuvSGaN4qE93f7TQTlerQQ8S+EQ0EyDoVL1A==", + "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 + }, + "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" + } + } + } + }, + "jest-pnp-resolver": { + "version": "1.2.2", + "resolved": "https://registry.npmjs.org/jest-pnp-resolver/-/jest-pnp-resolver-1.2.2.tgz", + "integrity": "sha512-olV41bKSMm8BdnuMsewT4jqlZ8+3TCARAXjZGT9jcoSnrfUnRCqnMoF9XEeoWjbzObpqF9dRhHQj0Xb9QdF6/w==", + "dev": true + }, + "jest-preset-angular": { + "version": "8.2.1", + "resolved": "https://registry.npmjs.org/jest-preset-angular/-/jest-preset-angular-8.2.1.tgz", + "integrity": "sha512-6t7lavnIHnZWz6a03jpZ5L7phMefi6SoBIRZ/GQdyML8YgwMtbJszbhUE+dh5lzmeNbd6AJ1gfleetGy2Rg4cQ==", + "dev": true, + "requires": { + "pretty-format": "^26.0.0", + "ts-jest": "^26.0.0" + }, + "dependencies": { + "@jest/types": { + "version": "26.6.2", + "resolved": "https://registry.npmjs.org/@jest/types/-/types-26.6.2.tgz", + "integrity": "sha512-fC6QCp7Sc5sX6g8Tvbmj4XUTbyrik0akgRy03yjXbQaBWWNWGE7SGtJk98m0N8nzegD/7SggrUlivxo5ax4KWQ==", + "dev": true, + "requires": { + "@types/istanbul-lib-coverage": "^2.0.0", + "@types/istanbul-reports": "^3.0.0", + "@types/node": "*", + "@types/yargs": "^15.0.0", + "chalk": "^4.0.0" + } + }, + "@types/istanbul-reports": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/@types/istanbul-reports/-/istanbul-reports-3.0.0.tgz", + "integrity": "sha512-nwKNbvnwJ2/mndE9ItP/zc2TCzw6uuodnF4EHYWD+gCQDVBuRQL5UzbZD0/ezy1iKsFU2ZQiDqg4M9dN4+wZgA==", + "dev": true, + "requires": { + "@types/istanbul-lib-report": "*" + } + }, + "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.0", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.0.tgz", + "integrity": "sha512-qwx12AxXe2Q5xQ43Ac//I6v5aXTipYrSESdOgzrN+9XjgEpyjpKuvSGaN4qE93f7TQTlerQQ8S+EQ0EyDoVL1A==", + "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 + }, + "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 + }, + "pretty-format": { + "version": "26.6.2", + "resolved": "https://registry.npmjs.org/pretty-format/-/pretty-format-26.6.2.tgz", + "integrity": "sha512-7AeGuCYNGmycyQbCqd/3PWH4eOoX/OiCa0uphp57NVTeAGdJGaAliecxwBDHYQCIvrW7aDBZCYeNTP/WX69mkg==", + "dev": true, + "requires": { + "@jest/types": "^26.6.2", + "ansi-regex": "^5.0.0", + "ansi-styles": "^4.0.0", + "react-is": "^17.0.1" + } + }, + "react-is": { + "version": "17.0.1", + "resolved": "https://registry.npmjs.org/react-is/-/react-is-17.0.1.tgz", + "integrity": "sha512-NAnt2iGDXohE5LI7uBnLnqvLQMtzhkiAOLXTmv+qnF9Ky7xAPcX8Up/xWIhxvLVGJvuLiNc4xQLtuqDRzb4fSA==", + "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" + } + } + } + }, + "jest-regex-util": { + "version": "26.0.0", + "resolved": "https://registry.npmjs.org/jest-regex-util/-/jest-regex-util-26.0.0.tgz", + "integrity": "sha512-Gv3ZIs/nA48/Zvjrl34bf+oD76JHiGDUxNOVgUjh3j890sblXryjY4rss71fPtD/njchl6PSE2hIhvyWa1eT0A==", + "dev": true + }, + "jest-resolve": { + "version": "26.6.2", + "resolved": "https://registry.npmjs.org/jest-resolve/-/jest-resolve-26.6.2.tgz", + "integrity": "sha512-sOxsZOq25mT1wRsfHcbtkInS+Ek7Q8jCHUB0ZUTP0tc/c41QHriU/NunqMfCUWsL4H3MHpvQD4QR9kSYhS7UvQ==", + "dev": true, + "requires": { + "@jest/types": "^26.6.2", + "chalk": "^4.0.0", + "graceful-fs": "^4.2.4", + "jest-pnp-resolver": "^1.2.2", + "jest-util": "^26.6.2", + "read-pkg-up": "^7.0.1", + "resolve": "^1.18.1", + "slash": "^3.0.0" + }, + "dependencies": { + "@jest/types": { + "version": "26.6.2", + "resolved": "https://registry.npmjs.org/@jest/types/-/types-26.6.2.tgz", + "integrity": "sha512-fC6QCp7Sc5sX6g8Tvbmj4XUTbyrik0akgRy03yjXbQaBWWNWGE7SGtJk98m0N8nzegD/7SggrUlivxo5ax4KWQ==", + "dev": true, + "requires": { + "@types/istanbul-lib-coverage": "^2.0.0", + "@types/istanbul-reports": "^3.0.0", + "@types/node": "*", + "@types/yargs": "^15.0.0", + "chalk": "^4.0.0" + } + }, + "@types/istanbul-reports": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/@types/istanbul-reports/-/istanbul-reports-3.0.0.tgz", + "integrity": "sha512-nwKNbvnwJ2/mndE9ItP/zc2TCzw6uuodnF4EHYWD+gCQDVBuRQL5UzbZD0/ezy1iKsFU2ZQiDqg4M9dN4+wZgA==", + "dev": true, + "requires": { + "@types/istanbul-lib-report": "*" + } + }, + "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.0", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.0.tgz", + "integrity": "sha512-qwx12AxXe2Q5xQ43Ac//I6v5aXTipYrSESdOgzrN+9XjgEpyjpKuvSGaN4qE93f7TQTlerQQ8S+EQ0EyDoVL1A==", + "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 + }, + "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 + }, + "parse-json": { + "version": "5.1.0", + "resolved": "https://registry.npmjs.org/parse-json/-/parse-json-5.1.0.tgz", + "integrity": "sha512-+mi/lmVVNKFNVyLXV31ERiy2CY5E1/F6QtJFEzoChPRwwngMNXRDQ9GJ5WdE2Z2P4AujsOi0/+2qHID68KwfIQ==", + "dev": true, + "requires": { + "@babel/code-frame": "^7.0.0", + "error-ex": "^1.3.1", + "json-parse-even-better-errors": "^2.3.0", + "lines-and-columns": "^1.1.6" + } + }, + "read-pkg": { + "version": "5.2.0", + "resolved": "https://registry.npmjs.org/read-pkg/-/read-pkg-5.2.0.tgz", + "integrity": "sha512-Ug69mNOpfvKDAc2Q8DRpMjjzdtrnv9HcSMX+4VsZxD1aZ6ZzrIE7rlzXBtWTyhULSMKg076AW6WR5iZpD0JiOg==", + "dev": true, + "requires": { + "@types/normalize-package-data": "^2.4.0", + "normalize-package-data": "^2.5.0", + "parse-json": "^5.0.0", + "type-fest": "^0.6.0" + }, + "dependencies": { + "type-fest": { + "version": "0.6.0", + "resolved": "https://registry.npmjs.org/type-fest/-/type-fest-0.6.0.tgz", + "integrity": "sha512-q+MB8nYR1KDLrgr4G5yemftpMC7/QLqVndBmEEdqzmNj5dcFOO4Oo8qlwZE3ULT3+Zim1F8Kq4cBnikNhlCMlg==", + "dev": true + } + } + }, + "read-pkg-up": { + "version": "7.0.1", + "resolved": "https://registry.npmjs.org/read-pkg-up/-/read-pkg-up-7.0.1.tgz", + "integrity": "sha512-zK0TB7Xd6JpCLmlLmufqykGE+/TlOePD6qKClNW7hHDKFh/J7/7gCWGR7joEQEW1bKq3a3yUZSObOoWLFQ4ohg==", + "dev": true, + "requires": { + "find-up": "^4.1.0", + "read-pkg": "^5.2.0", + "type-fest": "^0.8.1" + } + }, + "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" + } + }, + "type-fest": { + "version": "0.8.1", + "resolved": "https://registry.npmjs.org/type-fest/-/type-fest-0.8.1.tgz", + "integrity": "sha512-4dbzIzqvjtgiM5rw1k5rEHtBANKmdudhGyBEajN01fEyhaAIhsoKNy6y7+IN93IfpFtwY9iqi7kD+xwKhQsNJA==", + "dev": true + } + } + }, + "jest-resolve-dependencies": { + "version": "26.6.3", + "resolved": "https://registry.npmjs.org/jest-resolve-dependencies/-/jest-resolve-dependencies-26.6.3.tgz", + "integrity": "sha512-pVwUjJkxbhe4RY8QEWzN3vns2kqyuldKpxlxJlzEYfKSvY6/bMvxoFrYYzUO1Gx28yKWN37qyV7rIoIp2h8fTg==", + "dev": true, + "requires": { + "@jest/types": "^26.6.2", + "jest-regex-util": "^26.0.0", + "jest-snapshot": "^26.6.2" + }, + "dependencies": { + "@jest/types": { + "version": "26.6.2", + "resolved": "https://registry.npmjs.org/@jest/types/-/types-26.6.2.tgz", + "integrity": "sha512-fC6QCp7Sc5sX6g8Tvbmj4XUTbyrik0akgRy03yjXbQaBWWNWGE7SGtJk98m0N8nzegD/7SggrUlivxo5ax4KWQ==", + "dev": true, + "requires": { + "@types/istanbul-lib-coverage": "^2.0.0", + "@types/istanbul-reports": "^3.0.0", + "@types/node": "*", + "@types/yargs": "^15.0.0", + "chalk": "^4.0.0" + } + }, + "@types/istanbul-reports": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/@types/istanbul-reports/-/istanbul-reports-3.0.0.tgz", + "integrity": "sha512-nwKNbvnwJ2/mndE9ItP/zc2TCzw6uuodnF4EHYWD+gCQDVBuRQL5UzbZD0/ezy1iKsFU2ZQiDqg4M9dN4+wZgA==", + "dev": true, + "requires": { + "@types/istanbul-lib-report": "*" + } + }, + "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.0", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.0.tgz", + "integrity": "sha512-qwx12AxXe2Q5xQ43Ac//I6v5aXTipYrSESdOgzrN+9XjgEpyjpKuvSGaN4qE93f7TQTlerQQ8S+EQ0EyDoVL1A==", + "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 + }, + "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" + } + } + } + }, + "jest-runner": { + "version": "26.6.3", + "resolved": "https://registry.npmjs.org/jest-runner/-/jest-runner-26.6.3.tgz", + "integrity": "sha512-atgKpRHnaA2OvByG/HpGA4g6CSPS/1LK0jK3gATJAoptC1ojltpmVlYC3TYgdmGp+GLuhzpH30Gvs36szSL2JQ==", + "dev": true, + "requires": { + "@jest/console": "^26.6.2", + "@jest/environment": "^26.6.2", + "@jest/test-result": "^26.6.2", + "@jest/types": "^26.6.2", + "@types/node": "*", + "chalk": "^4.0.0", + "emittery": "^0.7.1", + "exit": "^0.1.2", + "graceful-fs": "^4.2.4", + "jest-config": "^26.6.3", + "jest-docblock": "^26.0.0", + "jest-haste-map": "^26.6.2", + "jest-leak-detector": "^26.6.2", + "jest-message-util": "^26.6.2", + "jest-resolve": "^26.6.2", + "jest-runtime": "^26.6.3", + "jest-util": "^26.6.2", + "jest-worker": "^26.6.2", + "source-map-support": "^0.5.6", + "throat": "^5.0.0" + }, + "dependencies": { + "@jest/types": { + "version": "26.6.2", + "resolved": "https://registry.npmjs.org/@jest/types/-/types-26.6.2.tgz", + "integrity": "sha512-fC6QCp7Sc5sX6g8Tvbmj4XUTbyrik0akgRy03yjXbQaBWWNWGE7SGtJk98m0N8nzegD/7SggrUlivxo5ax4KWQ==", + "dev": true, + "requires": { + "@types/istanbul-lib-coverage": "^2.0.0", + "@types/istanbul-reports": "^3.0.0", + "@types/node": "*", + "@types/yargs": "^15.0.0", + "chalk": "^4.0.0" + } + }, + "@types/istanbul-reports": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/@types/istanbul-reports/-/istanbul-reports-3.0.0.tgz", + "integrity": "sha512-nwKNbvnwJ2/mndE9ItP/zc2TCzw6uuodnF4EHYWD+gCQDVBuRQL5UzbZD0/ezy1iKsFU2ZQiDqg4M9dN4+wZgA==", + "dev": true, + "requires": { + "@types/istanbul-lib-report": "*" + } + }, + "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.0", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.0.tgz", + "integrity": "sha512-qwx12AxXe2Q5xQ43Ac//I6v5aXTipYrSESdOgzrN+9XjgEpyjpKuvSGaN4qE93f7TQTlerQQ8S+EQ0EyDoVL1A==", + "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 + }, + "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" + } + } + } + }, + "jest-runtime": { + "version": "26.6.3", + "resolved": "https://registry.npmjs.org/jest-runtime/-/jest-runtime-26.6.3.tgz", + "integrity": "sha512-lrzyR3N8sacTAMeonbqpnSka1dHNux2uk0qqDXVkMv2c/A3wYnvQ4EXuI013Y6+gSKSCxdaczvf4HF0mVXHRdw==", + "dev": true, + "requires": { + "@jest/console": "^26.6.2", + "@jest/environment": "^26.6.2", + "@jest/fake-timers": "^26.6.2", + "@jest/globals": "^26.6.2", + "@jest/source-map": "^26.6.2", + "@jest/test-result": "^26.6.2", + "@jest/transform": "^26.6.2", + "@jest/types": "^26.6.2", + "@types/yargs": "^15.0.0", + "chalk": "^4.0.0", + "cjs-module-lexer": "^0.6.0", + "collect-v8-coverage": "^1.0.0", + "exit": "^0.1.2", + "glob": "^7.1.3", + "graceful-fs": "^4.2.4", + "jest-config": "^26.6.3", + "jest-haste-map": "^26.6.2", + "jest-message-util": "^26.6.2", + "jest-mock": "^26.6.2", + "jest-regex-util": "^26.0.0", + "jest-resolve": "^26.6.2", + "jest-snapshot": "^26.6.2", + "jest-util": "^26.6.2", + "jest-validate": "^26.6.2", + "slash": "^3.0.0", + "strip-bom": "^4.0.0", + "yargs": "^15.4.1" + }, + "dependencies": { + "@jest/types": { + "version": "26.6.2", + "resolved": "https://registry.npmjs.org/@jest/types/-/types-26.6.2.tgz", + "integrity": "sha512-fC6QCp7Sc5sX6g8Tvbmj4XUTbyrik0akgRy03yjXbQaBWWNWGE7SGtJk98m0N8nzegD/7SggrUlivxo5ax4KWQ==", + "dev": true, + "requires": { + "@types/istanbul-lib-coverage": "^2.0.0", + "@types/istanbul-reports": "^3.0.0", + "@types/node": "*", + "@types/yargs": "^15.0.0", + "chalk": "^4.0.0" + } + }, + "@types/istanbul-reports": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/@types/istanbul-reports/-/istanbul-reports-3.0.0.tgz", + "integrity": "sha512-nwKNbvnwJ2/mndE9ItP/zc2TCzw6uuodnF4EHYWD+gCQDVBuRQL5UzbZD0/ezy1iKsFU2ZQiDqg4M9dN4+wZgA==", + "dev": true, + "requires": { + "@types/istanbul-lib-report": "*" + } + }, + "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.0", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.0.tgz", + "integrity": "sha512-qwx12AxXe2Q5xQ43Ac//I6v5aXTipYrSESdOgzrN+9XjgEpyjpKuvSGaN4qE93f7TQTlerQQ8S+EQ0EyDoVL1A==", + "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 + }, + "glob": { + "version": "7.1.6", + "resolved": "https://registry.npmjs.org/glob/-/glob-7.1.6.tgz", + "integrity": "sha512-LwaxwyZ72Lk7vZINtNNrywX0ZuLyStrdDtabefZKAY5ZGJhVtgdznluResxNmPitE0SAO+O26sWTHeKSI2wMBA==", + "dev": true, + "requires": { + "fs.realpath": "^1.0.0", + "inflight": "^1.0.4", + "inherits": "2", + "minimatch": "^3.0.4", + "once": "^1.3.0", + "path-is-absolute": "^1.0.0" + } + }, + "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 + }, + "strip-bom": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/strip-bom/-/strip-bom-4.0.0.tgz", + "integrity": "sha512-3xurFv5tEgii33Zi8Jtp55wEIILR9eh34FAW00PZf+JnSsTmV/ioewSgQl97JHvgjoRGwPShsWm+IdrxB35d0w==", + "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" + } + }, + "yargs": { + "version": "15.4.1", + "resolved": "https://registry.npmjs.org/yargs/-/yargs-15.4.1.tgz", + "integrity": "sha512-aePbxDmcYW++PaqBsJ+HYUFwCdv4LVvdnhBy78E57PIor8/OVvhMrADFFEDh8DHDFRv/O9i3lPhsENjO7QX0+A==", + "dev": true, + "requires": { + "cliui": "^6.0.0", + "decamelize": "^1.2.0", + "find-up": "^4.1.0", + "get-caller-file": "^2.0.1", + "require-directory": "^2.1.1", + "require-main-filename": "^2.0.0", + "set-blocking": "^2.0.0", + "string-width": "^4.2.0", + "which-module": "^2.0.0", + "y18n": "^4.0.0", + "yargs-parser": "^18.1.2" + } + } + } + }, + "jest-serializer": { + "version": "26.6.2", + "resolved": "https://registry.npmjs.org/jest-serializer/-/jest-serializer-26.6.2.tgz", + "integrity": "sha512-S5wqyz0DXnNJPd/xfIzZ5Xnp1HrJWBczg8mMfMpN78OJ5eDxXyf+Ygld9wX1DnUWbIbhM1YDY95NjR4CBXkb2g==", + "dev": true, + "requires": { + "@types/node": "*", + "graceful-fs": "^4.2.4" + } + }, + "jest-snapshot": { + "version": "26.6.2", + "resolved": "https://registry.npmjs.org/jest-snapshot/-/jest-snapshot-26.6.2.tgz", + "integrity": "sha512-OLhxz05EzUtsAmOMzuupt1lHYXCNib0ECyuZ/PZOx9TrZcC8vL0x+DUG3TL+GLX3yHG45e6YGjIm0XwDc3q3og==", + "dev": true, + "requires": { + "@babel/types": "^7.0.0", + "@jest/types": "^26.6.2", + "@types/babel__traverse": "^7.0.4", + "@types/prettier": "^2.0.0", + "chalk": "^4.0.0", + "expect": "^26.6.2", + "graceful-fs": "^4.2.4", + "jest-diff": "^26.6.2", + "jest-get-type": "^26.3.0", + "jest-haste-map": "^26.6.2", + "jest-matcher-utils": "^26.6.2", + "jest-message-util": "^26.6.2", + "jest-resolve": "^26.6.2", + "natural-compare": "^1.4.0", + "pretty-format": "^26.6.2", + "semver": "^7.3.2" + }, + "dependencies": { + "@jest/types": { + "version": "26.6.2", + "resolved": "https://registry.npmjs.org/@jest/types/-/types-26.6.2.tgz", + "integrity": "sha512-fC6QCp7Sc5sX6g8Tvbmj4XUTbyrik0akgRy03yjXbQaBWWNWGE7SGtJk98m0N8nzegD/7SggrUlivxo5ax4KWQ==", + "dev": true, + "requires": { + "@types/istanbul-lib-coverage": "^2.0.0", + "@types/istanbul-reports": "^3.0.0", + "@types/node": "*", + "@types/yargs": "^15.0.0", + "chalk": "^4.0.0" + } + }, + "@types/istanbul-reports": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/@types/istanbul-reports/-/istanbul-reports-3.0.0.tgz", + "integrity": "sha512-nwKNbvnwJ2/mndE9ItP/zc2TCzw6uuodnF4EHYWD+gCQDVBuRQL5UzbZD0/ezy1iKsFU2ZQiDqg4M9dN4+wZgA==", + "dev": true, + "requires": { + "@types/istanbul-lib-report": "*" + } + }, + "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.0", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.0.tgz", + "integrity": "sha512-qwx12AxXe2Q5xQ43Ac//I6v5aXTipYrSESdOgzrN+9XjgEpyjpKuvSGaN4qE93f7TQTlerQQ8S+EQ0EyDoVL1A==", + "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 + }, + "diff-sequences": { + "version": "26.6.2", + "resolved": "https://registry.npmjs.org/diff-sequences/-/diff-sequences-26.6.2.tgz", + "integrity": "sha512-Mv/TDa3nZ9sbc5soK+OoA74BsS3mL37yixCvUAQkiuA4Wz6YtwP/K47n2rv2ovzHZvoiQeA5FTQOschKkEwB0Q==", + "dev": true + }, + "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-diff": { + "version": "26.6.2", + "resolved": "https://registry.npmjs.org/jest-diff/-/jest-diff-26.6.2.tgz", + "integrity": "sha512-6m+9Z3Gv9wN0WFVasqjCL/06+EFCMTqDEUl/b87HYK2rAPTyfz4ZIuSlPhY51PIQRWx5TaxeF1qmXKe9gfN3sA==", + "dev": true, + "requires": { + "chalk": "^4.0.0", + "diff-sequences": "^26.6.2", + "jest-get-type": "^26.3.0", + "pretty-format": "^26.6.2" + } + }, + "jest-get-type": { + "version": "26.3.0", + "resolved": "https://registry.npmjs.org/jest-get-type/-/jest-get-type-26.3.0.tgz", + "integrity": "sha512-TpfaviN1R2pQWkIihlfEanwOXK0zcxrKEE4MlU6Tn7keoXdN6/3gK/xl0yEh8DOunn5pOVGKf8hB4R9gVh04ig==", + "dev": true + }, + "pretty-format": { + "version": "26.6.2", + "resolved": "https://registry.npmjs.org/pretty-format/-/pretty-format-26.6.2.tgz", + "integrity": "sha512-7AeGuCYNGmycyQbCqd/3PWH4eOoX/OiCa0uphp57NVTeAGdJGaAliecxwBDHYQCIvrW7aDBZCYeNTP/WX69mkg==", + "dev": true, + "requires": { + "@jest/types": "^26.6.2", + "ansi-regex": "^5.0.0", + "ansi-styles": "^4.0.0", + "react-is": "^17.0.1" + } + }, + "react-is": { + "version": "17.0.1", + "resolved": "https://registry.npmjs.org/react-is/-/react-is-17.0.1.tgz", + "integrity": "sha512-NAnt2iGDXohE5LI7uBnLnqvLQMtzhkiAOLXTmv+qnF9Ky7xAPcX8Up/xWIhxvLVGJvuLiNc4xQLtuqDRzb4fSA==", + "dev": true + }, + "semver": { + "version": "7.3.2", + "resolved": "https://registry.npmjs.org/semver/-/semver-7.3.2.tgz", + "integrity": "sha512-OrOb32TeeambH6UrhtShmF7CRDqhL6/5XpPNp2DuRH6+9QLw/orhp72j87v8Qa1ScDkvrrBNpZcDejAirJmfXQ==", + "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" + } + } + } + }, + "jest-sonar-reporter": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/jest-sonar-reporter/-/jest-sonar-reporter-2.0.0.tgz", + "integrity": "sha512-ZervDCgEX5gdUbdtWsjdipLN3bKJwpxbvhkYNXTAYvAckCihobSLr9OT/IuyNIRT1EZMDDwR6DroWtrq+IL64w==", + "dev": true, + "requires": { + "xml": "^1.0.1" + } + }, + "jest-util": { + "version": "26.6.2", + "resolved": "https://registry.npmjs.org/jest-util/-/jest-util-26.6.2.tgz", + "integrity": "sha512-MDW0fKfsn0OI7MS7Euz6h8HNDXVQ0gaM9uW6RjfDmd1DAFcaxX9OqIakHIqhbnmF08Cf2DLDG+ulq8YQQ0Lp0Q==", + "dev": true, + "requires": { + "@jest/types": "^26.6.2", + "@types/node": "*", + "chalk": "^4.0.0", + "graceful-fs": "^4.2.4", + "is-ci": "^2.0.0", + "micromatch": "^4.0.2" + }, + "dependencies": { + "@jest/types": { + "version": "26.6.2", + "resolved": "https://registry.npmjs.org/@jest/types/-/types-26.6.2.tgz", + "integrity": "sha512-fC6QCp7Sc5sX6g8Tvbmj4XUTbyrik0akgRy03yjXbQaBWWNWGE7SGtJk98m0N8nzegD/7SggrUlivxo5ax4KWQ==", + "dev": true, + "requires": { + "@types/istanbul-lib-coverage": "^2.0.0", + "@types/istanbul-reports": "^3.0.0", + "@types/node": "*", + "@types/yargs": "^15.0.0", + "chalk": "^4.0.0" + } + }, + "@types/istanbul-reports": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/@types/istanbul-reports/-/istanbul-reports-3.0.0.tgz", + "integrity": "sha512-nwKNbvnwJ2/mndE9ItP/zc2TCzw6uuodnF4EHYWD+gCQDVBuRQL5UzbZD0/ezy1iKsFU2ZQiDqg4M9dN4+wZgA==", + "dev": true, + "requires": { + "@types/istanbul-lib-report": "*" + } + }, + "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.0", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.0.tgz", + "integrity": "sha512-qwx12AxXe2Q5xQ43Ac//I6v5aXTipYrSESdOgzrN+9XjgEpyjpKuvSGaN4qE93f7TQTlerQQ8S+EQ0EyDoVL1A==", + "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 + }, + "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 + }, + "micromatch": { + "version": "4.0.2", + "resolved": "https://registry.npmjs.org/micromatch/-/micromatch-4.0.2.tgz", + "integrity": "sha512-y7FpHSbMUMoyPbYUSzO6PaZ6FyRnQOpHuKwbo1G+Knck95XVU4QAiKdGEnj5wwoS7PlOgthX/09u5iFJ+aYf5Q==", + "dev": true, + "requires": { + "braces": "^3.0.1", + "picomatch": "^2.0.5" + } + }, + "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" + } + } + } + }, + "jest-validate": { + "version": "26.6.2", + "resolved": "https://registry.npmjs.org/jest-validate/-/jest-validate-26.6.2.tgz", + "integrity": "sha512-NEYZ9Aeyj0i5rQqbq+tpIOom0YS1u2MVu6+euBsvpgIme+FOfRmoC4R5p0JiAUpaFvFy24xgrpMknarR/93XjQ==", + "dev": true, + "requires": { + "@jest/types": "^26.6.2", + "camelcase": "^6.0.0", + "chalk": "^4.0.0", + "jest-get-type": "^26.3.0", + "leven": "^3.1.0", + "pretty-format": "^26.6.2" + }, + "dependencies": { + "@jest/types": { + "version": "26.6.2", + "resolved": "https://registry.npmjs.org/@jest/types/-/types-26.6.2.tgz", + "integrity": "sha512-fC6QCp7Sc5sX6g8Tvbmj4XUTbyrik0akgRy03yjXbQaBWWNWGE7SGtJk98m0N8nzegD/7SggrUlivxo5ax4KWQ==", + "dev": true, + "requires": { + "@types/istanbul-lib-coverage": "^2.0.0", + "@types/istanbul-reports": "^3.0.0", + "@types/node": "*", + "@types/yargs": "^15.0.0", + "chalk": "^4.0.0" + } + }, + "@types/istanbul-reports": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/@types/istanbul-reports/-/istanbul-reports-3.0.0.tgz", + "integrity": "sha512-nwKNbvnwJ2/mndE9ItP/zc2TCzw6uuodnF4EHYWD+gCQDVBuRQL5UzbZD0/ezy1iKsFU2ZQiDqg4M9dN4+wZgA==", + "dev": true, + "requires": { + "@types/istanbul-lib-report": "*" + } + }, + "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" + } + }, + "camelcase": { + "version": "6.2.0", + "resolved": "https://registry.npmjs.org/camelcase/-/camelcase-6.2.0.tgz", + "integrity": "sha512-c7wVvbw3f37nuobQNtgsgG9POC9qMbNuMQmTCqZv23b6MIz0fcYpBiOlv9gEN/hdLdnZTDQhg6e9Dq5M1vKvfg==", + "dev": true + }, + "chalk": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.0.tgz", + "integrity": "sha512-qwx12AxXe2Q5xQ43Ac//I6v5aXTipYrSESdOgzrN+9XjgEpyjpKuvSGaN4qE93f7TQTlerQQ8S+EQ0EyDoVL1A==", + "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 + }, + "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-get-type": { + "version": "26.3.0", + "resolved": "https://registry.npmjs.org/jest-get-type/-/jest-get-type-26.3.0.tgz", + "integrity": "sha512-TpfaviN1R2pQWkIihlfEanwOXK0zcxrKEE4MlU6Tn7keoXdN6/3gK/xl0yEh8DOunn5pOVGKf8hB4R9gVh04ig==", + "dev": true + }, + "pretty-format": { + "version": "26.6.2", + "resolved": "https://registry.npmjs.org/pretty-format/-/pretty-format-26.6.2.tgz", + "integrity": "sha512-7AeGuCYNGmycyQbCqd/3PWH4eOoX/OiCa0uphp57NVTeAGdJGaAliecxwBDHYQCIvrW7aDBZCYeNTP/WX69mkg==", + "dev": true, + "requires": { + "@jest/types": "^26.6.2", + "ansi-regex": "^5.0.0", + "ansi-styles": "^4.0.0", + "react-is": "^17.0.1" + } + }, + "react-is": { + "version": "17.0.1", + "resolved": "https://registry.npmjs.org/react-is/-/react-is-17.0.1.tgz", + "integrity": "sha512-NAnt2iGDXohE5LI7uBnLnqvLQMtzhkiAOLXTmv+qnF9Ky7xAPcX8Up/xWIhxvLVGJvuLiNc4xQLtuqDRzb4fSA==", + "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" + } + } + } + }, + "jest-watcher": { + "version": "26.6.2", + "resolved": "https://registry.npmjs.org/jest-watcher/-/jest-watcher-26.6.2.tgz", + "integrity": "sha512-WKJob0P/Em2csiVthsI68p6aGKTIcsfjH9Gsx1f0A3Italz43e3ho0geSAVsmj09RWOELP1AZ/DXyJgOgDKxXQ==", + "dev": true, + "requires": { + "@jest/test-result": "^26.6.2", + "@jest/types": "^26.6.2", + "@types/node": "*", + "ansi-escapes": "^4.2.1", + "chalk": "^4.0.0", + "jest-util": "^26.6.2", + "string-length": "^4.0.1" + }, + "dependencies": { + "@jest/types": { + "version": "26.6.2", + "resolved": "https://registry.npmjs.org/@jest/types/-/types-26.6.2.tgz", + "integrity": "sha512-fC6QCp7Sc5sX6g8Tvbmj4XUTbyrik0akgRy03yjXbQaBWWNWGE7SGtJk98m0N8nzegD/7SggrUlivxo5ax4KWQ==", + "dev": true, + "requires": { + "@types/istanbul-lib-coverage": "^2.0.0", + "@types/istanbul-reports": "^3.0.0", + "@types/node": "*", + "@types/yargs": "^15.0.0", + "chalk": "^4.0.0" + } + }, + "@types/istanbul-reports": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/@types/istanbul-reports/-/istanbul-reports-3.0.0.tgz", + "integrity": "sha512-nwKNbvnwJ2/mndE9ItP/zc2TCzw6uuodnF4EHYWD+gCQDVBuRQL5UzbZD0/ezy1iKsFU2ZQiDqg4M9dN4+wZgA==", + "dev": true, + "requires": { + "@types/istanbul-lib-report": "*" + } + }, + "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.0", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.0.tgz", + "integrity": "sha512-qwx12AxXe2Q5xQ43Ac//I6v5aXTipYrSESdOgzrN+9XjgEpyjpKuvSGaN4qE93f7TQTlerQQ8S+EQ0EyDoVL1A==", + "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 + }, + "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" + } + } + } + }, + "jest-worker": { + "version": "26.6.2", + "resolved": "https://registry.npmjs.org/jest-worker/-/jest-worker-26.6.2.tgz", + "integrity": "sha512-KWYVV1c4i+jbMpaBC+U++4Va0cp8OisU185o73T1vo99hqi7w8tSJfUXYswwqqrjzwxa6KpRK54WhPvwf5w6PQ==", + "dev": true, + "requires": { + "@types/node": "*", + "merge-stream": "^2.0.0", + "supports-color": "^7.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": "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" + } + } + } + }, + "jhipster-core": { + "version": "7.3.4", + "resolved": "https://registry.npmjs.org/jhipster-core/-/jhipster-core-7.3.4.tgz", + "integrity": "sha512-AUhT69kNkqppaJZVfan/xnKG4Gs9Ggj7YLtTZFVe+xg+THrbMb5Ng7PL07PDlDw4KAEA33GMCwuAf65E8EpC4g==", + "dev": true, + "requires": { + "chevrotain": "7.0.1", + "fs-extra": "8.1.0", + "lodash": "4.17.15", + "winston": "3.2.1" + }, + "dependencies": { + "fs-extra": { + "version": "8.1.0", + "resolved": "https://registry.npmjs.org/fs-extra/-/fs-extra-8.1.0.tgz", + "integrity": "sha512-yhlQgA6mnOJUKOsRUFsgJdQCvkKhcz8tlZG5HBQfReYZy46OwLcY+Zia0mtdHsOo9y/hP+CxMN0TU9QxoOtG4g==", + "dev": true, + "requires": { + "graceful-fs": "^4.2.0", + "jsonfile": "^4.0.0", + "universalify": "^0.1.0" + } + }, + "lodash": { + "version": "4.17.15", + "resolved": "https://registry.npmjs.org/lodash/-/lodash-4.17.15.tgz", + "integrity": "sha512-8xOcRHvCjnocdS5cpwXQXVzmmh5e5+saE2QGoeQmbKmRS6J3VQppPOIt0MnmE+4xlZoumy0GPG0D0MVIQbNA1A==", + "dev": true + } + } + }, + "jmespath": { + "version": "0.15.0", + "resolved": "https://registry.npmjs.org/jmespath/-/jmespath-0.15.0.tgz", + "integrity": "sha1-o/Iiqarp+Wb10nx5ZRDigJF2Qhc=", + "dev": true + }, + "js-object-pretty-print": { + "version": "0.3.0", + "resolved": "https://registry.npmjs.org/js-object-pretty-print/-/js-object-pretty-print-0.3.0.tgz", + "integrity": "sha1-RnDkUAZu4ezPNRdMfRl/WqOLz3Q=", + "dev": true + }, + "js-tokens": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/js-tokens/-/js-tokens-4.0.0.tgz", + "integrity": "sha512-RdJUflcE3cUzKiMqQgsCu06FPu9UdIJO0beYbPhHN4k6apgJtifcoCtT9bcxOpYBtpD2kCM6Sbzg4CausW/PKQ==" + }, + "js-yaml": { + "version": "3.14.0", + "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-3.14.0.tgz", + "integrity": "sha512-/4IbIeHcD9VMHFqDR/gQ7EdZdLimOvW2DdcxFjdyyZ9NsbS+ccrXqVWDtab/lRl5AlUqmpBx8EhPaWR+OtY17A==", + "dev": true, + "requires": { + "argparse": "^1.0.7", + "esprima": "^4.0.0" + } + }, + "jsbn": { + "version": "0.1.1", + "resolved": "https://registry.npmjs.org/jsbn/-/jsbn-0.1.1.tgz", + "integrity": "sha1-peZUwuWi3rXyAdls77yoDA7y9RM=", + "dev": true + }, + "jsdom": { + "version": "16.4.0", + "resolved": "https://registry.npmjs.org/jsdom/-/jsdom-16.4.0.tgz", + "integrity": "sha512-lYMm3wYdgPhrl7pDcRmvzPhhrGVBeVhPIqeHjzeiHN3DFmD1RBpbExbi8vU7BJdH8VAZYovR8DMt0PNNDM7k8w==", + "dev": true, + "requires": { + "abab": "^2.0.3", + "acorn": "^7.1.1", + "acorn-globals": "^6.0.0", + "cssom": "^0.4.4", + "cssstyle": "^2.2.0", + "data-urls": "^2.0.0", + "decimal.js": "^10.2.0", + "domexception": "^2.0.1", + "escodegen": "^1.14.1", + "html-encoding-sniffer": "^2.0.1", + "is-potential-custom-element-name": "^1.0.0", + "nwsapi": "^2.2.0", + "parse5": "5.1.1", + "request": "^2.88.2", + "request-promise-native": "^1.0.8", + "saxes": "^5.0.0", + "symbol-tree": "^3.2.4", + "tough-cookie": "^3.0.1", + "w3c-hr-time": "^1.0.2", + "w3c-xmlserializer": "^2.0.0", + "webidl-conversions": "^6.1.0", + "whatwg-encoding": "^1.0.5", + "whatwg-mimetype": "^2.3.0", + "whatwg-url": "^8.0.0", + "ws": "^7.2.3", + "xml-name-validator": "^3.0.0" + }, + "dependencies": { + "tough-cookie": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/tough-cookie/-/tough-cookie-3.0.1.tgz", + "integrity": "sha512-yQyJ0u4pZsv9D4clxO69OEjLWYw+jbgspjTue4lTQZLfV0c5l1VmK2y1JK8E9ahdpltPOaAThPcp5nKPUgSnsg==", + "dev": true, + "requires": { + "ip-regex": "^2.1.0", + "psl": "^1.1.28", + "punycode": "^2.1.1" + } + }, + "ws": { + "version": "7.4.0", + "resolved": "https://registry.npmjs.org/ws/-/ws-7.4.0.tgz", + "integrity": "sha512-kyFwXuV/5ymf+IXhS6f0+eAFvydbaBW3zjpT6hUdAh/hbVjTIB5EHBGi0bPoCLSK2wcuz3BrEkB9LrYv1Nm4NQ==", + "dev": true + } + } + }, + "jsesc": { + "version": "2.5.2", + "resolved": "https://registry.npmjs.org/jsesc/-/jsesc-2.5.2.tgz", + "integrity": "sha512-OYu7XEzjkCQ3C5Ps3QIZsQfNpqoJyZZA99wd9aWd05NCtC5pWOkShK2mkL6HXQR6/Cy2lbNdPlZBpuQHXE63gA==" + }, + "json-parse-better-errors": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/json-parse-better-errors/-/json-parse-better-errors-1.0.2.tgz", + "integrity": "sha512-mrqyZKfX5EhL7hvqcV6WG1yYjnjeuYDzDhhcAAUrq8Po85NBQBJP+ZDUT75qZQ98IkUoBqdkExkukOU7Ts2wrw==", + "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": { + "version": "0.2.3", + "resolved": "https://registry.npmjs.org/json-schema/-/json-schema-0.2.3.tgz", + "integrity": "sha1-tIDIkuWaLwWVTOcnvT8qTogvnhM=", + "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-schema-typed": { + "version": "7.0.3", + "resolved": "https://registry.npmjs.org/json-schema-typed/-/json-schema-typed-7.0.3.tgz", + "integrity": "sha512-7DE8mpG+/fVw+dTpjbxnx47TaMnDfOI1jwft9g1VybltZCduyRQPJPvc+zzKY9WPHxhPWczyFuYa6I8Mw4iU5A==", + "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": "sha1-nbe1lJatPzz+8wp1FC0tkwrXJlE=", + "dev": true + }, + "json-stringify-safe": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/json-stringify-safe/-/json-stringify-safe-5.0.1.tgz", + "integrity": "sha1-Epai1Y/UXxmg9s4B1lcB4sc1tus=", + "dev": true + }, + "json3": { + "version": "3.3.3", + "resolved": "https://registry.npmjs.org/json3/-/json3-3.3.3.tgz", + "integrity": "sha512-c7/8mbUsKigAbLkD5B010BK4D9LZm7A1pNItkEwiUZRpIN66exu/e7YQWysGun+TRKaJp8MhemM+VkfWv42aCA==", + "dev": true + }, + "json5": { + "version": "2.1.3", + "resolved": "https://registry.npmjs.org/json5/-/json5-2.1.3.tgz", + "integrity": "sha512-KXPvOm8K9IJKFM0bmdn8QXh7udDh1g/giieX0NLCaMnb4hEiVFqnop2ImTXCc5e0/oHz3LTqmHGtExn5hfMkOA==", + "requires": { + "minimist": "^1.2.5" + } + }, + "jsonfile": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/jsonfile/-/jsonfile-4.0.0.tgz", + "integrity": "sha1-h3Gq4HmbZAdrdmQPygWPnBDjPss=", + "dev": true, + "requires": { + "graceful-fs": "^4.1.6" + } + }, + "jsonparse": { + "version": "1.3.1", + "resolved": "https://registry.npmjs.org/jsonparse/-/jsonparse-1.3.1.tgz", + "integrity": "sha1-P02uSpH6wxX3EGL4UhzCOfE2YoA=", + "dev": true + }, + "jsprim": { + "version": "1.4.1", + "resolved": "https://registry.npmjs.org/jsprim/-/jsprim-1.4.1.tgz", + "integrity": "sha1-MT5mvB5cwG5Di8G3SZwuXFastqI=", + "dev": true, + "requires": { + "assert-plus": "1.0.0", + "extsprintf": "1.3.0", + "json-schema": "0.2.3", + "verror": "1.10.0" + } + }, + "killable": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/killable/-/killable-1.0.1.tgz", + "integrity": "sha512-LzqtLKlUwirEUyl/nicirVmNiPvYs7l5n8wOPP7fyJVpUPkvCnW/vuiXGpylGUlnPDnB7311rARzAt3Mhswpjg==", + "dev": true + }, + "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 + }, + "kleur": { + "version": "3.0.3", + "resolved": "https://registry.npmjs.org/kleur/-/kleur-3.0.3.tgz", + "integrity": "sha512-eTIzlVOSUR+JxdDFepEYcBMtZ9Qqdef+rnzWdRZuMbOywu5tO2w2N7rqjoANZ5k9vywhL6Br1VRjUIgTQx4E8w==", + "dev": true + }, + "kuler": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/kuler/-/kuler-1.0.1.tgz", + "integrity": "sha512-J9nVUucG1p/skKul6DU3PUZrhs0LPulNaeUOox0IyXDi8S4CztTHs1gQphhuZmzXG7VOQSf6NJfKuzteQLv9gQ==", + "dev": true, + "requires": { + "colornames": "^1.1.1" + } + }, + "last-call-webpack-plugin": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/last-call-webpack-plugin/-/last-call-webpack-plugin-3.0.0.tgz", + "integrity": "sha512-7KI2l2GIZa9p2spzPIVZBYyNKkN+e/SQPpnjlTiPhdbDW3F86tdKKELxKpzJ5sgU19wQWsACULZmpTPYHeWO5w==", + "dev": true, + "requires": { + "lodash": "^4.17.5", + "webpack-sources": "^1.1.0" + } + }, + "lazy-cache": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/lazy-cache/-/lazy-cache-2.0.2.tgz", + "integrity": "sha1-uRkKT5EzVGlIQIWfio9whNiCImQ=", + "dev": true, + "requires": { + "set-getter": "^0.1.0" + } + }, + "lcid": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/lcid/-/lcid-1.0.0.tgz", + "integrity": "sha1-MIrMr6C8SDo4Z7S28rlQYlHRuDU=", + "dev": true, + "requires": { + "invert-kv": "^1.0.0" + } + }, + "leven": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/leven/-/leven-3.1.0.tgz", + "integrity": "sha512-qsda+H8jTaUaN/x5vzW2rzc+8Rw4TAQ/4KjB46IwK5VH+IlVeeeje/EoZRpiXvIqjFgK84QffqPztGI3VBLG1A==", + "dev": true + }, + "levn": { + "version": "0.3.0", + "resolved": "https://registry.npmjs.org/levn/-/levn-0.3.0.tgz", + "integrity": "sha1-OwmSTt+fCDwEkP3UwLxEIeBHZO4=", + "dev": true, + "requires": { + "prelude-ls": "~1.1.2", + "type-check": "~0.3.2" + } + }, + "limiter": { + "version": "1.1.5", + "resolved": "https://registry.npmjs.org/limiter/-/limiter-1.1.5.tgz", + "integrity": "sha512-FWWMIEOxz3GwUI4Ts/IvgVy6LPvoMPgjMdQ185nN6psJyBJ4yOpzqm695/h5umdLJg2vW3GR5iG11MAkR2AzJA==", + "dev": true + }, + "lines-and-columns": { + "version": "1.1.6", + "resolved": "https://registry.npmjs.org/lines-and-columns/-/lines-and-columns-1.1.6.tgz", + "integrity": "sha1-HADHQ7QzzQpOgHWPe2SldEDZ/wA=", + "dev": true + }, + "lint-staged": { + "version": "8.2.1", + "resolved": "https://registry.npmjs.org/lint-staged/-/lint-staged-8.2.1.tgz", + "integrity": "sha512-n0tDGR/rTCgQNwXnUf/eWIpPNddGWxC32ANTNYsj2k02iZb7Cz5ox2tytwBu+2r0zDXMEMKw7Y9OD/qsav561A==", + "dev": true, + "requires": { + "chalk": "^2.3.1", + "commander": "^2.14.1", + "cosmiconfig": "^5.2.0", + "debug": "^3.1.0", + "dedent": "^0.7.0", + "del": "^3.0.0", + "execa": "^1.0.0", + "g-status": "^2.0.2", + "is-glob": "^4.0.0", + "is-windows": "^1.0.2", + "listr": "^0.14.2", + "listr-update-renderer": "^0.5.0", + "lodash": "^4.17.11", + "log-symbols": "^2.2.0", + "micromatch": "^3.1.8", + "npm-which": "^3.0.1", + "p-map": "^1.1.1", + "path-is-inside": "^1.0.2", + "pify": "^3.0.0", + "please-upgrade-node": "^3.0.2", + "staged-git-files": "1.1.2", + "string-argv": "^0.0.2", + "stringify-object": "^3.2.2", + "yup": "^0.27.0" + }, + "dependencies": { + "cosmiconfig": { + "version": "5.2.1", + "resolved": "https://registry.npmjs.org/cosmiconfig/-/cosmiconfig-5.2.1.tgz", + "integrity": "sha512-H65gsXo1SKjf8zmrJ67eJk8aIRKV5ff2D4uKZIBZShbhGSpEmsQOPW/SKMKYhSTrqR7ufy6RP69rPogdaPh/kA==", + "dev": true, + "requires": { + "import-fresh": "^2.0.0", + "is-directory": "^0.3.1", + "js-yaml": "^3.13.1", + "parse-json": "^4.0.0" + } + }, + "debug": { + "version": "3.2.7", + "resolved": "https://registry.npmjs.org/debug/-/debug-3.2.7.tgz", + "integrity": "sha512-CFjzYYAi4ThfiQvizrFQevTTXHtnCqWfe7x1AhgEscTz6ZbLbfoLRLPugTQyBth6f8ZERVUSyWHFD/7Wu4t1XQ==", + "dev": true, + "requires": { + "ms": "^2.1.1" + } + }, + "import-fresh": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/import-fresh/-/import-fresh-2.0.0.tgz", + "integrity": "sha1-2BNVwVYS04bGH53dOSLUMEgipUY=", + "dev": true, + "requires": { + "caller-path": "^2.0.0", + "resolve-from": "^3.0.0" + } + }, + "log-symbols": { + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/log-symbols/-/log-symbols-2.2.0.tgz", + "integrity": "sha512-VeIAFslyIerEJLXHziedo2basKbMKtTw3vfn5IzG0XTjhAVEJyNHnL2p7vc+wBDSdQuUpNw3M2u6xb9QsAY5Eg==", + "dev": true, + "requires": { + "chalk": "^2.0.1" + } + }, + "p-map": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/p-map/-/p-map-1.2.0.tgz", + "integrity": "sha512-r6zKACMNhjPJMTl8KcFH4li//gkrXWfbD6feV8l6doRHlzljFWGJ2AP6iKaCJXyZmAUMOPtvbW7EXkbWO/pLEA==", + "dev": true + }, + "parse-json": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/parse-json/-/parse-json-4.0.0.tgz", + "integrity": "sha1-vjX1Qlvh9/bHRxhPmKeIy5lHfuA=", + "dev": true, + "requires": { + "error-ex": "^1.3.1", + "json-parse-better-errors": "^1.0.1" + } + }, + "pify": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/pify/-/pify-3.0.0.tgz", + "integrity": "sha1-5aSs0sEB/fPZpNB/DbxNtJ3SgXY=", + "dev": true + }, + "resolve-from": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/resolve-from/-/resolve-from-3.0.0.tgz", + "integrity": "sha1-six699nWiBvItuZTM17rywoYh0g=", + "dev": true + } + } + }, + "listr": { + "version": "0.14.3", + "resolved": "https://registry.npmjs.org/listr/-/listr-0.14.3.tgz", + "integrity": "sha512-RmAl7su35BFd/xoMamRjpIE4j3v+L28o8CT5YhAXQJm1fD+1l9ngXY8JAQRJ+tFK2i5njvi0iRUKV09vPwA0iA==", + "dev": true, + "requires": { + "@samverschueren/stream-to-observable": "^0.3.0", + "is-observable": "^1.1.0", + "is-promise": "^2.1.0", + "is-stream": "^1.1.0", + "listr-silent-renderer": "^1.1.1", + "listr-update-renderer": "^0.5.0", + "listr-verbose-renderer": "^0.5.0", + "p-map": "^2.0.0", + "rxjs": "^6.3.3" + }, + "dependencies": { + "p-map": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/p-map/-/p-map-2.1.0.tgz", + "integrity": "sha512-y3b8Kpd8OAN444hxfBbFfj1FY/RjtTd8tzYwhUqNYXx0fXx2iX4maP4Qr6qhIKbQXI02wTLAda4fYUbDagTUFw==", + "dev": true + } + } + }, + "listr-silent-renderer": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/listr-silent-renderer/-/listr-silent-renderer-1.1.1.tgz", + "integrity": "sha1-kktaN1cVN3C/Go4/v3S4u/P5JC4=", + "dev": true + }, + "listr-update-renderer": { + "version": "0.5.0", + "resolved": "https://registry.npmjs.org/listr-update-renderer/-/listr-update-renderer-0.5.0.tgz", + "integrity": "sha512-tKRsZpKz8GSGqoI/+caPmfrypiaq+OQCbd+CovEC24uk1h952lVj5sC7SqyFUm+OaJ5HN/a1YLt5cit2FMNsFA==", + "dev": true, + "requires": { + "chalk": "^1.1.3", + "cli-truncate": "^0.2.1", + "elegant-spinner": "^1.0.1", + "figures": "^1.7.0", + "indent-string": "^3.0.0", + "log-symbols": "^1.0.2", + "log-update": "^2.3.0", + "strip-ansi": "^3.0.1" + }, + "dependencies": { + "ansi-regex": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-2.1.1.tgz", + "integrity": "sha1-w7M6te42DYbg5ijwRorn7yfWVN8=", + "dev": true + }, + "ansi-styles": { + "version": "2.2.1", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-2.2.1.tgz", + "integrity": "sha1-tDLdM1i2NM914eRmQ2gkBTPB3b4=", + "dev": true + }, + "chalk": { + "version": "1.1.3", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-1.1.3.tgz", + "integrity": "sha1-qBFcVeSnAv5NFQq9OHKCKn4J/Jg=", + "dev": true, + "requires": { + "ansi-styles": "^2.2.1", + "escape-string-regexp": "^1.0.2", + "has-ansi": "^2.0.0", + "strip-ansi": "^3.0.0", + "supports-color": "^2.0.0" + } + }, + "figures": { + "version": "1.7.0", + "resolved": "https://registry.npmjs.org/figures/-/figures-1.7.0.tgz", + "integrity": "sha1-y+Hjr/zxzUS4DK3+0o3Hk6lwHS4=", + "dev": true, + "requires": { + "escape-string-regexp": "^1.0.5", + "object-assign": "^4.1.0" + } + }, + "indent-string": { + "version": "3.2.0", + "resolved": "https://registry.npmjs.org/indent-string/-/indent-string-3.2.0.tgz", + "integrity": "sha1-Sl/W0nzDMvN+VBmlBNu4NxBckok=", + "dev": true + }, + "log-symbols": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/log-symbols/-/log-symbols-1.0.2.tgz", + "integrity": "sha1-N2/3tY6jCGoPCfrMdGF+ylAeGhg=", + "dev": true, + "requires": { + "chalk": "^1.0.0" + } + }, + "strip-ansi": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-3.0.1.tgz", + "integrity": "sha1-ajhfuIU9lS1f8F0Oiq+UJ43GPc8=", + "dev": true, + "requires": { + "ansi-regex": "^2.0.0" + } + }, + "supports-color": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-2.0.0.tgz", + "integrity": "sha1-U10EXOa2Nj+kARcIRimZXp3zJMc=", + "dev": true + } + } + }, + "listr-verbose-renderer": { + "version": "0.5.0", + "resolved": "https://registry.npmjs.org/listr-verbose-renderer/-/listr-verbose-renderer-0.5.0.tgz", + "integrity": "sha512-04PDPqSlsqIOaaaGZ+41vq5FejI9auqTInicFRndCBgE3bXG8D6W1I+mWhk+1nqbHmyhla/6BUrd5OSiHwKRXw==", + "dev": true, + "requires": { + "chalk": "^2.4.1", + "cli-cursor": "^2.1.0", + "date-fns": "^1.27.2", + "figures": "^2.0.0" + }, + "dependencies": { + "cli-cursor": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/cli-cursor/-/cli-cursor-2.1.0.tgz", + "integrity": "sha1-s12sN2R5+sw+lHR9QdDQ9SOP/LU=", + "dev": true, + "requires": { + "restore-cursor": "^2.0.0" + } + }, + "figures": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/figures/-/figures-2.0.0.tgz", + "integrity": "sha1-OrGi0qYsi/tDGgyUy3l6L84nyWI=", + "dev": true, + "requires": { + "escape-string-regexp": "^1.0.5" + } + }, + "mimic-fn": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/mimic-fn/-/mimic-fn-1.2.0.tgz", + "integrity": "sha512-jf84uxzwiuiIVKiOLpfYk7N46TSy8ubTonmneY9vrpHNAnp0QBt2BxWV9dO3/j+BoVAb+a5G6YDPW3M5HOdMWQ==", + "dev": true + }, + "onetime": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/onetime/-/onetime-2.0.1.tgz", + "integrity": "sha1-BnQoIw/WdEOyeUsiu6UotoZ5YtQ=", + "dev": true, + "requires": { + "mimic-fn": "^1.0.0" + } + }, + "restore-cursor": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/restore-cursor/-/restore-cursor-2.0.0.tgz", + "integrity": "sha1-n37ih/gv0ybU/RYpI9YhKe7g368=", + "dev": true, + "requires": { + "onetime": "^2.0.0", + "signal-exit": "^3.0.2" + } + } + } + }, + "load-json-file": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/load-json-file/-/load-json-file-1.1.0.tgz", + "integrity": "sha1-lWkFcI1YtLq0wiYbBPWfMcmTdMA=", + "dev": true, + "requires": { + "graceful-fs": "^4.1.2", + "parse-json": "^2.2.0", + "pify": "^2.0.0", + "pinkie-promise": "^2.0.0", + "strip-bom": "^2.0.0" + } + }, + "loader-runner": { + "version": "2.4.0", + "resolved": "https://registry.npmjs.org/loader-runner/-/loader-runner-2.4.0.tgz", + "integrity": "sha512-Jsmr89RcXGIwivFY21FcRrisYZfvLMTWx5kOLc+JTxtpBOG6xML0vzbc6SEQG2FO9/4Fc3wW4LVcB5DmGflaRw==", + "dev": true + }, + "loader-utils": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/loader-utils/-/loader-utils-2.0.0.tgz", + "integrity": "sha512-rP4F0h2RaWSvPEkD7BLDFQnvSf+nK+wr3ESUjNTyAGobqrijmW92zc+SO6d4p4B1wh7+B/Jg1mkQe5NYUEHtHQ==", + "dev": true, + "requires": { + "big.js": "^5.2.2", + "emojis-list": "^3.0.0", + "json5": "^2.1.2" + } + }, + "localtunnel": { + "version": "1.9.2", + "resolved": "https://registry.npmjs.org/localtunnel/-/localtunnel-1.9.2.tgz", + "integrity": "sha512-NEKF7bDJE9U3xzJu3kbayF0WTvng6Pww7tzqNb/XtEARYwqw7CKEX7BvOMg98FtE9es2CRizl61gkV3hS8dqYg==", + "dev": true, + "requires": { + "axios": "0.19.0", + "debug": "4.1.1", + "openurl": "1.1.1", + "yargs": "6.6.0" + }, + "dependencies": { + "ansi-regex": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-2.1.1.tgz", + "integrity": "sha1-w7M6te42DYbg5ijwRorn7yfWVN8=", + "dev": true + }, + "camelcase": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/camelcase/-/camelcase-3.0.0.tgz", + "integrity": "sha1-MvxLn82vhF/N9+c7uXysImHwqwo=", + "dev": true + }, + "cliui": { + "version": "3.2.0", + "resolved": "https://registry.npmjs.org/cliui/-/cliui-3.2.0.tgz", + "integrity": "sha1-EgYBU3qRbSmUD5NNo7SNWFo5IT0=", + "dev": true, + "requires": { + "string-width": "^1.0.1", + "strip-ansi": "^3.0.1", + "wrap-ansi": "^2.0.0" + } + }, + "debug": { + "version": "4.1.1", + "resolved": "https://registry.npmjs.org/debug/-/debug-4.1.1.tgz", + "integrity": "sha512-pYAIzeRo8J6KPEaJ0VWOh5Pzkbw/RetuzehGM7QRRX5he4fPHx2rdKMB256ehJCkX+XRQm16eZLqLNS8RSZXZw==", + "dev": true, + "requires": { + "ms": "^2.1.1" + } + }, + "get-caller-file": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/get-caller-file/-/get-caller-file-1.0.3.tgz", + "integrity": "sha512-3t6rVToeoZfYSGd8YoLFR2DJkiQrIiUrGcjvFX2mDw3bn6k2OtwHN0TNCLbBO+w8qTvimhDkv+LSscbJY1vE6w==", + "dev": true + }, + "is-fullwidth-code-point": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-1.0.0.tgz", + "integrity": "sha1-754xOG8DGn8NZDr4L95QxFfvAMs=", + "dev": true, + "requires": { + "number-is-nan": "^1.0.0" + } + }, + "require-main-filename": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/require-main-filename/-/require-main-filename-1.0.1.tgz", + "integrity": "sha1-l/cXtp1IeE9fUmpsWqj/3aBVpNE=", + "dev": true + }, + "string-width": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/string-width/-/string-width-1.0.2.tgz", + "integrity": "sha1-EYvfW4zcUaKn5w0hHgfisLmxB9M=", + "dev": true, + "requires": { + "code-point-at": "^1.0.0", + "is-fullwidth-code-point": "^1.0.0", + "strip-ansi": "^3.0.0" + } + }, + "strip-ansi": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-3.0.1.tgz", + "integrity": "sha1-ajhfuIU9lS1f8F0Oiq+UJ43GPc8=", + "dev": true, + "requires": { + "ansi-regex": "^2.0.0" + } + }, + "which-module": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/which-module/-/which-module-1.0.0.tgz", + "integrity": "sha1-u6Y8qGGUiZT/MHc2CJ47lgJsKk8=", + "dev": true + }, + "wrap-ansi": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/wrap-ansi/-/wrap-ansi-2.1.0.tgz", + "integrity": "sha1-2Pw9KE3QV5T+hJc8rs3Rz4JP3YU=", + "dev": true, + "requires": { + "string-width": "^1.0.1", + "strip-ansi": "^3.0.1" + } + }, + "y18n": { + "version": "3.2.1", + "resolved": "https://registry.npmjs.org/y18n/-/y18n-3.2.1.tgz", + "integrity": "sha1-bRX7qITAhnnA136I53WegR4H+kE=", + "dev": true + }, + "yargs": { + "version": "6.6.0", + "resolved": "https://registry.npmjs.org/yargs/-/yargs-6.6.0.tgz", + "integrity": "sha1-eC7CHvQDNF+DCoCMo9UTr1YGUgg=", + "dev": true, + "requires": { + "camelcase": "^3.0.0", + "cliui": "^3.2.0", + "decamelize": "^1.1.1", + "get-caller-file": "^1.0.1", + "os-locale": "^1.4.0", + "read-pkg-up": "^1.0.1", + "require-directory": "^2.1.1", + "require-main-filename": "^1.0.1", + "set-blocking": "^2.0.0", + "string-width": "^1.0.2", + "which-module": "^1.0.0", + "y18n": "^3.2.1", + "yargs-parser": "^4.2.0" + } + }, + "yargs-parser": { + "version": "4.2.1", + "resolved": "https://registry.npmjs.org/yargs-parser/-/yargs-parser-4.2.1.tgz", + "integrity": "sha1-KczqwNxPA8bIe0qfIX3RjJ90hxw=", + "dev": true, + "requires": { + "camelcase": "^3.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==", + "requires": { + "p-locate": "^4.1.0" + } + }, + "lodash": { + "version": "4.17.20", + "resolved": "https://registry.npmjs.org/lodash/-/lodash-4.17.20.tgz", + "integrity": "sha512-PlhdFcillOINfeV7Ni6oF1TAEayyZBoZ8bcshTHqOYJYlrqzRK5hagpagky5o4HfCzzd1TRkXPMFq6cKk9rGmA==" + }, + "lodash._reinterpolate": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/lodash._reinterpolate/-/lodash._reinterpolate-3.0.0.tgz", + "integrity": "sha1-DM8tiRZq8Ds2Y8eWU4t1rG4RTZ0=", + "dev": true + }, + "lodash.debounce": { + "version": "4.0.8", + "resolved": "https://registry.npmjs.org/lodash.debounce/-/lodash.debounce-4.0.8.tgz", + "integrity": "sha1-gteb/zCmfEAF/9XiUVMArZyk168=", + "dev": true + }, + "lodash.difference": { + "version": "4.5.0", + "resolved": "https://registry.npmjs.org/lodash.difference/-/lodash.difference-4.5.0.tgz", + "integrity": "sha1-nMtOUF1Ia5FlE0V3KIWi3yf9AXw=", + "dev": true + }, + "lodash.isfinite": { + "version": "3.3.2", + "resolved": "https://registry.npmjs.org/lodash.isfinite/-/lodash.isfinite-3.3.2.tgz", + "integrity": "sha1-+4m2WpqAKBgz8LdHizpRBPiY67M=", + "dev": true + }, + "lodash.memoize": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/lodash.memoize/-/lodash.memoize-4.1.2.tgz", + "integrity": "sha1-vMbEmkKihA7Zl/Mj6tpezRguC/4=", + "dev": true + }, + "lodash.pad": { + "version": "4.5.1", + "resolved": "https://registry.npmjs.org/lodash.pad/-/lodash.pad-4.5.1.tgz", + "integrity": "sha1-QzCUmoM6fI2iLMIPaibE1Z3runA=", + "dev": true + }, + "lodash.padend": { + "version": "4.6.1", + "resolved": "https://registry.npmjs.org/lodash.padend/-/lodash.padend-4.6.1.tgz", + "integrity": "sha1-U8y6BH0G4VjTEfRdpiX05J5vFm4=", + "dev": true + }, + "lodash.padstart": { + "version": "4.6.1", + "resolved": "https://registry.npmjs.org/lodash.padstart/-/lodash.padstart-4.6.1.tgz", + "integrity": "sha1-0uPuv/DZ05rVD1y9G1KnvOa7YRs=", + "dev": true + }, + "lodash.sortby": { + "version": "4.7.0", + "resolved": "https://registry.npmjs.org/lodash.sortby/-/lodash.sortby-4.7.0.tgz", + "integrity": "sha1-7dFMgk4sycHgsKG0K7UhBRakJDg=", + "dev": true + }, + "lodash.template": { + "version": "4.5.0", + "resolved": "https://registry.npmjs.org/lodash.template/-/lodash.template-4.5.0.tgz", + "integrity": "sha512-84vYFxIkmidUiFxidA/KjjH9pAycqW+h980j7Fuz5qxRtO9pgB7MDFTdys1N7A5mcucRiDyEq4fusljItR1T/A==", + "dev": true, + "requires": { + "lodash._reinterpolate": "^3.0.0", + "lodash.templatesettings": "^4.0.0" + } + }, + "lodash.templatesettings": { + "version": "4.2.0", + "resolved": "https://registry.npmjs.org/lodash.templatesettings/-/lodash.templatesettings-4.2.0.tgz", + "integrity": "sha512-stgLz+i3Aa9mZgnjr/O+v9ruKZsPsndy7qPZOchbqk2cnTU1ZaldKK+v7m54WoKIyxiuMZTKT2H81F8BeAc3ZQ==", + "dev": true, + "requires": { + "lodash._reinterpolate": "^3.0.0" + } + }, + "lodash.uniq": { + "version": "4.5.0", + "resolved": "https://registry.npmjs.org/lodash.uniq/-/lodash.uniq-4.5.0.tgz", + "integrity": "sha1-0CJTc662Uq3BvILklFM5qEJ1R3M=", + "dev": true + }, + "log-symbols": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/log-symbols/-/log-symbols-3.0.0.tgz", + "integrity": "sha512-dSkNGuI7iG3mfvDzUuYZyvk5dD9ocYCYzNU6CYDE6+Xqd+gwme6Z00NS3dUh8mq/73HaEtT7m6W+yUPtU6BZnQ==", + "dev": true, + "requires": { + "chalk": "^2.4.2" + } + }, + "log-update": { + "version": "2.3.0", + "resolved": "https://registry.npmjs.org/log-update/-/log-update-2.3.0.tgz", + "integrity": "sha1-iDKP19HOeTiykoN0bwsbwSayRwg=", + "dev": true, + "requires": { + "ansi-escapes": "^3.0.0", + "cli-cursor": "^2.0.0", + "wrap-ansi": "^3.0.1" + }, + "dependencies": { + "ansi-escapes": { + "version": "3.2.0", + "resolved": "https://registry.npmjs.org/ansi-escapes/-/ansi-escapes-3.2.0.tgz", + "integrity": "sha512-cBhpre4ma+U0T1oM5fXg7Dy1Jw7zzwv7lt/GoCpr+hDQJoYnKVPLL4dCvSEFMmQurOQvSrwT7SL/DAlhBI97RQ==", + "dev": true + }, + "ansi-regex": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-3.0.0.tgz", + "integrity": "sha1-7QMXwyIGT3lGbAKWa922Bas32Zg=", + "dev": true + }, + "cli-cursor": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/cli-cursor/-/cli-cursor-2.1.0.tgz", + "integrity": "sha1-s12sN2R5+sw+lHR9QdDQ9SOP/LU=", + "dev": true, + "requires": { + "restore-cursor": "^2.0.0" + } + }, + "is-fullwidth-code-point": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-2.0.0.tgz", + "integrity": "sha1-o7MKXE8ZkYMWeqq5O+764937ZU8=", + "dev": true + }, + "mimic-fn": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/mimic-fn/-/mimic-fn-1.2.0.tgz", + "integrity": "sha512-jf84uxzwiuiIVKiOLpfYk7N46TSy8ubTonmneY9vrpHNAnp0QBt2BxWV9dO3/j+BoVAb+a5G6YDPW3M5HOdMWQ==", + "dev": true + }, + "onetime": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/onetime/-/onetime-2.0.1.tgz", + "integrity": "sha1-BnQoIw/WdEOyeUsiu6UotoZ5YtQ=", + "dev": true, + "requires": { + "mimic-fn": "^1.0.0" + } + }, + "restore-cursor": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/restore-cursor/-/restore-cursor-2.0.0.tgz", + "integrity": "sha1-n37ih/gv0ybU/RYpI9YhKe7g368=", + "dev": true, + "requires": { + "onetime": "^2.0.0", + "signal-exit": "^3.0.2" + } + }, + "string-width": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/string-width/-/string-width-2.1.1.tgz", + "integrity": "sha512-nOqH59deCq9SRHlxq1Aw85Jnt4w6KvLKqWVik6oA9ZklXLNIOlqg4F2yrT1MVaTjAqvVwdfeZ7w7aCvJD7ugkw==", + "dev": true, + "requires": { + "is-fullwidth-code-point": "^2.0.0", + "strip-ansi": "^4.0.0" + } + }, + "strip-ansi": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-4.0.0.tgz", + "integrity": "sha1-qEeQIusaw2iocTibY1JixQXuNo8=", + "dev": true, + "requires": { + "ansi-regex": "^3.0.0" + } + }, + "wrap-ansi": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/wrap-ansi/-/wrap-ansi-3.0.1.tgz", + "integrity": "sha1-KIoE2H7aXChuBg3+jxNc6NAH+Lo=", + "dev": true, + "requires": { + "string-width": "^2.1.1", + "strip-ansi": "^4.0.0" + } + } + } + }, + "logform": { + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/logform/-/logform-2.2.0.tgz", + "integrity": "sha512-N0qPlqfypFx7UHNn4B3lzS/b0uLqt2hmuoa+PpuXNYgozdJYAyauF5Ky0BWVjrxDlMWiT3qN4zPq3vVAfZy7Yg==", + "dev": true, + "requires": { + "colors": "^1.2.1", + "fast-safe-stringify": "^2.0.4", + "fecha": "^4.2.0", + "ms": "^2.1.1", + "triple-beam": "^1.3.0" + } + }, + "loglevel": { + "version": "1.7.0", + "resolved": "https://registry.npmjs.org/loglevel/-/loglevel-1.7.0.tgz", + "integrity": "sha512-i2sY04nal5jDcagM3FMfG++T69GEEM8CYuOfeOIvmXzOIcwE9a/CJPR0MFM97pYMj/u10lzz7/zd7+qwhrBTqQ==", + "dev": true + }, + "lower-case": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/lower-case/-/lower-case-2.0.1.tgz", + "integrity": "sha512-LiWgfDLLb1dwbFQZsSglpRj+1ctGnayXz3Uv0/WO8n558JycT5fg6zkNcnW0G68Nn0aEldTFeEfmjCfmqry/rQ==", + "dev": true, + "requires": { + "tslib": "^1.10.0" + }, + "dependencies": { + "tslib": { + "version": "1.14.1", + "resolved": "https://registry.npmjs.org/tslib/-/tslib-1.14.1.tgz", + "integrity": "sha512-Xni35NKzjgMrwevysHTCArtLDpPvye8zV/0E4EyYn43P7/7qvQwPh9BGkHewbMulVntbigmcT7rdX3BNo9wRJg==", + "dev": true + } + } + }, + "lowercase-keys": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/lowercase-keys/-/lowercase-keys-1.0.1.tgz", + "integrity": "sha512-G2Lj61tXDnVFFOi8VZds+SoQjtQC3dgokKdDG2mTm1tx4m50NUHBOZSBwQQHyy0V12A0JTG4icfZQH+xPyh8VA==", + "dev": true + }, + "lru-cache": { + "version": "6.0.0", + "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-6.0.0.tgz", + "integrity": "sha512-Jo6dJ04CmSjuznwJSS3pUeWmd/H0ffTlkXXgwZi+eq1UCmqQwCh+eLsYOYCwY991i2Fah4h1BEMCx4qThGbsiA==", + "dev": true, + "requires": { + "yallist": "^4.0.0" + } + }, + "macos-release": { + "version": "2.4.1", + "resolved": "https://registry.npmjs.org/macos-release/-/macos-release-2.4.1.tgz", + "integrity": "sha512-H/QHeBIN1fIGJX517pvK8IEK53yQOW7YcEI55oYtgjDdoCQQz7eJS94qt5kNrscReEyuD/JcdFCm2XBEcGOITg==", + "dev": true + }, + "magic-string": { + "version": "0.25.7", + "resolved": "https://registry.npmjs.org/magic-string/-/magic-string-0.25.7.tgz", + "integrity": "sha512-4CrMT5DOHTDk4HYDlzmwu4FVCcIYI8gauveasrdCu2IKIFOJ3f0v/8MDGJCDL9oD2ppz/Av1b0Nj345H9M+XIA==", + "dev": true, + "requires": { + "sourcemap-codec": "^1.4.4" + } + }, + "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" + }, + "dependencies": { + "semver": { + "version": "6.3.0", + "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.0.tgz", + "integrity": "sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw==", + "dev": true + } + } + }, + "make-error": { + "version": "1.3.6", + "resolved": "https://registry.npmjs.org/make-error/-/make-error-1.3.6.tgz", + "integrity": "sha512-s8UhlNe7vPKomQhC1qFelMokr/Sc3AgNbso3n74mVPA5LTZwkB9NlXf4XPamLxJE8h0gh73rM94xvwRT2CVInw==", + "dev": true + }, + "make-fetch-happen": { + "version": "5.0.2", + "resolved": "https://registry.npmjs.org/make-fetch-happen/-/make-fetch-happen-5.0.2.tgz", + "integrity": "sha512-07JHC0r1ykIoruKO8ifMXu+xEU8qOXDFETylktdug6vJDACnP+HKevOu3PXyNPzFyTSlz8vrBYlBO1JZRe8Cag==", + "dev": true, + "requires": { + "agentkeepalive": "^3.4.1", + "cacache": "^12.0.0", + "http-cache-semantics": "^3.8.1", + "http-proxy-agent": "^2.1.0", + "https-proxy-agent": "^2.2.3", + "lru-cache": "^5.1.1", + "mississippi": "^3.0.0", + "node-fetch-npm": "^2.0.2", + "promise-retry": "^1.1.1", + "socks-proxy-agent": "^4.0.0", + "ssri": "^6.0.0" + }, + "dependencies": { + "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, + "requires": { + "yallist": "^3.0.2" + } + }, + "yallist": { + "version": "3.1.1", + "resolved": "https://registry.npmjs.org/yallist/-/yallist-3.1.1.tgz", + "integrity": "sha512-a4UGQaWPH59mOXUYnAG2ewncQS4i4F43Tv3JoAM+s2VDAmS9NsK8GpDMLrCHPksFT7h3K6TOoUNn2pb7RoXx4g==", + "dev": true + } + } + }, + "makeerror": { + "version": "1.0.11", + "resolved": "https://registry.npmjs.org/makeerror/-/makeerror-1.0.11.tgz", + "integrity": "sha1-4BpckQnyr3lmDk6LlYd5AYT1qWw=", + "dev": true, + "requires": { + "tmpl": "1.0.x" + } + }, + "map-age-cleaner": { + "version": "0.1.3", + "resolved": "https://registry.npmjs.org/map-age-cleaner/-/map-age-cleaner-0.1.3.tgz", + "integrity": "sha512-bJzx6nMoP6PDLPBFmg7+xRKeFZvFboMrGlxmNj9ClvX53KrmvM5bXFXEWjbz4cz1AFn+jWJ9z/DJSz7hrs0w3w==", + "dev": true, + "requires": { + "p-defer": "^1.0.0" + } + }, + "map-cache": { + "version": "0.2.2", + "resolved": "https://registry.npmjs.org/map-cache/-/map-cache-0.2.2.tgz", + "integrity": "sha1-wyq9C9ZSXZsFFkW7TyasXcmKDb8=", + "dev": true + }, + "map-obj": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/map-obj/-/map-obj-4.1.0.tgz", + "integrity": "sha512-glc9y00wgtwcDmp7GaE/0b0OnxpNJsVf3ael/An6Fe2Q51LLwN1er6sdomLRzz5h0+yMpiYLhWYF5R7HeqVd4g==", + "dev": true + }, + "map-visit": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/map-visit/-/map-visit-1.0.0.tgz", + "integrity": "sha1-7Nyo8TFE5mDxtb1B8S80edmN+48=", + "dev": true, + "requires": { + "object-visit": "^1.0.0" + } + }, + "matcher": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/matcher/-/matcher-1.1.1.tgz", + "integrity": "sha512-+BmqxWIubKTRKNWx/ahnCkk3mG8m7OturVlqq6HiojGJTd5hVYbgZm6WzcYPCoB+KBT4Vd6R7WSRG2OADNaCjg==", + "dev": true, + "requires": { + "escape-string-regexp": "^1.0.4" + } + }, + "md5.js": { + "version": "1.3.5", + "resolved": "https://registry.npmjs.org/md5.js/-/md5.js-1.3.5.tgz", + "integrity": "sha512-xitP+WxNPcTTOgnTJcrhM0xvdPepipPSf3I8EIpGKeFLjt3PlJLIDG3u8EX53ZIubkb+5U2+3rELYpEhHhzdkg==", + "dev": true, + "requires": { + "hash-base": "^3.0.0", + "inherits": "^2.0.1", + "safe-buffer": "^5.1.2" + } + }, + "mdn-data": { + "version": "2.0.4", + "resolved": "https://registry.npmjs.org/mdn-data/-/mdn-data-2.0.4.tgz", + "integrity": "sha512-iV3XNKw06j5Q7mi6h+9vbx23Tv7JkjEVgKHW4pimwyDGWm0OIQntJJ+u1C6mg6mK1EaTv42XQ7w76yuzH7M2cA==", + "dev": true + }, + "media-typer": { + "version": "0.3.0", + "resolved": "https://registry.npmjs.org/media-typer/-/media-typer-0.3.0.tgz", + "integrity": "sha1-hxDXrwqmJvj/+hzgAWhUUmMlV0g=", + "dev": true + }, + "mem": { + "version": "5.1.1", + "resolved": "https://registry.npmjs.org/mem/-/mem-5.1.1.tgz", + "integrity": "sha512-qvwipnozMohxLXG1pOqoLiZKNkC4r4qqRucSoDwXowsNGDSULiqFTRUF05vcZWnwJSG22qTsynQhxbaMtnX9gw==", + "dev": true, + "requires": { + "map-age-cleaner": "^0.1.3", + "mimic-fn": "^2.1.0", + "p-is-promise": "^2.1.0" + } + }, + "mem-fs": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/mem-fs/-/mem-fs-1.2.0.tgz", + "integrity": "sha512-b8g0jWKdl8pM0LqAPdK9i8ERL7nYrzmJfRhxMiWH2uYdfYnb7uXnmwVb0ZGe7xyEl4lj+nLIU3yf4zPUT+XsVQ==", + "dev": true, + "requires": { + "through2": "^3.0.0", + "vinyl": "^2.0.1", + "vinyl-file": "^3.0.0" + }, + "dependencies": { + "through2": { + "version": "3.0.2", + "resolved": "https://registry.npmjs.org/through2/-/through2-3.0.2.tgz", + "integrity": "sha512-enaDQ4MUyP2W6ZyT6EsMzqBPZaM/avg8iuo+l2d3QCs0J+6RaqkHV/2/lOwDTueBHeJ/2LG9lrLW3d5rWPucuQ==", + "dev": true, + "requires": { + "inherits": "^2.0.4", + "readable-stream": "2 || 3" + } + } + } + }, + "mem-fs-editor": { + "version": "6.0.0", + "resolved": "https://registry.npmjs.org/mem-fs-editor/-/mem-fs-editor-6.0.0.tgz", + "integrity": "sha512-e0WfJAMm8Gv1mP5fEq/Blzy6Lt1VbLg7gNnZmZak7nhrBTibs+c6nQ4SKs/ZyJYHS1mFgDJeopsLAv7Ow0FMFg==", + "dev": true, + "requires": { + "commondir": "^1.0.1", + "deep-extend": "^0.6.0", + "ejs": "^2.6.1", + "glob": "^7.1.4", + "globby": "^9.2.0", + "isbinaryfile": "^4.0.0", + "mkdirp": "^0.5.0", + "multimatch": "^4.0.0", + "rimraf": "^2.6.3", + "through2": "^3.0.1", + "vinyl": "^2.2.0" + }, + "dependencies": { + "@nodelib/fs.stat": { + "version": "1.1.3", + "resolved": "https://registry.npmjs.org/@nodelib/fs.stat/-/fs.stat-1.1.3.tgz", + "integrity": "sha512-shAmDyaQC4H92APFoIaVDHCx5bStIocgvbwQyxPRrbUY20V1EYTbSDchWbuwlMG3V17cprZhA6+78JfB+3DTPw==", + "dev": true + }, + "array-union": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/array-union/-/array-union-1.0.2.tgz", + "integrity": "sha1-mjRBDk9OPaI96jdb5b5w8kd47Dk=", + "dev": true, + "requires": { + "array-uniq": "^1.0.1" + } + }, + "dir-glob": { + "version": "2.2.2", + "resolved": "https://registry.npmjs.org/dir-glob/-/dir-glob-2.2.2.tgz", + "integrity": "sha512-f9LBi5QWzIW3I6e//uxZoLBlUt9kcp66qo0sSCxL6YZKc75R1c4MFCoe/LaZiBGmgujvQdxc5Bn3QhfyvK5Hsw==", + "dev": true, + "requires": { + "path-type": "^3.0.0" + } + }, + "ejs": { + "version": "2.7.4", + "resolved": "https://registry.npmjs.org/ejs/-/ejs-2.7.4.tgz", + "integrity": "sha512-7vmuyh5+kuUyJKePhQfRQBhXV5Ce+RnaeeQArKu1EAMpL3WbgMt5WG6uQZpEVvYSSsxMXRKOewtDk9RaTKXRlA==", + "dev": true + }, + "fast-glob": { + "version": "2.2.7", + "resolved": "https://registry.npmjs.org/fast-glob/-/fast-glob-2.2.7.tgz", + "integrity": "sha512-g1KuQwHOZAmOZMuBtHdxDtju+T2RT8jgCC9aANsbpdiDDTSnjgfuVsIBNKbUeJI3oKMRExcfNDtJl4OhbffMsw==", + "dev": true, + "requires": { + "@mrmlnc/readdir-enhanced": "^2.2.1", + "@nodelib/fs.stat": "^1.1.2", + "glob-parent": "^3.1.0", + "is-glob": "^4.0.0", + "merge2": "^1.2.3", + "micromatch": "^3.1.10" + } + }, + "glob": { + "version": "7.1.6", + "resolved": "https://registry.npmjs.org/glob/-/glob-7.1.6.tgz", + "integrity": "sha512-LwaxwyZ72Lk7vZINtNNrywX0ZuLyStrdDtabefZKAY5ZGJhVtgdznluResxNmPitE0SAO+O26sWTHeKSI2wMBA==", + "dev": true, + "requires": { + "fs.realpath": "^1.0.0", + "inflight": "^1.0.4", + "inherits": "2", + "minimatch": "^3.0.4", + "once": "^1.3.0", + "path-is-absolute": "^1.0.0" + } + }, + "glob-parent": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-3.1.0.tgz", + "integrity": "sha1-nmr2KZ2NO9K9QEMIMr0RPfkGxa4=", + "dev": true, + "requires": { + "is-glob": "^3.1.0", + "path-dirname": "^1.0.0" + }, + "dependencies": { + "is-glob": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-3.1.0.tgz", + "integrity": "sha1-e6WuJCF4BKxwcHuWkiVnSGzD6Eo=", + "dev": true, + "requires": { + "is-extglob": "^2.1.0" + } + } + } + }, + "globby": { + "version": "9.2.0", + "resolved": "https://registry.npmjs.org/globby/-/globby-9.2.0.tgz", + "integrity": "sha512-ollPHROa5mcxDEkwg6bPt3QbEf4pDQSNtd6JPL1YvOvAo/7/0VAm9TccUeoTmarjPw4pfUthSCqcyfNB1I3ZSg==", + "dev": true, + "requires": { + "@types/glob": "^7.1.1", + "array-union": "^1.0.2", + "dir-glob": "^2.2.2", + "fast-glob": "^2.2.6", + "glob": "^7.1.3", + "ignore": "^4.0.3", + "pify": "^4.0.1", + "slash": "^2.0.0" + } + }, + "ignore": { + "version": "4.0.6", + "resolved": "https://registry.npmjs.org/ignore/-/ignore-4.0.6.tgz", + "integrity": "sha512-cyFDKrqc/YdcWFniJhzI42+AzS+gNwmUzOSFcRCQYwySuBBBy/KjuxWLZ/FHEH6Moq1NizMOBWyTcv8O4OZIMg==", + "dev": true + }, + "path-type": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/path-type/-/path-type-3.0.0.tgz", + "integrity": "sha512-T2ZUsdZFHgA3u4e5PfPbjd7HDDpxPnQb5jN0SrDsjNSuVXHJqtwTnWqG0B1jZrgmJ/7lj1EmVIByWt1gxGkWvg==", + "dev": true, + "requires": { + "pify": "^3.0.0" + }, + "dependencies": { + "pify": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/pify/-/pify-3.0.0.tgz", + "integrity": "sha1-5aSs0sEB/fPZpNB/DbxNtJ3SgXY=", + "dev": true + } + } + }, + "pify": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/pify/-/pify-4.0.1.tgz", + "integrity": "sha512-uB80kBFb/tfd68bVleG9T5GGsGPjJrLAUpR5PZIrhBnIaRTQRjqdJSsIKkOP6OAIFbj7GOrcudc5pNjZ+geV2g==", + "dev": true + }, + "rimraf": { + "version": "2.7.1", + "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-2.7.1.tgz", + "integrity": "sha512-uWjbaKIK3T1OSVptzX7Nl6PvQ3qAGtKEtVRjRuazjfL3Bx5eI409VZSqgND+4UNnmzLVdPj9FqFJNPqBZFve4w==", + "dev": true, + "requires": { + "glob": "^7.1.3" + } + }, + "slash": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/slash/-/slash-2.0.0.tgz", + "integrity": "sha512-ZYKh3Wh2z1PpEXWr0MpSBZ0V6mZHAQfYevttO11c51CaWjGTaadiKZ+wVt1PbMlDV5qhMFslpZCemhwOK7C89A==", + "dev": true + }, + "through2": { + "version": "3.0.2", + "resolved": "https://registry.npmjs.org/through2/-/through2-3.0.2.tgz", + "integrity": "sha512-enaDQ4MUyP2W6ZyT6EsMzqBPZaM/avg8iuo+l2d3QCs0J+6RaqkHV/2/lOwDTueBHeJ/2LG9lrLW3d5rWPucuQ==", + "dev": true, + "requires": { + "inherits": "^2.0.4", + "readable-stream": "2 || 3" + } + } + } + }, + "memory-fs": { + "version": "0.5.0", + "resolved": "https://registry.npmjs.org/memory-fs/-/memory-fs-0.5.0.tgz", + "integrity": "sha512-jA0rdU5KoQMC0e6ppoNRtpp6vjFq6+NY7r8hywnC7V+1Xj/MtHwGIbB1QaK/dunyjWteJzmkpd7ooeWg10T7GA==", + "dev": true, + "requires": { + "errno": "^0.1.3", + "readable-stream": "^2.0.1" + } + }, + "meow": { + "version": "6.1.0", + "resolved": "https://registry.npmjs.org/meow/-/meow-6.1.0.tgz", + "integrity": "sha512-iIAoeI01v6pmSfObAAWFoITAA4GgiT45m4SmJgoxtZfvI0fyZwhV4d0lTwiUXvAKIPlma05Feb2Xngl52Mj5Cg==", + "dev": true, + "requires": { + "@types/minimist": "^1.2.0", + "camelcase-keys": "^6.1.1", + "decamelize-keys": "^1.1.0", + "hard-rejection": "^2.0.0", + "minimist-options": "^4.0.1", + "normalize-package-data": "^2.5.0", + "read-pkg-up": "^7.0.0", + "redent": "^3.0.0", + "trim-newlines": "^3.0.0", + "type-fest": "^0.8.1", + "yargs-parser": "^18.1.1" + }, + "dependencies": { + "parse-json": { + "version": "5.1.0", + "resolved": "https://registry.npmjs.org/parse-json/-/parse-json-5.1.0.tgz", + "integrity": "sha512-+mi/lmVVNKFNVyLXV31ERiy2CY5E1/F6QtJFEzoChPRwwngMNXRDQ9GJ5WdE2Z2P4AujsOi0/+2qHID68KwfIQ==", + "dev": true, + "requires": { + "@babel/code-frame": "^7.0.0", + "error-ex": "^1.3.1", + "json-parse-even-better-errors": "^2.3.0", + "lines-and-columns": "^1.1.6" + } + }, + "read-pkg": { + "version": "5.2.0", + "resolved": "https://registry.npmjs.org/read-pkg/-/read-pkg-5.2.0.tgz", + "integrity": "sha512-Ug69mNOpfvKDAc2Q8DRpMjjzdtrnv9HcSMX+4VsZxD1aZ6ZzrIE7rlzXBtWTyhULSMKg076AW6WR5iZpD0JiOg==", + "dev": true, + "requires": { + "@types/normalize-package-data": "^2.4.0", + "normalize-package-data": "^2.5.0", + "parse-json": "^5.0.0", + "type-fest": "^0.6.0" + }, + "dependencies": { + "type-fest": { + "version": "0.6.0", + "resolved": "https://registry.npmjs.org/type-fest/-/type-fest-0.6.0.tgz", + "integrity": "sha512-q+MB8nYR1KDLrgr4G5yemftpMC7/QLqVndBmEEdqzmNj5dcFOO4Oo8qlwZE3ULT3+Zim1F8Kq4cBnikNhlCMlg==", + "dev": true + } + } + }, + "read-pkg-up": { + "version": "7.0.1", + "resolved": "https://registry.npmjs.org/read-pkg-up/-/read-pkg-up-7.0.1.tgz", + "integrity": "sha512-zK0TB7Xd6JpCLmlLmufqykGE+/TlOePD6qKClNW7hHDKFh/J7/7gCWGR7joEQEW1bKq3a3yUZSObOoWLFQ4ohg==", + "dev": true, + "requires": { + "find-up": "^4.1.0", + "read-pkg": "^5.2.0", + "type-fest": "^0.8.1" + } + }, + "type-fest": { + "version": "0.8.1", + "resolved": "https://registry.npmjs.org/type-fest/-/type-fest-0.8.1.tgz", + "integrity": "sha512-4dbzIzqvjtgiM5rw1k5rEHtBANKmdudhGyBEajN01fEyhaAIhsoKNy6y7+IN93IfpFtwY9iqi7kD+xwKhQsNJA==", + "dev": true + } + } + }, + "merge-descriptors": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/merge-descriptors/-/merge-descriptors-1.0.1.tgz", + "integrity": "sha1-sAqqVW3YtEVoFQ7J0blT8/kMu2E=", + "dev": true + }, + "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 + }, + "merge2": { + "version": "1.4.1", + "resolved": "https://registry.npmjs.org/merge2/-/merge2-1.4.1.tgz", + "integrity": "sha512-8q7VEgMJW4J8tcfVPy8g09NcQwZdbwFEqhe/WZkoIzjn/3TGDwtOCYtXGxA3O8tPzpczCCDgv+P2P5y00ZJOOg==", + "dev": true + }, + "methods": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/methods/-/methods-1.1.2.tgz", + "integrity": "sha1-VSmk1nZUE07cxSZmVoNbD4Ua/O4=", + "dev": true + }, + "micromatch": { + "version": "3.1.10", + "resolved": "https://registry.npmjs.org/micromatch/-/micromatch-3.1.10.tgz", + "integrity": "sha512-MWikgl9n9M3w+bpsY3He8L+w9eF9338xRl8IAO5viDizwSzziFEyUzo2xrrloB64ADbTf8uA8vRqqttDTOmccg==", + "dev": true, + "requires": { + "arr-diff": "^4.0.0", + "array-unique": "^0.3.2", + "braces": "^2.3.1", + "define-property": "^2.0.2", + "extend-shallow": "^3.0.2", + "extglob": "^2.0.4", + "fragment-cache": "^0.2.1", + "kind-of": "^6.0.2", + "nanomatch": "^1.2.9", + "object.pick": "^1.3.0", + "regex-not": "^1.0.0", + "snapdragon": "^0.8.1", + "to-regex": "^3.0.2" + }, + "dependencies": { + "braces": { + "version": "2.3.2", + "resolved": "https://registry.npmjs.org/braces/-/braces-2.3.2.tgz", + "integrity": "sha512-aNdbnj9P8PjdXU4ybaWLK2IF3jc/EoDYbC7AazW6to3TRsfXxscC9UXOB5iDiEQrkyIbWp2SLQda4+QAa7nc3w==", + "dev": true, + "requires": { + "arr-flatten": "^1.1.0", + "array-unique": "^0.3.2", + "extend-shallow": "^2.0.1", + "fill-range": "^4.0.0", + "isobject": "^3.0.1", + "repeat-element": "^1.1.2", + "snapdragon": "^0.8.1", + "snapdragon-node": "^2.0.1", + "split-string": "^3.0.2", + "to-regex": "^3.0.1" + }, + "dependencies": { + "extend-shallow": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/extend-shallow/-/extend-shallow-2.0.1.tgz", + "integrity": "sha1-Ua99YUrZqfYQ6huvu5idaxxWiQ8=", + "dev": true, + "requires": { + "is-extendable": "^0.1.0" + } + } + } + }, + "fill-range": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/fill-range/-/fill-range-4.0.0.tgz", + "integrity": "sha1-1USBHUKPmOsGpj3EAtJAPDKMOPc=", + "dev": true, + "requires": { + "extend-shallow": "^2.0.1", + "is-number": "^3.0.0", + "repeat-string": "^1.6.1", + "to-regex-range": "^2.1.0" + }, + "dependencies": { + "extend-shallow": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/extend-shallow/-/extend-shallow-2.0.1.tgz", + "integrity": "sha1-Ua99YUrZqfYQ6huvu5idaxxWiQ8=", + "dev": true, + "requires": { + "is-extendable": "^0.1.0" + } + } + } + }, + "is-number": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/is-number/-/is-number-3.0.0.tgz", + "integrity": "sha1-JP1iAaR4LPUFYcgQJ2r8fRLXEZU=", + "dev": true, + "requires": { + "kind-of": "^3.0.2" + }, + "dependencies": { + "kind-of": { + "version": "3.2.2", + "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-3.2.2.tgz", + "integrity": "sha1-MeohpzS6ubuw8yRm2JOupR5KPGQ=", + "dev": true, + "requires": { + "is-buffer": "^1.1.5" + } + } + } + }, + "to-regex-range": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/to-regex-range/-/to-regex-range-2.1.1.tgz", + "integrity": "sha1-fIDBe53+vlmeJzZ+DU3VWQFB2zg=", + "dev": true, + "requires": { + "is-number": "^3.0.0", + "repeat-string": "^1.6.1" + } + } + } + }, + "miller-rabin": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/miller-rabin/-/miller-rabin-4.0.1.tgz", + "integrity": "sha512-115fLhvZVqWwHPbClyntxEVfVDfl9DLLTuJvq3g2O/Oxi8AiNouAHvDSzHS0viUJc+V5vm3eq91Xwqn9dp4jRA==", + "dev": true, + "requires": { + "bn.js": "^4.0.0", + "brorand": "^1.0.1" + }, + "dependencies": { + "bn.js": { + "version": "4.11.9", + "resolved": "https://registry.npmjs.org/bn.js/-/bn.js-4.11.9.tgz", + "integrity": "sha512-E6QoYqCKZfgatHTdHzs1RRKP7ip4vvm+EyRUeE2RF0NblwVvb0p6jSVeNTOFxPn26QXN2o6SMfNxKp6kU8zQaw==", + "dev": true + } + } + }, + "mime": { + "version": "1.4.1", + "resolved": "https://registry.npmjs.org/mime/-/mime-1.4.1.tgz", + "integrity": "sha512-KI1+qOZu5DcW6wayYHSzR/tXKCDC5Om4s1z2QJjDULzLcmf3DvzS7oluY4HCTrc+9FiKmWUgeNLg7W3uIQvxtQ==", + "dev": true + }, + "mime-db": { + "version": "1.44.0", + "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.44.0.tgz", + "integrity": "sha512-/NOTfLrsPBVeH7YtFPgsVWveuL+4SjjYxaQ1xtM1KMFj7HdxlBlxeyNLzhyJVx7r4rZGJAZ/6lkKCitSc/Nmpg==", + "dev": true + }, + "mime-types": { + "version": "2.1.27", + "resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.1.27.tgz", + "integrity": "sha512-JIhqnCasI9yD+SsmkquHBxTSEuZdQX5BuQnS2Vc7puQQQ+8yiP5AY5uWhpdv4YL4VM5c6iliiYWPgJ/nJQLp7w==", + "dev": true, + "requires": { + "mime-db": "1.44.0" + } + }, + "mimic-fn": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/mimic-fn/-/mimic-fn-2.1.0.tgz", + "integrity": "sha512-OqbOk5oEQeAZ8WXWydlu9HJjz9WVdEIvamMCcXmuqUYjTknH/sqsWvhQ3vgwKFRR1HpjvNBKQ37nbJgYzGqGcg==", + "dev": true + }, + "min-indent": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/min-indent/-/min-indent-1.0.1.tgz", + "integrity": "sha512-I9jwMn07Sy/IwOj3zVkVik2JTvgpaykDZEigL6Rx6N9LbMywwUSMtxET+7lVoDLLd3O3IXwJwvuuns8UB/HeAg==", + "dev": true + }, + "mini-css-extract-plugin": { + "version": "0.9.0", + "resolved": "https://registry.npmjs.org/mini-css-extract-plugin/-/mini-css-extract-plugin-0.9.0.tgz", + "integrity": "sha512-lp3GeY7ygcgAmVIcRPBVhIkf8Us7FZjA+ILpal44qLdSu11wmjKQ3d9k15lfD7pO4esu9eUIAW7qiYIBppv40A==", + "dev": true, + "requires": { + "loader-utils": "^1.1.0", + "normalize-url": "1.9.1", + "schema-utils": "^1.0.0", + "webpack-sources": "^1.1.0" + }, + "dependencies": { + "json5": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/json5/-/json5-1.0.1.tgz", + "integrity": "sha512-aKS4WQjPenRxiQsC93MNfjx+nbF4PAdYzmd/1JIj8HYzqfbu86beTuNgXDzPknWk0n0uARlyewZo4s++ES36Ow==", + "dev": true, + "requires": { + "minimist": "^1.2.0" + } + }, + "loader-utils": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/loader-utils/-/loader-utils-1.4.0.tgz", + "integrity": "sha512-qH0WSMBtn/oHuwjy/NucEgbx5dbxxnxup9s4PVXJUDHZBQY+s0NWA9rJf53RBnQZxfch7euUui7hpoAPvALZdA==", + "dev": true, + "requires": { + "big.js": "^5.2.2", + "emojis-list": "^3.0.0", + "json5": "^1.0.1" + } + }, + "schema-utils": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/schema-utils/-/schema-utils-1.0.0.tgz", + "integrity": "sha512-i27Mic4KovM/lnGsy8whRCHhc7VicJajAjTrYg11K9zfZXnYIt4k5F+kZkwjnrhKzLic/HLU4j11mjsz2G/75g==", + "dev": true, + "requires": { + "ajv": "^6.1.0", + "ajv-errors": "^1.0.0", + "ajv-keywords": "^3.1.0" + } + } + } + }, + "minimalistic-assert": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/minimalistic-assert/-/minimalistic-assert-1.0.1.tgz", + "integrity": "sha512-UtJcAD4yEaGtjPezWuO9wC4nwUnVH/8/Im3yEHQP4b67cXlD/Qr9hdITCU1xDbSEXg2XKNaP8jsReV7vQd00/A==", + "dev": true + }, + "minimalistic-crypto-utils": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/minimalistic-crypto-utils/-/minimalistic-crypto-utils-1.0.1.tgz", + "integrity": "sha1-9sAMHAsIIkblxNmd+4x8CDsrWCo=", + "dev": true + }, + "minimatch": { + "version": "3.0.4", + "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.0.4.tgz", + "integrity": "sha512-yJHVQEhyqPLUTgt9B83PXu6W3rx4MvvHvSUvToogpwoGDOUQ+yDrR0HRot+yOCdCO7u4hX3pWft6kWBBcqh0UA==", + "requires": { + "brace-expansion": "^1.1.7" + } + }, + "minimist": { + "version": "1.2.5", + "resolved": "https://registry.npmjs.org/minimist/-/minimist-1.2.5.tgz", + "integrity": "sha512-FM9nNUYrRBAELZQT3xeZQ7fmMOBg6nWNmJKTcgsJeaLstP/UODVpGsr5OhXhhXg6f+qtJ8uiZ+PUxkDWcgIXLw==" + }, + "minimist-options": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/minimist-options/-/minimist-options-4.1.0.tgz", + "integrity": "sha512-Q4r8ghd80yhO/0j1O3B2BjweX3fiHg9cdOwjJd2J76Q135c+NDxGCqdYKQ1SKBuFfgWbAUzBfvYjPUEeNgqN1A==", + "dev": true, + "requires": { + "arrify": "^1.0.1", + "is-plain-obj": "^1.1.0", + "kind-of": "^6.0.3" + }, + "dependencies": { + "arrify": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/arrify/-/arrify-1.0.1.tgz", + "integrity": "sha1-iYUI2iIm84DfkEcoRWhJwVAaSw0=", + "dev": true + } + } + }, + "minipass": { + "version": "2.9.0", + "resolved": "https://registry.npmjs.org/minipass/-/minipass-2.9.0.tgz", + "integrity": "sha512-wxfUjg9WebH+CUDX/CdbRlh5SmfZiy/hpkxaRI16Y9W56Pa75sWgd/rvFilSgrauD9NyFymP/+JFV3KwzIsJeg==", + "dev": true, + "requires": { + "safe-buffer": "^5.1.2", + "yallist": "^3.0.0" + }, + "dependencies": { + "yallist": { + "version": "3.1.1", + "resolved": "https://registry.npmjs.org/yallist/-/yallist-3.1.1.tgz", + "integrity": "sha512-a4UGQaWPH59mOXUYnAG2ewncQS4i4F43Tv3JoAM+s2VDAmS9NsK8GpDMLrCHPksFT7h3K6TOoUNn2pb7RoXx4g==", + "dev": true + } + } + }, + "minipass-collect": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/minipass-collect/-/minipass-collect-1.0.2.tgz", + "integrity": "sha512-6T6lH0H8OG9kITm/Jm6tdooIbogG9e0tLgpY6mphXSm/A9u8Nq1ryBG+Qspiub9LjWlBPsPS3tWQ/Botq4FdxA==", + "dev": true, + "requires": { + "minipass": "^3.0.0" + }, + "dependencies": { + "minipass": { + "version": "3.1.3", + "resolved": "https://registry.npmjs.org/minipass/-/minipass-3.1.3.tgz", + "integrity": "sha512-Mgd2GdMVzY+x3IJ+oHnVM+KG3lA5c8tnabyJKmHSaG2kAGpudxuOf8ToDkhumF7UzME7DecbQE9uOZhNm7PuJg==", + "dev": true, + "requires": { + "yallist": "^4.0.0" + } + } + } + }, + "minipass-flush": { + "version": "1.0.5", + "resolved": "https://registry.npmjs.org/minipass-flush/-/minipass-flush-1.0.5.tgz", + "integrity": "sha512-JmQSYYpPUqX5Jyn1mXaRwOda1uQ8HP5KAT/oDSLCzt1BYRhQU0/hDtsB1ufZfEEzMZ9aAVmsBw8+FWsIXlClWw==", + "dev": true, + "requires": { + "minipass": "^3.0.0" + }, + "dependencies": { + "minipass": { + "version": "3.1.3", + "resolved": "https://registry.npmjs.org/minipass/-/minipass-3.1.3.tgz", + "integrity": "sha512-Mgd2GdMVzY+x3IJ+oHnVM+KG3lA5c8tnabyJKmHSaG2kAGpudxuOf8ToDkhumF7UzME7DecbQE9uOZhNm7PuJg==", + "dev": true, + "requires": { + "yallist": "^4.0.0" + } + } + } + }, + "minipass-pipeline": { + "version": "1.2.4", + "resolved": "https://registry.npmjs.org/minipass-pipeline/-/minipass-pipeline-1.2.4.tgz", + "integrity": "sha512-xuIq7cIOt09RPRJ19gdi4b+RiNvDFYe5JH+ggNvBqGqpQXcru3PcRmOZuHBKWK1Txf9+cQ+HMVN4d6z46LZP7A==", + "dev": true, + "requires": { + "minipass": "^3.0.0" + }, + "dependencies": { + "minipass": { + "version": "3.1.3", + "resolved": "https://registry.npmjs.org/minipass/-/minipass-3.1.3.tgz", + "integrity": "sha512-Mgd2GdMVzY+x3IJ+oHnVM+KG3lA5c8tnabyJKmHSaG2kAGpudxuOf8ToDkhumF7UzME7DecbQE9uOZhNm7PuJg==", + "dev": true, + "requires": { + "yallist": "^4.0.0" + } + } + } + }, + "minizlib": { + "version": "1.3.3", + "resolved": "https://registry.npmjs.org/minizlib/-/minizlib-1.3.3.tgz", + "integrity": "sha512-6ZYMOEnmVsdCeTJVE0W9ZD+pVnE8h9Hma/iOwwRDsdQoePpoX56/8B6z3P9VNwppJuBKNRuFDRNRqRWexT9G9Q==", + "dev": true, + "requires": { + "minipass": "^2.9.0" + } + }, + "mississippi": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/mississippi/-/mississippi-3.0.0.tgz", + "integrity": "sha512-x471SsVjUtBRtcvd4BzKE9kFC+/2TeWgKCgw0bZcw1b9l2X3QX5vCWgF+KaZaYm87Ss//rHnWryupDrgLvmSkA==", + "dev": true, + "requires": { + "concat-stream": "^1.5.0", + "duplexify": "^3.4.2", + "end-of-stream": "^1.1.0", + "flush-write-stream": "^1.0.0", + "from2": "^2.1.0", + "parallel-transform": "^1.1.0", + "pump": "^3.0.0", + "pumpify": "^1.3.3", + "stream-each": "^1.1.0", + "through2": "^2.0.0" + } + }, + "mitt": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/mitt/-/mitt-1.2.0.tgz", + "integrity": "sha512-r6lj77KlwqLhIUku9UWYes7KJtsczvolZkzp8hbaDPPaE24OmWl5s539Mytlj22siEQKosZ26qCBgda2PKwoJw==", + "dev": true + }, + "mixin-deep": { + "version": "1.3.2", + "resolved": "https://registry.npmjs.org/mixin-deep/-/mixin-deep-1.3.2.tgz", + "integrity": "sha512-WRoDn//mXBiJ1H40rqa3vH0toePwSsGb45iInWlTySa+Uu4k3tYUSxa2v1KqAiLtvlrSzaExqS1gtk96A9zvEA==", + "dev": true, + "requires": { + "for-in": "^1.0.2", + "is-extendable": "^1.0.1" + }, + "dependencies": { + "is-extendable": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/is-extendable/-/is-extendable-1.0.1.tgz", + "integrity": "sha512-arnXMxT1hhoKo9k1LZdmlNyJdDDfy2v0fXjFlmok4+i8ul/6WlbVge9bhM74OpNPQPMGUToDtz+KXa1PneJxOA==", + "dev": true, + "requires": { + "is-plain-object": "^2.0.4" + } + } + } + }, + "mkdirp": { + "version": "0.5.5", + "resolved": "https://registry.npmjs.org/mkdirp/-/mkdirp-0.5.5.tgz", + "integrity": "sha512-NKmAlESf6jMGym1++R0Ra7wvhV+wFW63FaSOFPwRahvea0gMUcGUhVeAg/0BC0wiv9ih5NYPB1Wn1UEI1/L+xQ==", + "dev": true, + "requires": { + "minimist": "^1.2.5" + } + }, + "moment": { + "version": "2.27.0", + "resolved": "https://registry.npmjs.org/moment/-/moment-2.27.0.tgz", + "integrity": "sha512-al0MUK7cpIcglMv3YF13qSgdAIqxHTO7brRtaz3DlSULbqfazqkc5kEjNrLDOM7fsjshoFIihnU8snrP7zUvhQ==" + }, + "moment-locales-webpack-plugin": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/moment-locales-webpack-plugin/-/moment-locales-webpack-plugin-1.2.0.tgz", + "integrity": "sha512-QAi5v0OlPUP7GXviKMtxnpBAo8WmTHrUNN7iciAhNOEAd9evCOvuN0g1N7ThIg3q11GLCkjY1zQ2saRcf/43nQ==", + "dev": true, + "requires": { + "lodash.difference": "^4.5.0" + } + }, + "move-concurrently": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/move-concurrently/-/move-concurrently-1.0.1.tgz", + "integrity": "sha1-viwAX9oy4LKa8fBdfEszIUxwH5I=", + "dev": true, + "requires": { + "aproba": "^1.1.1", + "copy-concurrently": "^1.0.0", + "fs-write-stream-atomic": "^1.0.8", + "mkdirp": "^0.5.1", + "rimraf": "^2.5.4", + "run-queue": "^1.0.3" + }, + "dependencies": { + "glob": { + "version": "7.1.6", + "resolved": "https://registry.npmjs.org/glob/-/glob-7.1.6.tgz", + "integrity": "sha512-LwaxwyZ72Lk7vZINtNNrywX0ZuLyStrdDtabefZKAY5ZGJhVtgdznluResxNmPitE0SAO+O26sWTHeKSI2wMBA==", + "dev": true, + "requires": { + "fs.realpath": "^1.0.0", + "inflight": "^1.0.4", + "inherits": "2", + "minimatch": "^3.0.4", + "once": "^1.3.0", + "path-is-absolute": "^1.0.0" + } + }, + "rimraf": { + "version": "2.7.1", + "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-2.7.1.tgz", + "integrity": "sha512-uWjbaKIK3T1OSVptzX7Nl6PvQ3qAGtKEtVRjRuazjfL3Bx5eI409VZSqgND+4UNnmzLVdPj9FqFJNPqBZFve4w==", + "dev": true, + "requires": { + "glob": "^7.1.3" + } + } + } + }, + "ms": { + "version": "2.1.2", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz", + "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==" + }, + "multicast-dns": { + "version": "6.2.3", + "resolved": "https://registry.npmjs.org/multicast-dns/-/multicast-dns-6.2.3.tgz", + "integrity": "sha512-ji6J5enbMyGRHIAkAOu3WdV8nggqviKCEKtXcOqfphZZtQrmHKycfynJ2V7eVPUA4NhJ6V7Wf4TmGbTwKE9B6g==", + "dev": true, + "requires": { + "dns-packet": "^1.3.1", + "thunky": "^1.0.2" + } + }, + "multicast-dns-service-types": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/multicast-dns-service-types/-/multicast-dns-service-types-1.1.0.tgz", + "integrity": "sha1-iZ8R2WhuXgXLkbNdXw5jt3PPyQE=", + "dev": true + }, + "multimatch": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/multimatch/-/multimatch-4.0.0.tgz", + "integrity": "sha512-lDmx79y1z6i7RNx0ZGCPq1bzJ6ZoDDKbvh7jxr9SJcWLkShMzXrHbYVpTdnhNM5MXpDUxCQ4DgqVttVXlBgiBQ==", + "dev": true, + "requires": { + "@types/minimatch": "^3.0.3", + "array-differ": "^3.0.0", + "array-union": "^2.1.0", + "arrify": "^2.0.1", + "minimatch": "^3.0.4" + } + }, + "mute-stream": { + "version": "0.0.8", + "resolved": "https://registry.npmjs.org/mute-stream/-/mute-stream-0.0.8.tgz", + "integrity": "sha512-nnbWWOkoWyUsTjKrhgD0dcz22mdkSnpYqbEjIm2nhwhuxlSkpywJmBo8h0ZqJdkp73mb90SssHkN4rsRaBAfAA==", + "dev": true + }, + "nanomatch": { + "version": "1.2.13", + "resolved": "https://registry.npmjs.org/nanomatch/-/nanomatch-1.2.13.tgz", + "integrity": "sha512-fpoe2T0RbHwBTBUOftAfBPaDEi06ufaUai0mE6Yn1kacc3SnTErfb/h+X94VXzI64rKFHYImXSvdwGGCmwOqCA==", + "dev": true, + "requires": { + "arr-diff": "^4.0.0", + "array-unique": "^0.3.2", + "define-property": "^2.0.2", + "extend-shallow": "^3.0.2", + "fragment-cache": "^0.2.1", + "is-windows": "^1.0.2", + "kind-of": "^6.0.2", + "object.pick": "^1.3.0", + "regex-not": "^1.0.0", + "snapdragon": "^0.8.1", + "to-regex": "^3.0.1" + } + }, + "natural-compare": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/natural-compare/-/natural-compare-1.4.0.tgz", + "integrity": "sha1-Sr6/7tdUHywnrPspvbvRXI1bpPc=", + "dev": true + }, + "negotiator": { + "version": "0.6.2", + "resolved": "https://registry.npmjs.org/negotiator/-/negotiator-0.6.2.tgz", + "integrity": "sha512-hZXc7K2e+PgeI1eDBe/10Ard4ekbfrrqG8Ep+8Jmf4JID2bNg7NvCPOZN+kfF574pFQI7mum2AUqDidoKqcTOw==", + "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 + }, + "ng-jhipster": { + "version": "0.15.0", + "resolved": "https://registry.npmjs.org/ng-jhipster/-/ng-jhipster-0.15.0.tgz", + "integrity": "sha512-tiTbf0zmnm9qdB73E++sTtB5HA0HlmtyZ4116UBQ0FEhJGb/xJgO1dyKzFliJ+365pZK0ulWygRjziNzjnl5SQ==", + "requires": { + "tslib": "^2.0.0" + } + }, + "ngx-cookie-service": { + "version": "3.0.4", + "resolved": "https://registry.npmjs.org/ngx-cookie-service/-/ngx-cookie-service-3.0.4.tgz", + "integrity": "sha512-g4KHpCWL2EtKatWqn8lz/DdyMQO8sDS7BNwCSvxF1ETTW5DPJnbdEwCjSdm84MmfqBh6JdgvoQd//rOxH5EbgQ==" + }, + "ngx-infinite-scroll": { + "version": "9.0.0", + "resolved": "https://registry.npmjs.org/ngx-infinite-scroll/-/ngx-infinite-scroll-9.0.0.tgz", + "integrity": "sha512-C8JhvrTip/AKv6f8oiB+GJjW+FGE3EPIc3Kk+v3MICGlW0nkOetyJDe/ejXr8tI8zwbNjbfvAKFRKenibDIW6w==", + "requires": { + "@scarf/scarf": "0.1.5", + "opencollective-postinstall": "^2.0.2" + } + }, + "ngx-webstorage": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/ngx-webstorage/-/ngx-webstorage-5.0.0.tgz", + "integrity": "sha512-m96dBjUgLCpaknLRKfsJMEik393xrSX0EwO3paNSkS5d+xj2/cAendE3NwJeKY/W1D9EkKAhCvSUDX9/bAwCUg==" + }, + "nice-try": { + "version": "1.0.5", + "resolved": "https://registry.npmjs.org/nice-try/-/nice-try-1.0.5.tgz", + "integrity": "sha512-1nh45deeb5olNY7eX82BkPO7SSxR5SSYJiPTrTdFUVYwAl8CKMA5N9PjTYkHiRjisVcxcQ1HXdLhx2qxxJzLNQ==", + "dev": true + }, + "no-case": { + "version": "3.0.3", + "resolved": "https://registry.npmjs.org/no-case/-/no-case-3.0.3.tgz", + "integrity": "sha512-ehY/mVQCf9BL0gKfsJBvFJen+1V//U+0HQMPrWct40ixE4jnv0bfvxDbWtAHL9EcaPEOJHVVYKoQn1TlZUB8Tw==", + "dev": true, + "requires": { + "lower-case": "^2.0.1", + "tslib": "^1.10.0" + }, + "dependencies": { + "tslib": { + "version": "1.14.1", + "resolved": "https://registry.npmjs.org/tslib/-/tslib-1.14.1.tgz", + "integrity": "sha512-Xni35NKzjgMrwevysHTCArtLDpPvye8zV/0E4EyYn43P7/7qvQwPh9BGkHewbMulVntbigmcT7rdX3BNo9wRJg==", + "dev": true + } + } + }, + "node-fetch-npm": { + "version": "2.0.4", + "resolved": "https://registry.npmjs.org/node-fetch-npm/-/node-fetch-npm-2.0.4.tgz", + "integrity": "sha512-iOuIQDWDyjhv9qSDrj9aq/klt6F9z1p2otB3AV7v3zBDcL/x+OfGsvGQZZCcMZbUf4Ujw1xGNQkjvGnVT22cKg==", + "dev": true, + "requires": { + "encoding": "^0.1.11", + "json-parse-better-errors": "^1.0.0", + "safe-buffer": "^5.1.1" + } + }, + "node-forge": { + "version": "0.10.0", + "resolved": "https://registry.npmjs.org/node-forge/-/node-forge-0.10.0.tgz", + "integrity": "sha512-PPmu8eEeG9saEUvI97fm4OYxXVB6bFvyNTyiUOBichBpFG8A1Ljw3bY62+5oOjDEMHRnd0Y7HQ+x7uzxOzC6JA==", + "dev": true + }, + "node-int64": { + "version": "0.4.0", + "resolved": "https://registry.npmjs.org/node-int64/-/node-int64-0.4.0.tgz", + "integrity": "sha1-h6kGXNs1XTGC2PlM4RGIuCXGijs=", + "dev": true + }, + "node-libs-browser": { + "version": "2.2.1", + "resolved": "https://registry.npmjs.org/node-libs-browser/-/node-libs-browser-2.2.1.tgz", + "integrity": "sha512-h/zcD8H9kaDZ9ALUWwlBUDo6TKF8a7qBSCSEGfjTVIYeqsioSKaAX+BN7NgiMGp6iSIXZ3PxgCu8KS3b71YK5Q==", + "dev": true, + "requires": { + "assert": "^1.1.1", + "browserify-zlib": "^0.2.0", + "buffer": "^4.3.0", + "console-browserify": "^1.1.0", + "constants-browserify": "^1.0.0", + "crypto-browserify": "^3.11.0", + "domain-browser": "^1.1.1", + "events": "^3.0.0", + "https-browserify": "^1.0.0", + "os-browserify": "^0.3.0", + "path-browserify": "0.0.1", + "process": "^0.11.10", + "punycode": "^1.2.4", + "querystring-es3": "^0.2.0", + "readable-stream": "^2.3.3", + "stream-browserify": "^2.0.1", + "stream-http": "^2.7.2", + "string_decoder": "^1.0.0", + "timers-browserify": "^2.0.4", + "tty-browserify": "0.0.0", + "url": "^0.11.0", + "util": "^0.11.0", + "vm-browserify": "^1.0.1" + }, + "dependencies": { + "events": { + "version": "3.2.0", + "resolved": "https://registry.npmjs.org/events/-/events-3.2.0.tgz", + "integrity": "sha512-/46HWwbfCX2xTawVfkKLGxMifJYQBWMwY1mjywRtb4c9x8l5NP3KoJtnIOiL1hfdRkIuYhETxQlo62IF8tcnlg==", + "dev": true + }, + "punycode": { + "version": "1.4.1", + "resolved": "https://registry.npmjs.org/punycode/-/punycode-1.4.1.tgz", + "integrity": "sha1-wNWmOycYgArY4esPpSachN1BhF4=", + "dev": true + }, + "url": { + "version": "0.11.0", + "resolved": "https://registry.npmjs.org/url/-/url-0.11.0.tgz", + "integrity": "sha1-ODjpfPxgUh63PFJajlW/3Z4uKPE=", + "dev": true, + "requires": { + "punycode": "1.3.2", + "querystring": "0.2.0" + }, + "dependencies": { + "punycode": { + "version": "1.3.2", + "resolved": "https://registry.npmjs.org/punycode/-/punycode-1.3.2.tgz", + "integrity": "sha1-llOgNvt8HuQjQvIyXM7v6jkmxI0=", + "dev": true + } + } + } + } + }, + "node-modules-regexp": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/node-modules-regexp/-/node-modules-regexp-1.0.0.tgz", + "integrity": "sha1-jZ2+KJZKSsVxLpExZCEHxx6Q7EA=", + "dev": true + }, + "node-notifier": { + "version": "8.0.0", + "resolved": "https://registry.npmjs.org/node-notifier/-/node-notifier-8.0.0.tgz", + "integrity": "sha512-46z7DUmcjoYdaWyXouuFNNfUo6eFa94t23c53c+lG/9Cvauk4a98rAUp9672X5dxGdQmLpPzTxzu8f/OeEPaFA==", + "dev": true, + "optional": true, + "requires": { + "growly": "^1.3.0", + "is-wsl": "^2.2.0", + "semver": "^7.3.2", + "shellwords": "^0.1.1", + "uuid": "^8.3.0", + "which": "^2.0.2" + }, + "dependencies": { + "semver": { + "version": "7.3.2", + "resolved": "https://registry.npmjs.org/semver/-/semver-7.3.2.tgz", + "integrity": "sha512-OrOb32TeeambH6UrhtShmF7CRDqhL6/5XpPNp2DuRH6+9QLw/orhp72j87v8Qa1ScDkvrrBNpZcDejAirJmfXQ==", + "dev": true, + "optional": true + }, + "uuid": { + "version": "8.3.1", + "resolved": "https://registry.npmjs.org/uuid/-/uuid-8.3.1.tgz", + "integrity": "sha512-FOmRr+FmWEIG8uhZv6C2bTgEVXsHk08kE7mPlrBbEe+c3r9pjceVPgupIfNIhc4yx55H69OXANrUaSuu9eInKg==", + "dev": true, + "optional": true + }, + "which": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/which/-/which-2.0.2.tgz", + "integrity": "sha512-BLI3Tl1TW3Pvl70l3yq3Y64i+awpwXqsGBYWkkqMtnbXgrMD+yj7rhW0kuEDxzJaYXGjEW5ogapKNMEKNMjibA==", + "dev": true, + "optional": true, + "requires": { + "isexe": "^2.0.0" + } + } + } + }, + "node-releases": { + "version": "1.1.67", + "resolved": "https://registry.npmjs.org/node-releases/-/node-releases-1.1.67.tgz", + "integrity": "sha512-V5QF9noGFl3EymEwUYzO+3NTDpGfQB4ve6Qfnzf3UNydMhjQRVPR1DZTuvWiLzaFJYw2fmDwAfnRNEVb64hSIg==", + "dev": true + }, + "normalize-package-data": { + "version": "2.5.0", + "resolved": "https://registry.npmjs.org/normalize-package-data/-/normalize-package-data-2.5.0.tgz", + "integrity": "sha512-/5CMN3T0R4XTj4DcGaexo+roZSdSFW/0AOOTROrjxzCG1wrWXEsGbRKevjlIL+ZDE4sZlJr5ED4YW0yqmkK+eA==", + "dev": true, + "requires": { + "hosted-git-info": "^2.1.4", + "resolve": "^1.10.0", + "semver": "2 || 3 || 4 || 5", + "validate-npm-package-license": "^3.0.1" + }, + "dependencies": { + "hosted-git-info": { + "version": "2.8.8", + "resolved": "https://registry.npmjs.org/hosted-git-info/-/hosted-git-info-2.8.8.tgz", + "integrity": "sha512-f/wzC2QaWBs7t9IYqB4T3sR1xviIViXJRJTWBlx2Gf3g0Xi5vI7Yy4koXQ1c9OYDGHN9sBy1DQ2AB8fqZBWhUg==", + "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 + }, + "normalize-range": { + "version": "0.1.2", + "resolved": "https://registry.npmjs.org/normalize-range/-/normalize-range-0.1.2.tgz", + "integrity": "sha1-LRDAa9/TEuqXd2laTShDlFa3WUI=", + "dev": true + }, + "normalize-url": { + "version": "1.9.1", + "resolved": "https://registry.npmjs.org/normalize-url/-/normalize-url-1.9.1.tgz", + "integrity": "sha1-LMDWazHqIwNkWENuNiDYWVTGbDw=", + "dev": true, + "requires": { + "object-assign": "^4.0.1", + "prepend-http": "^1.0.0", + "query-string": "^4.1.0", + "sort-keys": "^1.0.0" + } + }, + "npm-api": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/npm-api/-/npm-api-1.0.0.tgz", + "integrity": "sha512-gtJhIhGq07g9H5sIAB9TZzTySW8MYtcYqg+e+J+5q1GmDsDLLVfyvVBL1VklzjtRsElph11GUtLBS191RDOJxQ==", + "dev": true, + "requires": { + "JSONStream": "^1.3.5", + "clone-deep": "^4.0.1", + "download-stats": "^0.3.4", + "moment": "^2.24.0", + "paged-request": "^2.0.1", + "request": "^2.88.0" + } + }, + "npm-bundled": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/npm-bundled/-/npm-bundled-1.1.1.tgz", + "integrity": "sha512-gqkfgGePhTpAEgUsGEgcq1rqPXA+tv/aVBlgEzfXwA1yiUJF7xtEt3CtVwOjNYQOVknDk0F20w58Fnm3EtG0fA==", + "dev": true, + "requires": { + "npm-normalize-package-bin": "^1.0.1" + } + }, + "npm-install-checks": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/npm-install-checks/-/npm-install-checks-4.0.0.tgz", + "integrity": "sha512-09OmyDkNLYwqKPOnbI8exiOZU2GVVmQp7tgez2BPi5OZC8M82elDAps7sxC4l//uSUtotWqoEIDwjRvWH4qz8w==", + "dev": true, + "requires": { + "semver": "^7.1.1" + }, + "dependencies": { + "semver": { + "version": "7.3.2", + "resolved": "https://registry.npmjs.org/semver/-/semver-7.3.2.tgz", + "integrity": "sha512-OrOb32TeeambH6UrhtShmF7CRDqhL6/5XpPNp2DuRH6+9QLw/orhp72j87v8Qa1ScDkvrrBNpZcDejAirJmfXQ==", + "dev": true + } + } + }, + "npm-normalize-package-bin": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/npm-normalize-package-bin/-/npm-normalize-package-bin-1.0.1.tgz", + "integrity": "sha512-EPfafl6JL5/rU+ot6P3gRSCpPDW5VmIzX959Ob1+ySFUuuYHWHekXpwdUZcKP5C+DS4GEtdJluwBjnsNDl+fSA==", + "dev": true + }, + "npm-package-arg": { + "version": "8.0.1", + "resolved": "https://registry.npmjs.org/npm-package-arg/-/npm-package-arg-8.0.1.tgz", + "integrity": "sha512-/h5Fm6a/exByzFSTm7jAyHbgOqErl9qSNJDQF32Si/ZzgwT2TERVxRxn3Jurw1wflgyVVAxnFR4fRHPM7y1ClQ==", + "dev": true, + "requires": { + "hosted-git-info": "^3.0.2", + "semver": "^7.0.0", + "validate-npm-package-name": "^3.0.0" + }, + "dependencies": { + "semver": { + "version": "7.3.2", + "resolved": "https://registry.npmjs.org/semver/-/semver-7.3.2.tgz", + "integrity": "sha512-OrOb32TeeambH6UrhtShmF7CRDqhL6/5XpPNp2DuRH6+9QLw/orhp72j87v8Qa1ScDkvrrBNpZcDejAirJmfXQ==", + "dev": true + } + } + }, + "npm-packlist": { + "version": "1.4.8", + "resolved": "https://registry.npmjs.org/npm-packlist/-/npm-packlist-1.4.8.tgz", + "integrity": "sha512-5+AZgwru5IevF5ZdnFglB5wNlHG1AOOuw28WhUq8/8emhBmLv6jX5by4WJCh7lW0uSYZYS6DXqIsyZVIXRZU9A==", + "dev": true, + "requires": { + "ignore-walk": "^3.0.1", + "npm-bundled": "^1.0.1", + "npm-normalize-package-bin": "^1.0.1" + } + }, + "npm-path": { + "version": "2.0.4", + "resolved": "https://registry.npmjs.org/npm-path/-/npm-path-2.0.4.tgz", + "integrity": "sha512-IFsj0R9C7ZdR5cP+ET342q77uSRdtWOlWpih5eC+lu29tIDbNEgDbzgVJ5UFvYHWhxDZ5TFkJafFioO0pPQjCw==", + "dev": true, + "requires": { + "which": "^1.2.10" + } + }, + "npm-pick-manifest": { + "version": "6.1.0", + "resolved": "https://registry.npmjs.org/npm-pick-manifest/-/npm-pick-manifest-6.1.0.tgz", + "integrity": "sha512-ygs4k6f54ZxJXrzT0x34NybRlLeZ4+6nECAIbr2i0foTnijtS1TJiyzpqtuUAJOps/hO0tNDr8fRV5g+BtRlTw==", + "dev": true, + "requires": { + "npm-install-checks": "^4.0.0", + "npm-package-arg": "^8.0.0", + "semver": "^7.0.0" + }, + "dependencies": { + "semver": { + "version": "7.3.2", + "resolved": "https://registry.npmjs.org/semver/-/semver-7.3.2.tgz", + "integrity": "sha512-OrOb32TeeambH6UrhtShmF7CRDqhL6/5XpPNp2DuRH6+9QLw/orhp72j87v8Qa1ScDkvrrBNpZcDejAirJmfXQ==", + "dev": true + } + } + }, + "npm-registry-fetch": { + "version": "4.0.7", + "resolved": "https://registry.npmjs.org/npm-registry-fetch/-/npm-registry-fetch-4.0.7.tgz", + "integrity": "sha512-cny9v0+Mq6Tjz+e0erFAB+RYJ/AVGzkjnISiobqP8OWj9c9FLoZZu8/SPSKJWE17F1tk4018wfjV+ZbIbqC7fQ==", + "dev": true, + "requires": { + "JSONStream": "^1.3.4", + "bluebird": "^3.5.1", + "figgy-pudding": "^3.4.1", + "lru-cache": "^5.1.1", + "make-fetch-happen": "^5.0.0", + "npm-package-arg": "^6.1.0", + "safe-buffer": "^5.2.0" + }, + "dependencies": { + "hosted-git-info": { + "version": "2.8.8", + "resolved": "https://registry.npmjs.org/hosted-git-info/-/hosted-git-info-2.8.8.tgz", + "integrity": "sha512-f/wzC2QaWBs7t9IYqB4T3sR1xviIViXJRJTWBlx2Gf3g0Xi5vI7Yy4koXQ1c9OYDGHN9sBy1DQ2AB8fqZBWhUg==", + "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, + "requires": { + "yallist": "^3.0.2" + } + }, + "npm-package-arg": { + "version": "6.1.1", + "resolved": "https://registry.npmjs.org/npm-package-arg/-/npm-package-arg-6.1.1.tgz", + "integrity": "sha512-qBpssaL3IOZWi5vEKUKW0cO7kzLeT+EQO9W8RsLOZf76KF9E/K9+wH0C7t06HXPpaH8WH5xF1MExLuCwbTqRUg==", + "dev": true, + "requires": { + "hosted-git-info": "^2.7.1", + "osenv": "^0.1.5", + "semver": "^5.6.0", + "validate-npm-package-name": "^3.0.0" + } + }, + "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 + }, + "yallist": { + "version": "3.1.1", + "resolved": "https://registry.npmjs.org/yallist/-/yallist-3.1.1.tgz", + "integrity": "sha512-a4UGQaWPH59mOXUYnAG2ewncQS4i4F43Tv3JoAM+s2VDAmS9NsK8GpDMLrCHPksFT7h3K6TOoUNn2pb7RoXx4g==", + "dev": true + } + } + }, + "npm-run-path": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/npm-run-path/-/npm-run-path-2.0.2.tgz", + "integrity": "sha1-NakjLfo11wZ7TLLd8jV7GHFTbF8=", + "dev": true, + "requires": { + "path-key": "^2.0.0" + } + }, + "npm-which": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/npm-which/-/npm-which-3.0.1.tgz", + "integrity": "sha1-kiXybsOihcIJyuZ8OxGmtKtxQKo=", + "dev": true, + "requires": { + "commander": "^2.9.0", + "npm-path": "^2.0.2", + "which": "^1.2.10" + } + }, + "npmlog": { + "version": "2.0.4", + "resolved": "https://registry.npmjs.org/npmlog/-/npmlog-2.0.4.tgz", + "integrity": "sha1-mLUlMPJRTKkNCexbIsiEZyI3VpI=", + "dev": true, + "requires": { + "ansi": "~0.3.1", + "are-we-there-yet": "~1.1.2", + "gauge": "~1.2.5" + } + }, + "nth-check": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/nth-check/-/nth-check-1.0.2.tgz", + "integrity": "sha512-WeBOdju8SnzPN5vTUJYxYUxLeXpCaVP5i5e0LF8fg7WORF2Wd7wFX/pk0tYZk7s8T+J7VLy0Da6J1+wCT0AtHg==", + "dev": true, + "requires": { + "boolbase": "~1.0.0" + } + }, + "num2fraction": { + "version": "1.2.2", + "resolved": "https://registry.npmjs.org/num2fraction/-/num2fraction-1.2.2.tgz", + "integrity": "sha1-b2gragJ6Tp3fpFZM0lidHU5mnt4=", + "dev": true + }, + "number-is-nan": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/number-is-nan/-/number-is-nan-1.0.1.tgz", + "integrity": "sha1-CXtgK1NCKlIsGvuHkDGDNpQaAR0=", + "dev": true + }, + "nwsapi": { + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/nwsapi/-/nwsapi-2.2.0.tgz", + "integrity": "sha512-h2AatdwYH+JHiZpv7pt/gSX1XoRGb7L/qSIeuqA6GwYoF9w1vP1cw42TO0aI2pNyshRK5893hNSl+1//vHK7hQ==", + "dev": true + }, + "oauth-sign": { + "version": "0.9.0", + "resolved": "https://registry.npmjs.org/oauth-sign/-/oauth-sign-0.9.0.tgz", + "integrity": "sha512-fexhUFFPTGV8ybAtSIGbV6gOkSv8UtRbDBnAyLQw4QPKkgNlsH2ByPGtMUqdWkos6YCRmAqViwgZrJc/mRDzZQ==", + "dev": true + }, + "object-assign": { + "version": "4.1.1", + "resolved": "https://registry.npmjs.org/object-assign/-/object-assign-4.1.1.tgz", + "integrity": "sha1-IQmtx5ZYh8/AXLvUQsrIv7s2CGM=", + "dev": true + }, + "object-component": { + "version": "0.0.3", + "resolved": "https://registry.npmjs.org/object-component/-/object-component-0.0.3.tgz", + "integrity": "sha1-8MaapQ78lbhmwYb0AKM3acsvEpE=", + "dev": true + }, + "object-copy": { + "version": "0.1.0", + "resolved": "https://registry.npmjs.org/object-copy/-/object-copy-0.1.0.tgz", + "integrity": "sha1-fn2Fi3gb18mRpBupde04EnVOmYw=", + "dev": true, + "requires": { + "copy-descriptor": "^0.1.0", + "define-property": "^0.2.5", + "kind-of": "^3.0.3" + }, + "dependencies": { + "define-property": { + "version": "0.2.5", + "resolved": "https://registry.npmjs.org/define-property/-/define-property-0.2.5.tgz", + "integrity": "sha1-w1se+RjsPJkPmlvFe+BKrOxcgRY=", + "dev": true, + "requires": { + "is-descriptor": "^0.1.0" + } + }, + "kind-of": { + "version": "3.2.2", + "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-3.2.2.tgz", + "integrity": "sha1-MeohpzS6ubuw8yRm2JOupR5KPGQ=", + "dev": true, + "requires": { + "is-buffer": "^1.1.5" + } + } + } + }, + "object-hash": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/object-hash/-/object-hash-2.0.3.tgz", + "integrity": "sha512-JPKn0GMu+Fa3zt3Bmr66JhokJU5BaNBIh4ZeTlaCBzrBsOeXzwcKKAK1tbLiPKgvwmPXsDvvLHoWh5Bm7ofIYg==", + "dev": true + }, + "object-inspect": { + "version": "1.8.0", + "resolved": "https://registry.npmjs.org/object-inspect/-/object-inspect-1.8.0.tgz", + "integrity": "sha512-jLdtEOB112fORuypAyl/50VRVIBIdVQOSUUGQHzJ4xBSbit81zRarz7GThkEFZy1RceYrWYcPcBFPQwHyAc1gA==", + "dev": true + }, + "object-is": { + "version": "1.1.3", + "resolved": "https://registry.npmjs.org/object-is/-/object-is-1.1.3.tgz", + "integrity": "sha512-teyqLvFWzLkq5B9ki8FVWA902UER2qkxmdA4nLf+wjOLAWgxzCWZNCxpDq9MvE8MmhWNr+I8w3BN49Vx36Y6Xg==", + "dev": true, + "requires": { + "define-properties": "^1.1.3", + "es-abstract": "^1.18.0-next.1" + }, + "dependencies": { + "es-abstract": { + "version": "1.18.0-next.1", + "resolved": "https://registry.npmjs.org/es-abstract/-/es-abstract-1.18.0-next.1.tgz", + "integrity": "sha512-I4UGspA0wpZXWENrdA0uHbnhte683t3qT/1VFH9aX2dA5PPSf6QW5HHXf5HImaqPmjXaVeVk4RGWnaylmV7uAA==", + "dev": true, + "requires": { + "es-to-primitive": "^1.2.1", + "function-bind": "^1.1.1", + "has": "^1.0.3", + "has-symbols": "^1.0.1", + "is-callable": "^1.2.2", + "is-negative-zero": "^2.0.0", + "is-regex": "^1.1.1", + "object-inspect": "^1.8.0", + "object-keys": "^1.1.1", + "object.assign": "^4.1.1", + "string.prototype.trimend": "^1.0.1", + "string.prototype.trimstart": "^1.0.1" + } + } + } + }, + "object-keys": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/object-keys/-/object-keys-1.1.1.tgz", + "integrity": "sha512-NuAESUOUMrlIXOfHKzD6bpPu3tYt3xvjNdRIQ+FeT0lNb4K8WR70CaDxhuNguS2XG+GjkyMwOzsN5ZktImfhLA==", + "dev": true + }, + "object-visit": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/object-visit/-/object-visit-1.0.1.tgz", + "integrity": "sha1-95xEk68MU3e1n+OdOV5BBC3QRbs=", + "dev": true, + "requires": { + "isobject": "^3.0.0" + } + }, + "object.assign": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/object.assign/-/object.assign-4.1.2.tgz", + "integrity": "sha512-ixT2L5THXsApyiUPYKmW+2EHpXXe5Ii3M+f4e+aJFAHao5amFRW6J0OO6c/LU8Be47utCx2GL89hxGB6XSmKuQ==", + "dev": true, + "requires": { + "call-bind": "^1.0.0", + "define-properties": "^1.1.3", + "has-symbols": "^1.0.1", + "object-keys": "^1.1.1" + } + }, + "object.getownpropertydescriptors": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/object.getownpropertydescriptors/-/object.getownpropertydescriptors-2.1.0.tgz", + "integrity": "sha512-Z53Oah9A3TdLoblT7VKJaTDdXdT+lQO+cNpKVnya5JDe9uLvzu1YyY1yFDFrcxrlRgWrEFH0jJtD/IbuwjcEVg==", + "dev": true, + "requires": { + "define-properties": "^1.1.3", + "es-abstract": "^1.17.0-next.1" + } + }, + "object.pick": { + "version": "1.3.0", + "resolved": "https://registry.npmjs.org/object.pick/-/object.pick-1.3.0.tgz", + "integrity": "sha1-h6EKxMFpS9Lhy/U1kaZhQftd10c=", + "dev": true, + "requires": { + "isobject": "^3.0.1" + } + }, + "object.values": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/object.values/-/object.values-1.1.1.tgz", + "integrity": "sha512-WTa54g2K8iu0kmS/us18jEmdv1a4Wi//BZ/DTVYEcH0XhLM5NYdpDHja3gt57VrZLcNAO2WGA+KpWsDBaHt6eA==", + "dev": true, + "requires": { + "define-properties": "^1.1.3", + "es-abstract": "^1.17.0-next.1", + "function-bind": "^1.1.1", + "has": "^1.0.3" + } + }, + "obuf": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/obuf/-/obuf-1.1.2.tgz", + "integrity": "sha512-PX1wu0AmAdPqOL1mWhqmlOd8kOIZQwGZw6rh7uby9fTc5lhaOWFLX3I6R1hrF9k3zUY40e6igsLGkDXK92LJNg==", + "dev": true + }, + "on-finished": { + "version": "2.3.0", + "resolved": "https://registry.npmjs.org/on-finished/-/on-finished-2.3.0.tgz", + "integrity": "sha1-IPEzZIGwg811M3mSoWlxqi2QaUc=", + "dev": true, + "requires": { + "ee-first": "1.1.1" + } + }, + "on-headers": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/on-headers/-/on-headers-1.0.2.tgz", + "integrity": "sha512-pZAE+FJLoyITytdqK0U5s+FIpjN0JP3OzFi/u8Rx+EV5/W+JTWGXG8xFzevE7AjBfDqHv/8vL8qQsIhHnqRkrA==", + "dev": true + }, + "once": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/once/-/once-1.4.0.tgz", + "integrity": "sha1-WDsap3WWHUsROsF9nFC6753Xa9E=", + "requires": { + "wrappy": "1" + } + }, + "one-time": { + "version": "0.0.4", + "resolved": "https://registry.npmjs.org/one-time/-/one-time-0.0.4.tgz", + "integrity": "sha1-+M33eISCb+Tf+T46nMN7HkSAdC4=", + "dev": true + }, + "onetime": { + "version": "5.1.2", + "resolved": "https://registry.npmjs.org/onetime/-/onetime-5.1.2.tgz", + "integrity": "sha512-kbpaSSGJTWdAY5KPVeMOKXSrPtr8C8C7wodJbcsd51jRnmD+GZu8Y0VoU6Dm5Z4vWr0Ig/1NKuWRKf7j5aaYSg==", + "dev": true, + "requires": { + "mimic-fn": "^2.1.0" + } + }, + "open": { + "version": "7.0.4", + "resolved": "https://registry.npmjs.org/open/-/open-7.0.4.tgz", + "integrity": "sha512-brSA+/yq+b08Hsr4c8fsEW2CRzk1BmfN3SAK/5VCHQ9bdoZJ4qa/+AfR0xHjlbbZUyPkUHs1b8x1RqdyZdkVqQ==", + "dev": true, + "requires": { + "is-docker": "^2.0.0", + "is-wsl": "^2.1.1" + } + }, + "opencollective-postinstall": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/opencollective-postinstall/-/opencollective-postinstall-2.0.3.tgz", + "integrity": "sha512-8AV/sCtuzUeTo8gQK5qDZzARrulB3egtLzFgteqB2tcT4Mw7B8Kt7JcDHmltjz6FOAHsvTevk70gZEbhM4ZS9Q==" + }, + "opener": { + "version": "1.5.2", + "resolved": "https://registry.npmjs.org/opener/-/opener-1.5.2.tgz", + "integrity": "sha512-ur5UIdyw5Y7yEj9wLzhqXiy6GZ3Mwx0yGI+5sMn2r0N0v3cKJvUmFH5yPP+WXh9e0xfyzyJX95D8l088DNFj7A==", + "dev": true + }, + "openurl": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/openurl/-/openurl-1.1.1.tgz", + "integrity": "sha1-OHW0sO96UsFW8NtB1GCduw+Us4c=", + "dev": true + }, + "opn": { + "version": "5.3.0", + "resolved": "https://registry.npmjs.org/opn/-/opn-5.3.0.tgz", + "integrity": "sha512-bYJHo/LOmoTd+pfiYhfZDnf9zekVJrY+cnS2a5F2x+w5ppvTqObojTP7WiFG+kVZs9Inw+qQ/lw7TroWwhdd2g==", + "dev": true, + "requires": { + "is-wsl": "^1.1.0" + }, + "dependencies": { + "is-wsl": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/is-wsl/-/is-wsl-1.1.0.tgz", + "integrity": "sha1-HxbkqiKwTRM2tmGIpmrzxgDDpm0=", + "dev": true + } + } + }, + "optimize-css-assets-webpack-plugin": { + "version": "5.0.3", + "resolved": "https://registry.npmjs.org/optimize-css-assets-webpack-plugin/-/optimize-css-assets-webpack-plugin-5.0.3.tgz", + "integrity": "sha512-q9fbvCRS6EYtUKKSwI87qm2IxlyJK5b4dygW1rKUBT6mMDhdG5e5bZT63v6tnJR9F9FB/H5a0HTmtw+laUBxKA==", + "dev": true, + "requires": { + "cssnano": "^4.1.10", + "last-call-webpack-plugin": "^3.0.0" + } + }, + "optionator": { + "version": "0.8.3", + "resolved": "https://registry.npmjs.org/optionator/-/optionator-0.8.3.tgz", + "integrity": "sha512-+IW9pACdk3XWmmTXG8m3upGUJst5XRGzxMRjXzAuJ1XnIFNvfhjjIuYkDvysnPQ7qzqVzLt78BCruntqRhWQbA==", + "dev": true, + "requires": { + "deep-is": "~0.1.3", + "fast-levenshtein": "~2.0.6", + "levn": "~0.3.0", + "prelude-ls": "~1.1.2", + "type-check": "~0.3.2", + "word-wrap": "~1.2.3" + } + }, + "ora": { + "version": "4.0.4", + "resolved": "https://registry.npmjs.org/ora/-/ora-4.0.4.tgz", + "integrity": "sha512-77iGeVU1cIdRhgFzCK8aw1fbtT1B/iZAvWjS+l/o1x0RShMgxHUZaD2yDpWsNCPwXg9z1ZA78Kbdvr8kBmG/Ww==", + "dev": true, + "requires": { + "chalk": "^3.0.0", + "cli-cursor": "^3.1.0", + "cli-spinners": "^2.2.0", + "is-interactive": "^1.0.0", + "log-symbols": "^3.0.0", + "mute-stream": "0.0.8", + "strip-ansi": "^6.0.0", + "wcwidth": "^1.0.1" + }, + "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": "3.0.0", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-3.0.0.tgz", + "integrity": "sha512-4D3B6Wf41KOYRFdszmDqMCGq5VV/uMAB273JILmO+3jAlh8X4qDtdtgCR3fxtbLEMzSx22QdhnDcJvu2u1fVwg==", + "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 + }, + "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" + } + } + } + }, + "original": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/original/-/original-1.0.2.tgz", + "integrity": "sha512-hyBVl6iqqUOJ8FqRe+l/gS8H+kKYjrEndd5Pm1MfBtsEKA038HkkdbAl/72EAXGyonD/PFsvmVG+EvcIpliMBg==", + "dev": true, + "requires": { + "url-parse": "^1.4.3" + } + }, + "os-browserify": { + "version": "0.3.0", + "resolved": "https://registry.npmjs.org/os-browserify/-/os-browserify-0.3.0.tgz", + "integrity": "sha1-hUNzx/XCMVkU/Jv8a9gjj92h7Cc=", + "dev": true + }, + "os-homedir": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/os-homedir/-/os-homedir-1.0.2.tgz", + "integrity": "sha1-/7xJiDNuDoM94MFox+8VISGqf7M=", + "dev": true + }, + "os-locale": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/os-locale/-/os-locale-1.4.0.tgz", + "integrity": "sha1-IPnxeuKe00XoveWDsT0gCYA8FNk=", + "dev": true, + "requires": { + "lcid": "^1.0.0" + } + }, + "os-name": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/os-name/-/os-name-3.1.0.tgz", + "integrity": "sha512-h8L+8aNjNcMpo/mAIBPn5PXCM16iyPGjHNWo6U1YO8sJTMHtEtyczI6QJnLoplswm6goopQkqc7OAnjhWcugVg==", + "dev": true, + "requires": { + "macos-release": "^2.2.0", + "windows-release": "^3.1.0" + } + }, + "os-shim": { + "version": "0.1.3", + "resolved": "https://registry.npmjs.org/os-shim/-/os-shim-0.1.3.tgz", + "integrity": "sha1-a2LDeRz3kJ6jXtRuF2WLtBfLORc=", + "dev": true + }, + "os-tmpdir": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/os-tmpdir/-/os-tmpdir-1.0.2.tgz", + "integrity": "sha1-u+Z0BseaqFxc/sdm/lc0VV36EnQ=", + "dev": true + }, + "osenv": { + "version": "0.1.5", + "resolved": "https://registry.npmjs.org/osenv/-/osenv-0.1.5.tgz", + "integrity": "sha512-0CWcCECdMVc2Rw3U5w9ZjqX6ga6ubk1xDVKxtBQPK7wis/0F2r9T6k4ydGYhecl7YUBxBVxhL5oisPsNxAPe2g==", + "dev": true, + "requires": { + "os-homedir": "^1.0.0", + "os-tmpdir": "^1.0.0" + } + }, + "p-defer": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/p-defer/-/p-defer-1.0.0.tgz", + "integrity": "sha1-n26xgvbJqozXQwBKfU+WsZaw+ww=", + "dev": true + }, + "p-each-series": { + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/p-each-series/-/p-each-series-2.2.0.tgz", + "integrity": "sha512-ycIL2+1V32th+8scbpTvyHNaHe02z0sjgh91XXjAk+ZeXoPN4Z46DVUnzdso0aX4KckKw0FNNFHdjZ2UsZvxiA==", + "dev": true + }, + "p-finally": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/p-finally/-/p-finally-1.0.0.tgz", + "integrity": "sha1-P7z7FbiZpEEjs0ttzBi3JDNqLK4=", + "dev": true + }, + "p-is-promise": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/p-is-promise/-/p-is-promise-2.1.0.tgz", + "integrity": "sha512-Y3W0wlRPK8ZMRbNq97l4M5otioeA5lm1z7bkNkxCka8HSPjR0xRWmpCmc9utiaLP9Jb1eD8BgeIxTW4AIF45Pg==", + "dev": true + }, + "p-limit": { + "version": "2.3.0", + "resolved": "https://registry.npmjs.org/p-limit/-/p-limit-2.3.0.tgz", + "integrity": "sha512-//88mFWSJx8lxCzwdAABTJL2MyWB12+eIY7MDL2SqLmAkeKU9qxRvWuSyTjm3FUmpBEMuFfckAIqEaVGUDxb6w==", + "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==", + "requires": { + "p-limit": "^2.2.0" + } + }, + "p-map": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/p-map/-/p-map-4.0.0.tgz", + "integrity": "sha512-/bjOqmgETBYB5BoEeGVea8dmvHb2m9GLy1E9W43yeyfP6QQCZGFNa+XRceJEuDB6zqr+gKpIAmlLebMpykw/MQ==", + "dev": true, + "requires": { + "aggregate-error": "^3.0.0" + } + }, + "p-retry": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/p-retry/-/p-retry-3.0.1.tgz", + "integrity": "sha512-XE6G4+YTTkT2a0UWb2kjZe8xNwf8bIbnqpc/IS/idOBVhyves0mK5OJgeocjx7q5pvX/6m23xuzVPYT1uGM73w==", + "dev": true, + "requires": { + "retry": "^0.12.0" + }, + "dependencies": { + "retry": { + "version": "0.12.0", + "resolved": "https://registry.npmjs.org/retry/-/retry-0.12.0.tgz", + "integrity": "sha1-G0KmJmoh8HQh0bC1S33BZ7AcATs=", + "dev": true + } + } + }, + "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==" + }, + "pacote": { + "version": "9.5.12", + "resolved": "https://registry.npmjs.org/pacote/-/pacote-9.5.12.tgz", + "integrity": "sha512-BUIj/4kKbwWg4RtnBncXPJd15piFSVNpTzY0rysSr3VnMowTYgkGKcaHrbReepAkjTr8lH2CVWRi58Spg2CicQ==", + "dev": true, + "requires": { + "bluebird": "^3.5.3", + "cacache": "^12.0.2", + "chownr": "^1.1.2", + "figgy-pudding": "^3.5.1", + "get-stream": "^4.1.0", + "glob": "^7.1.3", + "infer-owner": "^1.0.4", + "lru-cache": "^5.1.1", + "make-fetch-happen": "^5.0.0", + "minimatch": "^3.0.4", + "minipass": "^2.3.5", + "mississippi": "^3.0.0", + "mkdirp": "^0.5.1", + "normalize-package-data": "^2.4.0", + "npm-normalize-package-bin": "^1.0.0", + "npm-package-arg": "^6.1.0", + "npm-packlist": "^1.1.12", + "npm-pick-manifest": "^3.0.0", + "npm-registry-fetch": "^4.0.0", + "osenv": "^0.1.5", + "promise-inflight": "^1.0.1", + "promise-retry": "^1.1.1", + "protoduck": "^5.0.1", + "rimraf": "^2.6.2", + "safe-buffer": "^5.1.2", + "semver": "^5.6.0", + "ssri": "^6.0.1", + "tar": "^4.4.10", + "unique-filename": "^1.1.1", + "which": "^1.3.1" + }, + "dependencies": { + "glob": { + "version": "7.1.6", + "resolved": "https://registry.npmjs.org/glob/-/glob-7.1.6.tgz", + "integrity": "sha512-LwaxwyZ72Lk7vZINtNNrywX0ZuLyStrdDtabefZKAY5ZGJhVtgdznluResxNmPitE0SAO+O26sWTHeKSI2wMBA==", + "dev": true, + "requires": { + "fs.realpath": "^1.0.0", + "inflight": "^1.0.4", + "inherits": "2", + "minimatch": "^3.0.4", + "once": "^1.3.0", + "path-is-absolute": "^1.0.0" + } + }, + "hosted-git-info": { + "version": "2.8.8", + "resolved": "https://registry.npmjs.org/hosted-git-info/-/hosted-git-info-2.8.8.tgz", + "integrity": "sha512-f/wzC2QaWBs7t9IYqB4T3sR1xviIViXJRJTWBlx2Gf3g0Xi5vI7Yy4koXQ1c9OYDGHN9sBy1DQ2AB8fqZBWhUg==", + "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, + "requires": { + "yallist": "^3.0.2" + } + }, + "npm-package-arg": { + "version": "6.1.1", + "resolved": "https://registry.npmjs.org/npm-package-arg/-/npm-package-arg-6.1.1.tgz", + "integrity": "sha512-qBpssaL3IOZWi5vEKUKW0cO7kzLeT+EQO9W8RsLOZf76KF9E/K9+wH0C7t06HXPpaH8WH5xF1MExLuCwbTqRUg==", + "dev": true, + "requires": { + "hosted-git-info": "^2.7.1", + "osenv": "^0.1.5", + "semver": "^5.6.0", + "validate-npm-package-name": "^3.0.0" + } + }, + "npm-pick-manifest": { + "version": "3.0.2", + "resolved": "https://registry.npmjs.org/npm-pick-manifest/-/npm-pick-manifest-3.0.2.tgz", + "integrity": "sha512-wNprTNg+X5nf+tDi+hbjdHhM4bX+mKqv6XmPh7B5eG+QY9VARfQPfCEH013H5GqfNj6ee8Ij2fg8yk0mzps1Vw==", + "dev": true, + "requires": { + "figgy-pudding": "^3.5.1", + "npm-package-arg": "^6.0.0", + "semver": "^5.4.1" + } + }, + "rimraf": { + "version": "2.7.1", + "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-2.7.1.tgz", + "integrity": "sha512-uWjbaKIK3T1OSVptzX7Nl6PvQ3qAGtKEtVRjRuazjfL3Bx5eI409VZSqgND+4UNnmzLVdPj9FqFJNPqBZFve4w==", + "dev": true, + "requires": { + "glob": "^7.1.3" + } + }, + "yallist": { + "version": "3.1.1", + "resolved": "https://registry.npmjs.org/yallist/-/yallist-3.1.1.tgz", + "integrity": "sha512-a4UGQaWPH59mOXUYnAG2ewncQS4i4F43Tv3JoAM+s2VDAmS9NsK8GpDMLrCHPksFT7h3K6TOoUNn2pb7RoXx4g==", + "dev": true + } + } + }, + "paged-request": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/paged-request/-/paged-request-2.0.1.tgz", + "integrity": "sha512-C0bB/PFk9rQskD1YEiz7uuchzqKDQGgdsEHN1ahify0UUWzgmMK4NDG9fhlQg2waogmNFwEvEeHfMRvJySpdVw==", + "dev": true, + "requires": { + "axios": "^0.18.0" + }, + "dependencies": { + "axios": { + "version": "0.18.1", + "resolved": "https://registry.npmjs.org/axios/-/axios-0.18.1.tgz", + "integrity": "sha512-0BfJq4NSfQXd+SkFdrvFbG7addhYSBA2mQwISr46pD6E5iqkWg02RAs8vyTT/j0RTnoYmeXauBuSv1qKwR179g==", + "dev": true, + "requires": { + "follow-redirects": "1.5.10", + "is-buffer": "^2.0.2" + } + }, + "is-buffer": { + "version": "2.0.5", + "resolved": "https://registry.npmjs.org/is-buffer/-/is-buffer-2.0.5.tgz", + "integrity": "sha512-i2R6zNFDwgEHJyQUtJEk0XFi1i0dPFn/oqjK3/vPCcDeJvW5NQ83V8QbicfF1SupOaB0h8ntgBC2YiE7dfyctQ==", + "dev": true + } + } + }, + "pako": { + "version": "1.0.11", + "resolved": "https://registry.npmjs.org/pako/-/pako-1.0.11.tgz", + "integrity": "sha512-4hLB8Py4zZce5s4yd9XzopqwVv/yGNhV1Bl8NTmCq1763HeK2+EwVTv+leGeL13Dnh2wfbqowVPXCIO0z4taYw==", + "dev": true + }, + "parallel-transform": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/parallel-transform/-/parallel-transform-1.2.0.tgz", + "integrity": "sha512-P2vSmIu38uIlvdcU7fDkyrxj33gTUy/ABO5ZUbGowxNCopBq/OoD42bP4UmMrJoPyk4Uqf0mu3mtWBhHCZD8yg==", + "dev": true, + "requires": { + "cyclist": "^1.0.1", + "inherits": "^2.0.3", + "readable-stream": "^2.1.5" + } + }, + "param-case": { + "version": "3.0.3", + "resolved": "https://registry.npmjs.org/param-case/-/param-case-3.0.3.tgz", + "integrity": "sha512-VWBVyimc1+QrzappRs7waeN2YmoZFCGXWASRYX1/rGHtXqEcrGEIDm+jqIwFa2fRXNgQEwrxaYuIrX0WcAguTA==", + "dev": true, + "requires": { + "dot-case": "^3.0.3", + "tslib": "^1.10.0" + }, + "dependencies": { + "tslib": { + "version": "1.14.1", + "resolved": "https://registry.npmjs.org/tslib/-/tslib-1.14.1.tgz", + "integrity": "sha512-Xni35NKzjgMrwevysHTCArtLDpPvye8zV/0E4EyYn43P7/7qvQwPh9BGkHewbMulVntbigmcT7rdX3BNo9wRJg==", + "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" + } + }, + "parse-asn1": { + "version": "5.1.6", + "resolved": "https://registry.npmjs.org/parse-asn1/-/parse-asn1-5.1.6.tgz", + "integrity": "sha512-RnZRo1EPU6JBnra2vGHj0yhp6ebyjBZpmUCLHWiFhxlzvBCCpAuZ7elsBp1PVAbQN0/04VD/19rfzlBSwLstMw==", + "dev": true, + "requires": { + "asn1.js": "^5.2.0", + "browserify-aes": "^1.0.0", + "evp_bytestokey": "^1.0.0", + "pbkdf2": "^3.0.3", + "safe-buffer": "^5.1.1" + } + }, + "parse-cache-control": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/parse-cache-control/-/parse-cache-control-1.0.1.tgz", + "integrity": "sha1-juqz5U+laSD+Fro493+iGqzC104=", + "dev": true + }, + "parse-gitignore": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/parse-gitignore/-/parse-gitignore-1.0.1.tgz", + "integrity": "sha512-UGyowyjtx26n65kdAMWhm6/3uy5uSrpcuH7tt+QEVudiBoVS+eqHxD5kbi9oWVRwj7sCzXqwuM+rUGw7earl6A==", + "dev": true + }, + "parse-json": { + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/parse-json/-/parse-json-2.2.0.tgz", + "integrity": "sha1-9ID0BDTvgHQfhGkJn43qGPVaTck=", + "dev": true, + "requires": { + "error-ex": "^1.2.0" + } + }, + "parse-passwd": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/parse-passwd/-/parse-passwd-1.0.0.tgz", + "integrity": "sha1-bVuTSkVpk7I9N/QKOC1vFmao5cY=", + "dev": true + }, + "parse-srcset": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/parse-srcset/-/parse-srcset-1.0.2.tgz", + "integrity": "sha1-8r0iH2zJcKk42IVWq8WJyqqiveE=", + "dev": true + }, + "parse5": { + "version": "5.1.1", + "resolved": "https://registry.npmjs.org/parse5/-/parse5-5.1.1.tgz", + "integrity": "sha512-ugq4DFI0Ptb+WWjAdOK16+u/nHfiIrcE+sh8kZMaM0WllQKLI9rOUq6c2b7cwPkXdzfQESqvoqK6ug7U/Yyzug==", + "dev": true + }, + "parseqs": { + "version": "0.0.6", + "resolved": "https://registry.npmjs.org/parseqs/-/parseqs-0.0.6.tgz", + "integrity": "sha512-jeAGzMDbfSHHA091hr0r31eYfTig+29g3GKKE/PPbEQ65X0lmMwlEoqmhzu0iztID5uJpZsFlUPDP8ThPL7M8w==", + "dev": true + }, + "parseuri": { + "version": "0.0.6", + "resolved": "https://registry.npmjs.org/parseuri/-/parseuri-0.0.6.tgz", + "integrity": "sha512-AUjen8sAkGgao7UyCX6Ahv0gIK2fABKmYjvP4xmy5JaKvcbTRueIqIPHLAfq30xJddqSE033IOMUSOMCcK3Sow==", + "dev": true + }, + "parseurl": { + "version": "1.3.3", + "resolved": "https://registry.npmjs.org/parseurl/-/parseurl-1.3.3.tgz", + "integrity": "sha512-CiyeOxFT/JZyN5m0z9PfXw4SCBJ6Sygz1Dpl0wqjlhDEGGBP1GnsUVEL0p63hoG1fcj3fHynXi9NYO4nWOL+qQ==", + "dev": true + }, + "pascal-case": { + "version": "3.1.1", + "resolved": "https://registry.npmjs.org/pascal-case/-/pascal-case-3.1.1.tgz", + "integrity": "sha512-XIeHKqIrsquVTQL2crjq3NfJUxmdLasn3TYOU0VBM+UX2a6ztAWBlJQBePLGY7VHW8+2dRadeIPK5+KImwTxQA==", + "dev": true, + "requires": { + "no-case": "^3.0.3", + "tslib": "^1.10.0" + }, + "dependencies": { + "tslib": { + "version": "1.14.1", + "resolved": "https://registry.npmjs.org/tslib/-/tslib-1.14.1.tgz", + "integrity": "sha512-Xni35NKzjgMrwevysHTCArtLDpPvye8zV/0E4EyYn43P7/7qvQwPh9BGkHewbMulVntbigmcT7rdX3BNo9wRJg==", + "dev": true + } + } + }, + "pascalcase": { + "version": "0.1.1", + "resolved": "https://registry.npmjs.org/pascalcase/-/pascalcase-0.1.1.tgz", + "integrity": "sha1-s2PlXoAGym/iF4TS2yK9FdeRfxQ=", + "dev": true + }, + "path-browserify": { + "version": "0.0.1", + "resolved": "https://registry.npmjs.org/path-browserify/-/path-browserify-0.0.1.tgz", + "integrity": "sha512-BapA40NHICOS+USX9SN4tyhq+A2RrN/Ws5F0Z5aMHDp98Fl86lX8Oti8B7uN93L4Ifv4fHOEA+pQw87gmMO/lQ==", + "dev": true + }, + "path-dirname": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/path-dirname/-/path-dirname-1.0.2.tgz", + "integrity": "sha1-zDPSTVJeCZpTiMAzbG4yuRYGCeA=", + "dev": true + }, + "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==" + }, + "path-is-absolute": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/path-is-absolute/-/path-is-absolute-1.0.1.tgz", + "integrity": "sha1-F0uSaHNVNP+8es5r9TpanhtcX18=" + }, + "path-is-inside": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/path-is-inside/-/path-is-inside-1.0.2.tgz", + "integrity": "sha1-NlQX3t5EQw0cEa9hAn+s8HS9/FM=", + "dev": true + }, + "path-key": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/path-key/-/path-key-2.0.1.tgz", + "integrity": "sha1-QRyttXTFoUDTpLGRDUDYDMn0C0A=", + "dev": true + }, + "path-parse": { + "version": "1.0.6", + "resolved": "https://registry.npmjs.org/path-parse/-/path-parse-1.0.6.tgz", + "integrity": "sha512-GSmOT2EbHrINBf9SR7CDELwlJ8AENk3Qn7OikK4nFYAu3Ote2+JYNVvkpAEQm3/TLNEJFD/xZJjzyxg3KBWOzw==" + }, + "path-to-regexp": { + "version": "0.1.7", + "resolved": "https://registry.npmjs.org/path-to-regexp/-/path-to-regexp-0.1.7.tgz", + "integrity": "sha1-32BBeABfUi8V60SQ5yR6G/qmf4w=", + "dev": true + }, + "path-type": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/path-type/-/path-type-1.1.0.tgz", + "integrity": "sha1-WcRPfuSR2nBNpBXaWkBwuk+P5EE=", + "dev": true, + "requires": { + "graceful-fs": "^4.1.2", + "pify": "^2.0.0", + "pinkie-promise": "^2.0.0" + } + }, + "pbkdf2": { + "version": "3.1.1", + "resolved": "https://registry.npmjs.org/pbkdf2/-/pbkdf2-3.1.1.tgz", + "integrity": "sha512-4Ejy1OPxi9f2tt1rRV7Go7zmfDQ+ZectEQz3VGUQhgq62HtIRPDyG/JtnwIxs6x3uNMwo2V7q1fMvKjb+Tnpqg==", + "dev": true, + "requires": { + "create-hash": "^1.1.2", + "create-hmac": "^1.1.4", + "ripemd160": "^2.0.1", + "safe-buffer": "^5.0.1", + "sha.js": "^2.4.8" + } + }, + "performance-now": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/performance-now/-/performance-now-2.1.0.tgz", + "integrity": "sha1-Ywn04OX6kT7BxpMHrjZLSzd8nns=", + "dev": true + }, + "picomatch": { + "version": "2.2.2", + "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-2.2.2.tgz", + "integrity": "sha512-q0M/9eZHzmr0AulXyPwNfZjtwZ/RBZlbN3K3CErVrk50T2ASYI7Bye0EvekFY3IP1Nt2DHu0re+V2ZHIpMkuWg==", + "dev": true + }, + "pify": { + "version": "2.3.0", + "resolved": "https://registry.npmjs.org/pify/-/pify-2.3.0.tgz", + "integrity": "sha1-7RQaasBDqEnqWISY59yosVMw6Qw=", + "dev": true + }, + "pinkie": { + "version": "2.0.4", + "resolved": "https://registry.npmjs.org/pinkie/-/pinkie-2.0.4.tgz", + "integrity": "sha1-clVrgM+g1IqXToDnckjoDtT3+HA=", + "dev": true + }, + "pinkie-promise": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/pinkie-promise/-/pinkie-promise-2.0.1.tgz", + "integrity": "sha1-ITXW36ejWMBprJsXh3YogihFD/o=", + "dev": true, + "requires": { + "pinkie": "^2.0.0" + } + }, + "pirates": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/pirates/-/pirates-4.0.1.tgz", + "integrity": "sha512-WuNqLTbMI3tmfef2TKxlQmAiLHKtFhlsCZnPIpuv2Ow0RDVO8lfy1Opf4NUzlMXLjPl+Men7AuVdX6TA+s+uGA==", + "dev": true, + "requires": { + "node-modules-regexp": "^1.0.0" + } + }, + "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" + } + }, + "pkg-up": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/pkg-up/-/pkg-up-3.1.0.tgz", + "integrity": "sha512-nDywThFk1i4BQK4twPQ6TA4RT8bDY96yeuCVBWL3ePARCiEKDRSrNGbFIgUJpLp+XeIR65v8ra7WuJOFUBtkMA==", + "dev": true, + "requires": { + "find-up": "^3.0.0" + }, + "dependencies": { + "find-up": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/find-up/-/find-up-3.0.0.tgz", + "integrity": "sha512-1yD6RmLI1XBfxugvORwlck6f75tYL+iR0jqwsOrOxMZyGYqUuDhJ0l4AXdO1iX/FTs9cBAMEk1gWSEx1kSbylg==", + "dev": true, + "requires": { + "locate-path": "^3.0.0" + } + }, + "locate-path": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-3.0.0.tgz", + "integrity": "sha512-7AO748wWnIhNqAuaty2ZWHkQHRSNfPVIsPIfwEOWO22AmaoVrWavlOcMR5nzTLNYvp36X220/maaRsrec1G65A==", + "dev": true, + "requires": { + "p-locate": "^3.0.0", + "path-exists": "^3.0.0" + } + }, + "p-locate": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-3.0.0.tgz", + "integrity": "sha512-x+12w/To+4GFfgJhBEpiDcLozRJGegY+Ei7/z0tSLkMmxGZNybVMSfWj9aJn8Z5Fc7dBUNJOOVgPv2H7IwulSQ==", + "dev": true, + "requires": { + "p-limit": "^2.0.0" + } + }, + "path-exists": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/path-exists/-/path-exists-3.0.0.tgz", + "integrity": "sha1-zg6+ql94yxiSXqfYENe1mwEP1RU=", + "dev": true + } + } + }, + "please-upgrade-node": { + "version": "3.2.0", + "resolved": "https://registry.npmjs.org/please-upgrade-node/-/please-upgrade-node-3.2.0.tgz", + "integrity": "sha512-gQR3WpIgNIKwBMVLkpMUeR3e1/E1y42bqDQZfql+kDeXd8COYfM8PQA4X6y7a8u9Ua9FHmsrrmirW2vHs45hWg==", + "dev": true, + "requires": { + "semver-compare": "^1.0.0" + } + }, + "plugin-error": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/plugin-error/-/plugin-error-1.0.1.tgz", + "integrity": "sha512-L1zP0dk7vGweZME2i+EeakvUNqSrdiI3F91TwEoYiGrAfUXmVv6fJIq4g82PAXxNsWOp0J7ZqQy/3Szz0ajTxA==", + "dev": true, + "requires": { + "ansi-colors": "^1.0.1", + "arr-diff": "^4.0.0", + "arr-union": "^3.1.0", + "extend-shallow": "^3.0.2" + }, + "dependencies": { + "ansi-colors": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/ansi-colors/-/ansi-colors-1.1.0.tgz", + "integrity": "sha512-SFKX67auSNoVR38N3L+nvsPjOE0bybKTYbkf5tRvushrAPQ9V75huw0ZxBkKVeRU9kqH3d6HA4xTckbwZ4ixmA==", + "dev": true, + "requires": { + "ansi-wrap": "^0.1.0" + } + } + } + }, + "pluralize": { + "version": "8.0.0", + "resolved": "https://registry.npmjs.org/pluralize/-/pluralize-8.0.0.tgz", + "integrity": "sha512-Nc3IT5yHzflTfbjgqWcCPpo7DaKy4FnpB0l/zCAW0Tc7jxAiuqSxHasntB3D7887LSrA93kDJ9IXovxJYxyLCA==", + "dev": true + }, + "portfinder": { + "version": "1.0.28", + "resolved": "https://registry.npmjs.org/portfinder/-/portfinder-1.0.28.tgz", + "integrity": "sha512-Se+2isanIcEqf2XMHjyUKskczxbPH7dQnlMjXX6+dybayyHvAf/TCgyMRlzf/B6QDhAEFOGes0pzRo3by4AbMA==", + "dev": true, + "requires": { + "async": "^2.6.2", + "debug": "^3.1.1", + "mkdirp": "^0.5.5" + }, + "dependencies": { + "async": { + "version": "2.6.3", + "resolved": "https://registry.npmjs.org/async/-/async-2.6.3.tgz", + "integrity": "sha512-zflvls11DCy+dQWzTW2dzuilv8Z5X/pjfmZOWba6TNIVDm+2UDaJmXSOXlasHKfNBs8oo3M0aT50fDEWfKZjXg==", + "dev": true, + "requires": { + "lodash": "^4.17.14" + } + }, + "debug": { + "version": "3.2.7", + "resolved": "https://registry.npmjs.org/debug/-/debug-3.2.7.tgz", + "integrity": "sha512-CFjzYYAi4ThfiQvizrFQevTTXHtnCqWfe7x1AhgEscTz6ZbLbfoLRLPugTQyBth6f8ZERVUSyWHFD/7Wu4t1XQ==", + "dev": true, + "requires": { + "ms": "^2.1.1" + } + } + } + }, + "portscanner": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/portscanner/-/portscanner-2.1.1.tgz", + "integrity": "sha1-6rtAnk3iSVD1oqUW01rnaTQ/u5Y=", + "dev": true, + "requires": { + "async": "1.5.2", + "is-number-like": "^1.0.3" + } + }, + "posix-character-classes": { + "version": "0.1.1", + "resolved": "https://registry.npmjs.org/posix-character-classes/-/posix-character-classes-0.1.1.tgz", + "integrity": "sha1-AerA/jta9xoqbAL+q7jB/vfgDqs=", + "dev": true + }, + "postcss": { + "version": "7.0.35", + "resolved": "https://registry.npmjs.org/postcss/-/postcss-7.0.35.tgz", + "integrity": "sha512-3QT8bBJeX/S5zKTTjTCIjRF3If4avAT6kqxcASlTWEtAFCb9NH0OUxNDfgZSWdP5fJnBYCMEWkIFfWeugjzYMg==", + "dev": true, + "requires": { + "chalk": "^2.4.2", + "source-map": "^0.6.1", + "supports-color": "^6.1.0" + }, + "dependencies": { + "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 + }, + "supports-color": { + "version": "6.1.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-6.1.0.tgz", + "integrity": "sha512-qe1jfm1Mg7Nq/NSh6XE24gPXROEVsWHxC1LIx//XNlD9iw7YZQGjZNjYN7xGaEG6iKdA8EtNFW6R0gjnVXp+wQ==", + "dev": true, + "requires": { + "has-flag": "^3.0.0" + } + } + } + }, + "postcss-calc": { + "version": "7.0.5", + "resolved": "https://registry.npmjs.org/postcss-calc/-/postcss-calc-7.0.5.tgz", + "integrity": "sha512-1tKHutbGtLtEZF6PT4JSihCHfIVldU72mZ8SdZHIYriIZ9fh9k9aWSppaT8rHsyI3dX+KSR+W+Ix9BMY3AODrg==", + "dev": true, + "requires": { + "postcss": "^7.0.27", + "postcss-selector-parser": "^6.0.2", + "postcss-value-parser": "^4.0.2" + } + }, + "postcss-colormin": { + "version": "4.0.3", + "resolved": "https://registry.npmjs.org/postcss-colormin/-/postcss-colormin-4.0.3.tgz", + "integrity": "sha512-WyQFAdDZpExQh32j0U0feWisZ0dmOtPl44qYmJKkq9xFWY3p+4qnRzCHeNrkeRhwPHz9bQ3mo0/yVkaply0MNw==", + "dev": true, + "requires": { + "browserslist": "^4.0.0", + "color": "^3.0.0", + "has": "^1.0.0", + "postcss": "^7.0.0", + "postcss-value-parser": "^3.0.0" + }, + "dependencies": { + "postcss-value-parser": { + "version": "3.3.1", + "resolved": "https://registry.npmjs.org/postcss-value-parser/-/postcss-value-parser-3.3.1.tgz", + "integrity": "sha512-pISE66AbVkp4fDQ7VHBwRNXzAAKJjw4Vw7nWI/+Q3vuly7SNfgYXvm6i5IgFylHGK5sP/xHAbB7N49OS4gWNyQ==", + "dev": true + } + } + }, + "postcss-convert-values": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/postcss-convert-values/-/postcss-convert-values-4.0.1.tgz", + "integrity": "sha512-Kisdo1y77KUC0Jmn0OXU/COOJbzM8cImvw1ZFsBgBgMgb1iL23Zs/LXRe3r+EZqM3vGYKdQ2YJVQ5VkJI+zEJQ==", + "dev": true, + "requires": { + "postcss": "^7.0.0", + "postcss-value-parser": "^3.0.0" + }, + "dependencies": { + "postcss-value-parser": { + "version": "3.3.1", + "resolved": "https://registry.npmjs.org/postcss-value-parser/-/postcss-value-parser-3.3.1.tgz", + "integrity": "sha512-pISE66AbVkp4fDQ7VHBwRNXzAAKJjw4Vw7nWI/+Q3vuly7SNfgYXvm6i5IgFylHGK5sP/xHAbB7N49OS4gWNyQ==", + "dev": true + } + } + }, + "postcss-discard-comments": { + "version": "4.0.2", + "resolved": "https://registry.npmjs.org/postcss-discard-comments/-/postcss-discard-comments-4.0.2.tgz", + "integrity": "sha512-RJutN259iuRf3IW7GZyLM5Sw4GLTOH8FmsXBnv8Ab/Tc2k4SR4qbV4DNbyyY4+Sjo362SyDmW2DQ7lBSChrpkg==", + "dev": true, + "requires": { + "postcss": "^7.0.0" + } + }, + "postcss-discard-duplicates": { + "version": "4.0.2", + "resolved": "https://registry.npmjs.org/postcss-discard-duplicates/-/postcss-discard-duplicates-4.0.2.tgz", + "integrity": "sha512-ZNQfR1gPNAiXZhgENFfEglF93pciw0WxMkJeVmw8eF+JZBbMD7jp6C67GqJAXVZP2BWbOztKfbsdmMp/k8c6oQ==", + "dev": true, + "requires": { + "postcss": "^7.0.0" + } + }, + "postcss-discard-empty": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/postcss-discard-empty/-/postcss-discard-empty-4.0.1.tgz", + "integrity": "sha512-B9miTzbznhDjTfjvipfHoqbWKwd0Mj+/fL5s1QOz06wufguil+Xheo4XpOnc4NqKYBCNqqEzgPv2aPBIJLox0w==", + "dev": true, + "requires": { + "postcss": "^7.0.0" + } + }, + "postcss-discard-overridden": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/postcss-discard-overridden/-/postcss-discard-overridden-4.0.1.tgz", + "integrity": "sha512-IYY2bEDD7g1XM1IDEsUT4//iEYCxAmP5oDSFMVU/JVvT7gh+l4fmjciLqGgwjdWpQIdb0Che2VX00QObS5+cTg==", + "dev": true, + "requires": { + "postcss": "^7.0.0" + } + }, + "postcss-load-config": { + "version": "2.1.2", + "resolved": "https://registry.npmjs.org/postcss-load-config/-/postcss-load-config-2.1.2.tgz", + "integrity": "sha512-/rDeGV6vMUo3mwJZmeHfEDvwnTKKqQ0S7OHUi/kJvvtx3aWtyWG2/0ZWnzCt2keEclwN6Tf0DST2v9kITdOKYw==", + "dev": true, + "requires": { + "cosmiconfig": "^5.0.0", + "import-cwd": "^2.0.0" + }, + "dependencies": { + "cosmiconfig": { + "version": "5.2.1", + "resolved": "https://registry.npmjs.org/cosmiconfig/-/cosmiconfig-5.2.1.tgz", + "integrity": "sha512-H65gsXo1SKjf8zmrJ67eJk8aIRKV5ff2D4uKZIBZShbhGSpEmsQOPW/SKMKYhSTrqR7ufy6RP69rPogdaPh/kA==", + "dev": true, + "requires": { + "import-fresh": "^2.0.0", + "is-directory": "^0.3.1", + "js-yaml": "^3.13.1", + "parse-json": "^4.0.0" + } + }, + "import-fresh": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/import-fresh/-/import-fresh-2.0.0.tgz", + "integrity": "sha1-2BNVwVYS04bGH53dOSLUMEgipUY=", + "dev": true, + "requires": { + "caller-path": "^2.0.0", + "resolve-from": "^3.0.0" + } + }, + "parse-json": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/parse-json/-/parse-json-4.0.0.tgz", + "integrity": "sha1-vjX1Qlvh9/bHRxhPmKeIy5lHfuA=", + "dev": true, + "requires": { + "error-ex": "^1.3.1", + "json-parse-better-errors": "^1.0.1" + } + }, + "resolve-from": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/resolve-from/-/resolve-from-3.0.0.tgz", + "integrity": "sha1-six699nWiBvItuZTM17rywoYh0g=", + "dev": true + } + } + }, + "postcss-loader": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/postcss-loader/-/postcss-loader-3.0.0.tgz", + "integrity": "sha512-cLWoDEY5OwHcAjDnkyRQzAXfs2jrKjXpO/HQFcc5b5u/r7aa471wdmChmwfnv7x2u840iat/wi0lQ5nbRgSkUA==", + "dev": true, + "requires": { + "loader-utils": "^1.1.0", + "postcss": "^7.0.0", + "postcss-load-config": "^2.0.0", + "schema-utils": "^1.0.0" + }, + "dependencies": { + "json5": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/json5/-/json5-1.0.1.tgz", + "integrity": "sha512-aKS4WQjPenRxiQsC93MNfjx+nbF4PAdYzmd/1JIj8HYzqfbu86beTuNgXDzPknWk0n0uARlyewZo4s++ES36Ow==", + "dev": true, + "requires": { + "minimist": "^1.2.0" + } + }, + "loader-utils": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/loader-utils/-/loader-utils-1.4.0.tgz", + "integrity": "sha512-qH0WSMBtn/oHuwjy/NucEgbx5dbxxnxup9s4PVXJUDHZBQY+s0NWA9rJf53RBnQZxfch7euUui7hpoAPvALZdA==", + "dev": true, + "requires": { + "big.js": "^5.2.2", + "emojis-list": "^3.0.0", + "json5": "^1.0.1" + } + }, + "schema-utils": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/schema-utils/-/schema-utils-1.0.0.tgz", + "integrity": "sha512-i27Mic4KovM/lnGsy8whRCHhc7VicJajAjTrYg11K9zfZXnYIt4k5F+kZkwjnrhKzLic/HLU4j11mjsz2G/75g==", + "dev": true, + "requires": { + "ajv": "^6.1.0", + "ajv-errors": "^1.0.0", + "ajv-keywords": "^3.1.0" + } + } + } + }, + "postcss-merge-longhand": { + "version": "4.0.11", + "resolved": "https://registry.npmjs.org/postcss-merge-longhand/-/postcss-merge-longhand-4.0.11.tgz", + "integrity": "sha512-alx/zmoeXvJjp7L4mxEMjh8lxVlDFX1gqWHzaaQewwMZiVhLo42TEClKaeHbRf6J7j82ZOdTJ808RtN0ZOZwvw==", + "dev": true, + "requires": { + "css-color-names": "0.0.4", + "postcss": "^7.0.0", + "postcss-value-parser": "^3.0.0", + "stylehacks": "^4.0.0" + }, + "dependencies": { + "postcss-value-parser": { + "version": "3.3.1", + "resolved": "https://registry.npmjs.org/postcss-value-parser/-/postcss-value-parser-3.3.1.tgz", + "integrity": "sha512-pISE66AbVkp4fDQ7VHBwRNXzAAKJjw4Vw7nWI/+Q3vuly7SNfgYXvm6i5IgFylHGK5sP/xHAbB7N49OS4gWNyQ==", + "dev": true + } + } + }, + "postcss-merge-rules": { + "version": "4.0.3", + "resolved": "https://registry.npmjs.org/postcss-merge-rules/-/postcss-merge-rules-4.0.3.tgz", + "integrity": "sha512-U7e3r1SbvYzO0Jr3UT/zKBVgYYyhAz0aitvGIYOYK5CPmkNih+WDSsS5tvPrJ8YMQYlEMvsZIiqmn7HdFUaeEQ==", + "dev": true, + "requires": { + "browserslist": "^4.0.0", + "caniuse-api": "^3.0.0", + "cssnano-util-same-parent": "^4.0.0", + "postcss": "^7.0.0", + "postcss-selector-parser": "^3.0.0", + "vendors": "^1.0.0" + }, + "dependencies": { + "postcss-selector-parser": { + "version": "3.1.2", + "resolved": "https://registry.npmjs.org/postcss-selector-parser/-/postcss-selector-parser-3.1.2.tgz", + "integrity": "sha512-h7fJ/5uWuRVyOtkO45pnt1Ih40CEleeyCHzipqAZO2e5H20g25Y48uYnFUiShvY4rZWNJ/Bib/KVPmanaCtOhA==", + "dev": true, + "requires": { + "dot-prop": "^5.2.0", + "indexes-of": "^1.0.1", + "uniq": "^1.0.1" + } + } + } + }, + "postcss-minify-font-values": { + "version": "4.0.2", + "resolved": "https://registry.npmjs.org/postcss-minify-font-values/-/postcss-minify-font-values-4.0.2.tgz", + "integrity": "sha512-j85oO6OnRU9zPf04+PZv1LYIYOprWm6IA6zkXkrJXyRveDEuQggG6tvoy8ir8ZwjLxLuGfNkCZEQG7zan+Hbtg==", + "dev": true, + "requires": { + "postcss": "^7.0.0", + "postcss-value-parser": "^3.0.0" + }, + "dependencies": { + "postcss-value-parser": { + "version": "3.3.1", + "resolved": "https://registry.npmjs.org/postcss-value-parser/-/postcss-value-parser-3.3.1.tgz", + "integrity": "sha512-pISE66AbVkp4fDQ7VHBwRNXzAAKJjw4Vw7nWI/+Q3vuly7SNfgYXvm6i5IgFylHGK5sP/xHAbB7N49OS4gWNyQ==", + "dev": true + } + } + }, + "postcss-minify-gradients": { + "version": "4.0.2", + "resolved": "https://registry.npmjs.org/postcss-minify-gradients/-/postcss-minify-gradients-4.0.2.tgz", + "integrity": "sha512-qKPfwlONdcf/AndP1U8SJ/uzIJtowHlMaSioKzebAXSG4iJthlWC9iSWznQcX4f66gIWX44RSA841HTHj3wK+Q==", + "dev": true, + "requires": { + "cssnano-util-get-arguments": "^4.0.0", + "is-color-stop": "^1.0.0", + "postcss": "^7.0.0", + "postcss-value-parser": "^3.0.0" + }, + "dependencies": { + "postcss-value-parser": { + "version": "3.3.1", + "resolved": "https://registry.npmjs.org/postcss-value-parser/-/postcss-value-parser-3.3.1.tgz", + "integrity": "sha512-pISE66AbVkp4fDQ7VHBwRNXzAAKJjw4Vw7nWI/+Q3vuly7SNfgYXvm6i5IgFylHGK5sP/xHAbB7N49OS4gWNyQ==", + "dev": true + } + } + }, + "postcss-minify-params": { + "version": "4.0.2", + "resolved": "https://registry.npmjs.org/postcss-minify-params/-/postcss-minify-params-4.0.2.tgz", + "integrity": "sha512-G7eWyzEx0xL4/wiBBJxJOz48zAKV2WG3iZOqVhPet/9geefm/Px5uo1fzlHu+DOjT+m0Mmiz3jkQzVHe6wxAWg==", + "dev": true, + "requires": { + "alphanum-sort": "^1.0.0", + "browserslist": "^4.0.0", + "cssnano-util-get-arguments": "^4.0.0", + "postcss": "^7.0.0", + "postcss-value-parser": "^3.0.0", + "uniqs": "^2.0.0" + }, + "dependencies": { + "postcss-value-parser": { + "version": "3.3.1", + "resolved": "https://registry.npmjs.org/postcss-value-parser/-/postcss-value-parser-3.3.1.tgz", + "integrity": "sha512-pISE66AbVkp4fDQ7VHBwRNXzAAKJjw4Vw7nWI/+Q3vuly7SNfgYXvm6i5IgFylHGK5sP/xHAbB7N49OS4gWNyQ==", + "dev": true + } + } + }, + "postcss-minify-selectors": { + "version": "4.0.2", + "resolved": "https://registry.npmjs.org/postcss-minify-selectors/-/postcss-minify-selectors-4.0.2.tgz", + "integrity": "sha512-D5S1iViljXBj9kflQo4YutWnJmwm8VvIsU1GeXJGiG9j8CIg9zs4voPMdQDUmIxetUOh60VilsNzCiAFTOqu3g==", + "dev": true, + "requires": { + "alphanum-sort": "^1.0.0", + "has": "^1.0.0", + "postcss": "^7.0.0", + "postcss-selector-parser": "^3.0.0" + }, + "dependencies": { + "postcss-selector-parser": { + "version": "3.1.2", + "resolved": "https://registry.npmjs.org/postcss-selector-parser/-/postcss-selector-parser-3.1.2.tgz", + "integrity": "sha512-h7fJ/5uWuRVyOtkO45pnt1Ih40CEleeyCHzipqAZO2e5H20g25Y48uYnFUiShvY4rZWNJ/Bib/KVPmanaCtOhA==", + "dev": true, + "requires": { + "dot-prop": "^5.2.0", + "indexes-of": "^1.0.1", + "uniq": "^1.0.1" + } + } + } + }, + "postcss-modules-extract-imports": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/postcss-modules-extract-imports/-/postcss-modules-extract-imports-2.0.0.tgz", + "integrity": "sha512-LaYLDNS4SG8Q5WAWqIJgdHPJrDDr/Lv775rMBFUbgjTz6j34lUznACHcdRWroPvXANP2Vj7yNK57vp9eFqzLWQ==", + "dev": true, + "requires": { + "postcss": "^7.0.5" + } + }, + "postcss-modules-local-by-default": { + "version": "3.0.3", + "resolved": "https://registry.npmjs.org/postcss-modules-local-by-default/-/postcss-modules-local-by-default-3.0.3.tgz", + "integrity": "sha512-e3xDq+LotiGesympRlKNgaJ0PCzoUIdpH0dj47iWAui/kyTgh3CiAr1qP54uodmJhl6p9rN6BoNcdEDVJx9RDw==", + "dev": true, + "requires": { + "icss-utils": "^4.1.1", + "postcss": "^7.0.32", + "postcss-selector-parser": "^6.0.2", + "postcss-value-parser": "^4.1.0" + } + }, + "postcss-modules-scope": { + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/postcss-modules-scope/-/postcss-modules-scope-2.2.0.tgz", + "integrity": "sha512-YyEgsTMRpNd+HmyC7H/mh3y+MeFWevy7V1evVhJWewmMbjDHIbZbOXICC2y+m1xI1UVfIT1HMW/O04Hxyu9oXQ==", + "dev": true, + "requires": { + "postcss": "^7.0.6", + "postcss-selector-parser": "^6.0.0" + } + }, + "postcss-modules-values": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/postcss-modules-values/-/postcss-modules-values-3.0.0.tgz", + "integrity": "sha512-1//E5jCBrZ9DmRX+zCtmQtRSV6PV42Ix7Bzj9GbwJceduuf7IqP8MgeTXuRDHOWj2m0VzZD5+roFWDuU8RQjcg==", + "dev": true, + "requires": { + "icss-utils": "^4.0.0", + "postcss": "^7.0.6" + } + }, + "postcss-normalize-charset": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/postcss-normalize-charset/-/postcss-normalize-charset-4.0.1.tgz", + "integrity": "sha512-gMXCrrlWh6G27U0hF3vNvR3w8I1s2wOBILvA87iNXaPvSNo5uZAMYsZG7XjCUf1eVxuPfyL4TJ7++SGZLc9A3g==", + "dev": true, + "requires": { + "postcss": "^7.0.0" + } + }, + "postcss-normalize-display-values": { + "version": "4.0.2", + "resolved": "https://registry.npmjs.org/postcss-normalize-display-values/-/postcss-normalize-display-values-4.0.2.tgz", + "integrity": "sha512-3F2jcsaMW7+VtRMAqf/3m4cPFhPD3EFRgNs18u+k3lTJJlVe7d0YPO+bnwqo2xg8YiRpDXJI2u8A0wqJxMsQuQ==", + "dev": true, + "requires": { + "cssnano-util-get-match": "^4.0.0", + "postcss": "^7.0.0", + "postcss-value-parser": "^3.0.0" + }, + "dependencies": { + "postcss-value-parser": { + "version": "3.3.1", + "resolved": "https://registry.npmjs.org/postcss-value-parser/-/postcss-value-parser-3.3.1.tgz", + "integrity": "sha512-pISE66AbVkp4fDQ7VHBwRNXzAAKJjw4Vw7nWI/+Q3vuly7SNfgYXvm6i5IgFylHGK5sP/xHAbB7N49OS4gWNyQ==", + "dev": true + } + } + }, + "postcss-normalize-positions": { + "version": "4.0.2", + "resolved": "https://registry.npmjs.org/postcss-normalize-positions/-/postcss-normalize-positions-4.0.2.tgz", + "integrity": "sha512-Dlf3/9AxpxE+NF1fJxYDeggi5WwV35MXGFnnoccP/9qDtFrTArZ0D0R+iKcg5WsUd8nUYMIl8yXDCtcrT8JrdA==", + "dev": true, + "requires": { + "cssnano-util-get-arguments": "^4.0.0", + "has": "^1.0.0", + "postcss": "^7.0.0", + "postcss-value-parser": "^3.0.0" + }, + "dependencies": { + "postcss-value-parser": { + "version": "3.3.1", + "resolved": "https://registry.npmjs.org/postcss-value-parser/-/postcss-value-parser-3.3.1.tgz", + "integrity": "sha512-pISE66AbVkp4fDQ7VHBwRNXzAAKJjw4Vw7nWI/+Q3vuly7SNfgYXvm6i5IgFylHGK5sP/xHAbB7N49OS4gWNyQ==", + "dev": true + } + } + }, + "postcss-normalize-repeat-style": { + "version": "4.0.2", + "resolved": "https://registry.npmjs.org/postcss-normalize-repeat-style/-/postcss-normalize-repeat-style-4.0.2.tgz", + "integrity": "sha512-qvigdYYMpSuoFs3Is/f5nHdRLJN/ITA7huIoCyqqENJe9PvPmLhNLMu7QTjPdtnVf6OcYYO5SHonx4+fbJE1+Q==", + "dev": true, + "requires": { + "cssnano-util-get-arguments": "^4.0.0", + "cssnano-util-get-match": "^4.0.0", + "postcss": "^7.0.0", + "postcss-value-parser": "^3.0.0" + }, + "dependencies": { + "postcss-value-parser": { + "version": "3.3.1", + "resolved": "https://registry.npmjs.org/postcss-value-parser/-/postcss-value-parser-3.3.1.tgz", + "integrity": "sha512-pISE66AbVkp4fDQ7VHBwRNXzAAKJjw4Vw7nWI/+Q3vuly7SNfgYXvm6i5IgFylHGK5sP/xHAbB7N49OS4gWNyQ==", + "dev": true + } + } + }, + "postcss-normalize-string": { + "version": "4.0.2", + "resolved": "https://registry.npmjs.org/postcss-normalize-string/-/postcss-normalize-string-4.0.2.tgz", + "integrity": "sha512-RrERod97Dnwqq49WNz8qo66ps0swYZDSb6rM57kN2J+aoyEAJfZ6bMx0sx/F9TIEX0xthPGCmeyiam/jXif0eA==", + "dev": true, + "requires": { + "has": "^1.0.0", + "postcss": "^7.0.0", + "postcss-value-parser": "^3.0.0" + }, + "dependencies": { + "postcss-value-parser": { + "version": "3.3.1", + "resolved": "https://registry.npmjs.org/postcss-value-parser/-/postcss-value-parser-3.3.1.tgz", + "integrity": "sha512-pISE66AbVkp4fDQ7VHBwRNXzAAKJjw4Vw7nWI/+Q3vuly7SNfgYXvm6i5IgFylHGK5sP/xHAbB7N49OS4gWNyQ==", + "dev": true + } + } + }, + "postcss-normalize-timing-functions": { + "version": "4.0.2", + "resolved": "https://registry.npmjs.org/postcss-normalize-timing-functions/-/postcss-normalize-timing-functions-4.0.2.tgz", + "integrity": "sha512-acwJY95edP762e++00Ehq9L4sZCEcOPyaHwoaFOhIwWCDfik6YvqsYNxckee65JHLKzuNSSmAdxwD2Cud1Z54A==", + "dev": true, + "requires": { + "cssnano-util-get-match": "^4.0.0", + "postcss": "^7.0.0", + "postcss-value-parser": "^3.0.0" + }, + "dependencies": { + "postcss-value-parser": { + "version": "3.3.1", + "resolved": "https://registry.npmjs.org/postcss-value-parser/-/postcss-value-parser-3.3.1.tgz", + "integrity": "sha512-pISE66AbVkp4fDQ7VHBwRNXzAAKJjw4Vw7nWI/+Q3vuly7SNfgYXvm6i5IgFylHGK5sP/xHAbB7N49OS4gWNyQ==", + "dev": true + } + } + }, + "postcss-normalize-unicode": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/postcss-normalize-unicode/-/postcss-normalize-unicode-4.0.1.tgz", + "integrity": "sha512-od18Uq2wCYn+vZ/qCOeutvHjB5jm57ToxRaMeNuf0nWVHaP9Hua56QyMF6fs/4FSUnVIw0CBPsU0K4LnBPwYwg==", + "dev": true, + "requires": { + "browserslist": "^4.0.0", + "postcss": "^7.0.0", + "postcss-value-parser": "^3.0.0" + }, + "dependencies": { + "postcss-value-parser": { + "version": "3.3.1", + "resolved": "https://registry.npmjs.org/postcss-value-parser/-/postcss-value-parser-3.3.1.tgz", + "integrity": "sha512-pISE66AbVkp4fDQ7VHBwRNXzAAKJjw4Vw7nWI/+Q3vuly7SNfgYXvm6i5IgFylHGK5sP/xHAbB7N49OS4gWNyQ==", + "dev": true + } + } + }, + "postcss-normalize-url": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/postcss-normalize-url/-/postcss-normalize-url-4.0.1.tgz", + "integrity": "sha512-p5oVaF4+IHwu7VpMan/SSpmpYxcJMtkGppYf0VbdH5B6hN8YNmVyJLuY9FmLQTzY3fag5ESUUHDqM+heid0UVA==", + "dev": true, + "requires": { + "is-absolute-url": "^2.0.0", + "normalize-url": "^3.0.0", + "postcss": "^7.0.0", + "postcss-value-parser": "^3.0.0" + }, + "dependencies": { + "normalize-url": { + "version": "3.3.0", + "resolved": "https://registry.npmjs.org/normalize-url/-/normalize-url-3.3.0.tgz", + "integrity": "sha512-U+JJi7duF1o+u2pynbp2zXDW2/PADgC30f0GsHZtRh+HOcXHnw137TrNlyxxRvWW5fjKd3bcLHPxofWuCjaeZg==", + "dev": true + }, + "postcss-value-parser": { + "version": "3.3.1", + "resolved": "https://registry.npmjs.org/postcss-value-parser/-/postcss-value-parser-3.3.1.tgz", + "integrity": "sha512-pISE66AbVkp4fDQ7VHBwRNXzAAKJjw4Vw7nWI/+Q3vuly7SNfgYXvm6i5IgFylHGK5sP/xHAbB7N49OS4gWNyQ==", + "dev": true + } + } + }, + "postcss-normalize-whitespace": { + "version": "4.0.2", + "resolved": "https://registry.npmjs.org/postcss-normalize-whitespace/-/postcss-normalize-whitespace-4.0.2.tgz", + "integrity": "sha512-tO8QIgrsI3p95r8fyqKV+ufKlSHh9hMJqACqbv2XknufqEDhDvbguXGBBqxw9nsQoXWf0qOqppziKJKHMD4GtA==", + "dev": true, + "requires": { + "postcss": "^7.0.0", + "postcss-value-parser": "^3.0.0" + }, + "dependencies": { + "postcss-value-parser": { + "version": "3.3.1", + "resolved": "https://registry.npmjs.org/postcss-value-parser/-/postcss-value-parser-3.3.1.tgz", + "integrity": "sha512-pISE66AbVkp4fDQ7VHBwRNXzAAKJjw4Vw7nWI/+Q3vuly7SNfgYXvm6i5IgFylHGK5sP/xHAbB7N49OS4gWNyQ==", + "dev": true + } + } + }, + "postcss-ordered-values": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/postcss-ordered-values/-/postcss-ordered-values-4.1.2.tgz", + "integrity": "sha512-2fCObh5UanxvSxeXrtLtlwVThBvHn6MQcu4ksNT2tsaV2Fg76R2CV98W7wNSlX+5/pFwEyaDwKLLoEV7uRybAw==", + "dev": true, + "requires": { + "cssnano-util-get-arguments": "^4.0.0", + "postcss": "^7.0.0", + "postcss-value-parser": "^3.0.0" + }, + "dependencies": { + "postcss-value-parser": { + "version": "3.3.1", + "resolved": "https://registry.npmjs.org/postcss-value-parser/-/postcss-value-parser-3.3.1.tgz", + "integrity": "sha512-pISE66AbVkp4fDQ7VHBwRNXzAAKJjw4Vw7nWI/+Q3vuly7SNfgYXvm6i5IgFylHGK5sP/xHAbB7N49OS4gWNyQ==", + "dev": true + } + } + }, + "postcss-reduce-initial": { + "version": "4.0.3", + "resolved": "https://registry.npmjs.org/postcss-reduce-initial/-/postcss-reduce-initial-4.0.3.tgz", + "integrity": "sha512-gKWmR5aUulSjbzOfD9AlJiHCGH6AEVLaM0AV+aSioxUDd16qXP1PCh8d1/BGVvpdWn8k/HiK7n6TjeoXN1F7DA==", + "dev": true, + "requires": { + "browserslist": "^4.0.0", + "caniuse-api": "^3.0.0", + "has": "^1.0.0", + "postcss": "^7.0.0" + } + }, + "postcss-reduce-transforms": { + "version": "4.0.2", + "resolved": "https://registry.npmjs.org/postcss-reduce-transforms/-/postcss-reduce-transforms-4.0.2.tgz", + "integrity": "sha512-EEVig1Q2QJ4ELpJXMZR8Vt5DQx8/mo+dGWSR7vWXqcob2gQLyQGsionYcGKATXvQzMPn6DSN1vTN7yFximdIAg==", + "dev": true, + "requires": { + "cssnano-util-get-match": "^4.0.0", + "has": "^1.0.0", + "postcss": "^7.0.0", + "postcss-value-parser": "^3.0.0" + }, + "dependencies": { + "postcss-value-parser": { + "version": "3.3.1", + "resolved": "https://registry.npmjs.org/postcss-value-parser/-/postcss-value-parser-3.3.1.tgz", + "integrity": "sha512-pISE66AbVkp4fDQ7VHBwRNXzAAKJjw4Vw7nWI/+Q3vuly7SNfgYXvm6i5IgFylHGK5sP/xHAbB7N49OS4gWNyQ==", + "dev": true + } + } + }, + "postcss-selector-parser": { + "version": "6.0.4", + "resolved": "https://registry.npmjs.org/postcss-selector-parser/-/postcss-selector-parser-6.0.4.tgz", + "integrity": "sha512-gjMeXBempyInaBqpp8gODmwZ52WaYsVOsfr4L4lDQ7n3ncD6mEyySiDtgzCT+NYC0mmeOLvtsF8iaEf0YT6dBw==", + "dev": true, + "requires": { + "cssesc": "^3.0.0", + "indexes-of": "^1.0.1", + "uniq": "^1.0.1", + "util-deprecate": "^1.0.2" + } + }, + "postcss-svgo": { + "version": "4.0.2", + "resolved": "https://registry.npmjs.org/postcss-svgo/-/postcss-svgo-4.0.2.tgz", + "integrity": "sha512-C6wyjo3VwFm0QgBy+Fu7gCYOkCmgmClghO+pjcxvrcBKtiKt0uCF+hvbMO1fyv5BMImRK90SMb+dwUnfbGd+jw==", + "dev": true, + "requires": { + "is-svg": "^3.0.0", + "postcss": "^7.0.0", + "postcss-value-parser": "^3.0.0", + "svgo": "^1.0.0" + }, + "dependencies": { + "postcss-value-parser": { + "version": "3.3.1", + "resolved": "https://registry.npmjs.org/postcss-value-parser/-/postcss-value-parser-3.3.1.tgz", + "integrity": "sha512-pISE66AbVkp4fDQ7VHBwRNXzAAKJjw4Vw7nWI/+Q3vuly7SNfgYXvm6i5IgFylHGK5sP/xHAbB7N49OS4gWNyQ==", + "dev": true + } + } + }, + "postcss-unique-selectors": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/postcss-unique-selectors/-/postcss-unique-selectors-4.0.1.tgz", + "integrity": "sha512-+JanVaryLo9QwZjKrmJgkI4Fn8SBgRO6WXQBJi7KiAVPlmxikB5Jzc4EvXMT2H0/m0RjrVVm9rGNhZddm/8Spg==", + "dev": true, + "requires": { + "alphanum-sort": "^1.0.0", + "postcss": "^7.0.0", + "uniqs": "^2.0.0" + } + }, + "postcss-value-parser": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/postcss-value-parser/-/postcss-value-parser-4.1.0.tgz", + "integrity": "sha512-97DXOFbQJhk71ne5/Mt6cOu6yxsSfM0QGQyl0L25Gca4yGWEGJaig7l7gbCX623VqTBNGLRLaVUCnNkcedlRSQ==", + "dev": true + }, + "prelude-ls": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/prelude-ls/-/prelude-ls-1.1.2.tgz", + "integrity": "sha1-IZMqVJ9eUv/ZqCf1cOBL5iqX2lQ=", + "dev": true + }, + "prepend-http": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/prepend-http/-/prepend-http-1.0.4.tgz", + "integrity": "sha1-1PRWKwzjaW5BrFLQ4ALlemNdxtw=", + "dev": true + }, + "prettier": { + "version": "2.1.2", + "resolved": "https://registry.npmjs.org/prettier/-/prettier-2.1.2.tgz", + "integrity": "sha512-16c7K+x4qVlJg9rEbXl7HEGmQyZlG4R9AgP+oHKRMsMsuk8s+ATStlf1NpDqyBI1HpVyfjLOeMhH2LvuNvV5Vg==", + "dev": true + }, + "prettier-plugin-java": { + "version": "0.8.2", + "resolved": "https://registry.npmjs.org/prettier-plugin-java/-/prettier-plugin-java-0.8.2.tgz", + "integrity": "sha512-G11PMutlfhUjLqmrB0gW9n6hTIvlnXX+lhzgH3Uhc1QSlDQoLBky5lNSRgvdYSDPpSF3DmyYd3/7U3Ez8n0ckg==", + "dev": true, + "requires": { + "java-parser": "0.8.2", + "lodash": "4.17.20", + "prettier": "2.1.1" + }, + "dependencies": { + "prettier": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/prettier/-/prettier-2.1.1.tgz", + "integrity": "sha512-9bY+5ZWCfqj3ghYBLxApy2zf6m+NJo5GzmLTpr9FsApsfjriNnS2dahWReHMi7qNPhhHl9SYHJs2cHZLgexNIw==", + "dev": true + } + } + }, + "pretty-bytes": { + "version": "5.4.1", + "resolved": "https://registry.npmjs.org/pretty-bytes/-/pretty-bytes-5.4.1.tgz", + "integrity": "sha512-s1Iam6Gwz3JI5Hweaz4GoCD1WUNUIyzePFy5+Js2hjwGVt2Z79wNN+ZKOZ2vB6C+Xs6njyB84Z1IthQg8d9LxA==", + "dev": true + }, + "pretty-error": { + "version": "2.1.2", + "resolved": "https://registry.npmjs.org/pretty-error/-/pretty-error-2.1.2.tgz", + "integrity": "sha512-EY5oDzmsX5wvuynAByrmY0P0hcp+QpnAKbJng2A2MPjVKXCxrDSUkzghVJ4ZGPIv+JC4gX8fPUWscC0RtjsWGw==", + "dev": true, + "requires": { + "lodash": "^4.17.20", + "renderkid": "^2.0.4" + } + }, + "pretty-format": { + "version": "25.5.0", + "resolved": "https://registry.npmjs.org/pretty-format/-/pretty-format-25.5.0.tgz", + "integrity": "sha512-kbo/kq2LQ/A/is0PQwsEHM7Ca6//bGPPvU6UnsdDRSKTWxT/ru/xb88v4BJf6a69H+uTytOEsTusT9ksd/1iWQ==", + "dev": true, + "requires": { + "@jest/types": "^25.5.0", + "ansi-regex": "^5.0.0", + "ansi-styles": "^4.0.0", + "react-is": "^16.12.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" + } + }, + "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 + } + } + }, + "process": { + "version": "0.11.10", + "resolved": "https://registry.npmjs.org/process/-/process-0.11.10.tgz", + "integrity": "sha1-czIwDoQBYb2j5podHZGn1LwW8YI=", + "dev": true + }, + "process-nextick-args": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/process-nextick-args/-/process-nextick-args-2.0.1.tgz", + "integrity": "sha512-3ouUOpQhtgrbOa17J7+uxOTpITYWaGP7/AhoR3+A+/1e9skrzelGi/dXzEYyvbxubEF6Wn2ypscTKiKJFFn1ag==", + "dev": true + }, + "progress": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/progress/-/progress-2.0.3.tgz", + "integrity": "sha512-7PiHtLll5LdnKIMw100I+8xJXR5gW2QwWYkT6iJva0bXitZKa/XMrSbdmg3r2Xnaidz9Qumd0VPaMrZlF9V9sA==", + "dev": true + }, + "promise": { + "version": "8.1.0", + "resolved": "https://registry.npmjs.org/promise/-/promise-8.1.0.tgz", + "integrity": "sha512-W04AqnILOL/sPRXziNicCjSNRruLAuIHEOVBazepu0545DDNGYHz7ar9ZgZ1fMU8/MA4mVxp5rkBWRi6OXIy3Q==", + "dev": true, + "requires": { + "asap": "~2.0.6" + } + }, + "promise-inflight": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/promise-inflight/-/promise-inflight-1.0.1.tgz", + "integrity": "sha1-mEcocL8igTL8vdhoEputEsPAKeM=", + "dev": true + }, + "promise-retry": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/promise-retry/-/promise-retry-1.1.1.tgz", + "integrity": "sha1-ZznpaOMFHaIM5kl/srUPaRHfPW0=", + "dev": true, + "requires": { + "err-code": "^1.0.0", + "retry": "^0.10.0" + } + }, + "prompts": { + "version": "2.4.0", + "resolved": "https://registry.npmjs.org/prompts/-/prompts-2.4.0.tgz", + "integrity": "sha512-awZAKrk3vN6CroQukBL+R9051a4R3zCZBlJm/HBfrSZ8iTpYix3VX1vU4mveiLpiwmOJT4wokTF9m6HUk4KqWQ==", + "dev": true, + "requires": { + "kleur": "^3.0.3", + "sisteransi": "^1.0.5" + } + }, + "property-expr": { + "version": "1.5.1", + "resolved": "https://registry.npmjs.org/property-expr/-/property-expr-1.5.1.tgz", + "integrity": "sha512-CGuc0VUTGthpJXL36ydB6jnbyOf/rAHFvmVrJlH+Rg0DqqLFQGAP6hIaxD/G0OAmBJPhXDHuEJigrp0e0wFV6g==", + "dev": true + }, + "protoduck": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/protoduck/-/protoduck-5.0.1.tgz", + "integrity": "sha512-WxoCeDCoCBY55BMvj4cAEjdVUFGRWed9ZxPlqTKYyw1nDDTQ4pqmnIMAGfJlg7Dx35uB/M+PHJPTmGOvaCaPTg==", + "dev": true, + "requires": { + "genfun": "^5.0.0" + } + }, + "proxy-addr": { + "version": "2.0.6", + "resolved": "https://registry.npmjs.org/proxy-addr/-/proxy-addr-2.0.6.tgz", + "integrity": "sha512-dh/frvCBVmSsDYzw6n926jv974gddhkFPfiN8hPOi30Wax25QZyZEGveluCgliBnqmuM+UJmBErbAUFIoDbjOw==", + "dev": true, + "requires": { + "forwarded": "~0.1.2", + "ipaddr.js": "1.9.1" + } + }, + "prr": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/prr/-/prr-1.0.1.tgz", + "integrity": "sha1-0/wRS6BplaRexok/SEzrHXj19HY=", + "dev": true + }, + "psl": { + "version": "1.8.0", + "resolved": "https://registry.npmjs.org/psl/-/psl-1.8.0.tgz", + "integrity": "sha512-RIdOzyoavK+hA18OGGWDqUTsCLhtA7IcZ/6NCs4fFJaHBDab+pDDmDIByWFRQJq2Cd7r1OoQxBGKOaztq+hjIQ==", + "dev": true + }, + "public-encrypt": { + "version": "4.0.3", + "resolved": "https://registry.npmjs.org/public-encrypt/-/public-encrypt-4.0.3.tgz", + "integrity": "sha512-zVpa8oKZSz5bTMTFClc1fQOnyyEzpl5ozpi1B5YcvBrdohMjH2rfsBtyXcuNuwjsDIXmBYlF2N5FlJYhR29t8Q==", + "dev": true, + "requires": { + "bn.js": "^4.1.0", + "browserify-rsa": "^4.0.0", + "create-hash": "^1.1.0", + "parse-asn1": "^5.0.0", + "randombytes": "^2.0.1", + "safe-buffer": "^5.1.2" + }, + "dependencies": { + "bn.js": { + "version": "4.11.9", + "resolved": "https://registry.npmjs.org/bn.js/-/bn.js-4.11.9.tgz", + "integrity": "sha512-E6QoYqCKZfgatHTdHzs1RRKP7ip4vvm+EyRUeE2RF0NblwVvb0p6jSVeNTOFxPn26QXN2o6SMfNxKp6kU8zQaw==", + "dev": true + } + } + }, + "pump": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/pump/-/pump-3.0.0.tgz", + "integrity": "sha512-LwZy+p3SFs1Pytd/jYct4wpv49HiYCqd9Rlc5ZVdk0V+8Yzv6jR5Blk3TRmPL1ft69TxP0IMZGJ+WPFU2BFhww==", + "dev": true, + "requires": { + "end-of-stream": "^1.1.0", + "once": "^1.3.1" + } + }, + "pumpify": { + "version": "1.5.1", + "resolved": "https://registry.npmjs.org/pumpify/-/pumpify-1.5.1.tgz", + "integrity": "sha512-oClZI37HvuUJJxSKKrC17bZ9Cu0ZYhEAGPsPUy9KlMUmv9dKX2o77RUmq7f3XjIxbwyGwYzbzQ1L2Ks8sIradQ==", + "dev": true, + "requires": { + "duplexify": "^3.6.0", + "inherits": "^2.0.3", + "pump": "^2.0.0" + }, + "dependencies": { + "pump": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/pump/-/pump-2.0.1.tgz", + "integrity": "sha512-ruPMNRkN3MHP1cWJc9OWr+T/xDP0jhXYCLfJcBuX54hhfIBnaQmAUMfDcG4DM5UMWByBbJY69QSphm3jtDKIkA==", + "dev": true, + "requires": { + "end-of-stream": "^1.1.0", + "once": "^1.3.1" + } + } + } + }, + "punycode": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/punycode/-/punycode-2.1.1.tgz", + "integrity": "sha512-XRsRjdf+j5ml+y/6GKHPZbrF/8p2Yga0JPtdqTIY2Xe5ohJPD9saDJJLPvp9+NSBprVvevdXZybnj2cv8OEd0A==", + "dev": true + }, + "q": { + "version": "1.5.1", + "resolved": "https://registry.npmjs.org/q/-/q-1.5.1.tgz", + "integrity": "sha1-fjL3W0E4EpHQRhHxvxQQmsAGUdc=", + "dev": true + }, + "qs": { + "version": "6.5.2", + "resolved": "https://registry.npmjs.org/qs/-/qs-6.5.2.tgz", + "integrity": "sha512-N5ZAX4/LxJmF+7wN74pUD6qAh9/wnvdQcjq9TZjevvXzSUo7bfmw91saqMjzGS2xq91/odN2dW/WOl7qQHNDGA==", + "dev": true + }, + "query-string": { + "version": "4.3.4", + "resolved": "https://registry.npmjs.org/query-string/-/query-string-4.3.4.tgz", + "integrity": "sha1-u7aTucqRXCMlFbIosaArYJBD2+s=", + "dev": true, + "requires": { + "object-assign": "^4.1.0", + "strict-uri-encode": "^1.0.0" + } + }, + "querystring": { + "version": "0.2.0", + "resolved": "https://registry.npmjs.org/querystring/-/querystring-0.2.0.tgz", + "integrity": "sha1-sgmEkgO7Jd+CDadW50cAWHhSFiA=", + "dev": true + }, + "querystring-es3": { + "version": "0.2.1", + "resolved": "https://registry.npmjs.org/querystring-es3/-/querystring-es3-0.2.1.tgz", + "integrity": "sha1-nsYfeQSYdXB9aUFFlv2Qek1xHnM=", + "dev": true + }, + "querystringify": { + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/querystringify/-/querystringify-2.2.0.tgz", + "integrity": "sha512-FIqgj2EUvTa7R50u0rGsyTftzjYmv/a3hO345bZNrqabNqjtgiDMgmo4mkUjd+nzU5oF3dClKqFIPUKybUyqoQ==", + "dev": true + }, + "quick-lru": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/quick-lru/-/quick-lru-4.0.1.tgz", + "integrity": "sha512-ARhCpm70fzdcvNQfPoy49IaanKkTlRWF2JMzqhcJbhSFRZv7nPTvZJdcY7301IPmvW+/p0RgIWnQDLJxifsQ7g==", + "dev": true + }, + "randexp": { + "version": "0.5.3", + "resolved": "https://registry.npmjs.org/randexp/-/randexp-0.5.3.tgz", + "integrity": "sha512-U+5l2KrcMNOUPYvazA3h5ekF80FHTUG+87SEAmHZmolh1M+i/WyTCxVzmi+tidIa1tM4BSe8g2Y/D3loWDjj+w==", + "dev": true, + "requires": { + "drange": "^1.0.2", + "ret": "^0.2.0" + }, + "dependencies": { + "ret": { + "version": "0.2.2", + "resolved": "https://registry.npmjs.org/ret/-/ret-0.2.2.tgz", + "integrity": "sha512-M0b3YWQs7R3Z917WRQy1HHA7Ba7D8hvZg6UE5mLykJxQVE2ju0IXbGlaHPPlkY+WN7wFP+wUMXmBFA0aV6vYGQ==", + "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" + } + }, + "randomfill": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/randomfill/-/randomfill-1.0.4.tgz", + "integrity": "sha512-87lcbR8+MhcWcUiQ+9e+Rwx8MyR2P7qnt15ynUlbm3TU/fjbgz4GsvfSUDTemtCCtVCqb4ZcEFlyPNTh9bBTLw==", + "dev": true, + "requires": { + "randombytes": "^2.0.5", + "safe-buffer": "^5.1.0" + } + }, + "range-parser": { + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/range-parser/-/range-parser-1.2.1.tgz", + "integrity": "sha512-Hrgsx+orqoygnmhFbKaHE6c296J+HTAQXoxEF6gNupROmmGJRoyzfG3ccAveqCBrwr/2yxQ5BVd/GTl5agOwSg==", + "dev": true + }, + "raw-body": { + "version": "2.4.1", + "resolved": "https://registry.npmjs.org/raw-body/-/raw-body-2.4.1.tgz", + "integrity": "sha512-9WmIKF6mkvA0SLmA2Knm9+qj89e+j1zqgyn8aXGd7+nAduPoqgI9lO57SAZNn/Byzo5P7JhXTyg9PzaJbH73bA==", + "dev": true, + "requires": { + "bytes": "3.1.0", + "http-errors": "1.7.3", + "iconv-lite": "0.4.24", + "unpipe": "1.0.0" + }, + "dependencies": { + "iconv-lite": { + "version": "0.4.24", + "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.4.24.tgz", + "integrity": "sha512-v3MXnZAcvnywkTUEZomIActle7RXXeedOR31wwl7VlyoXO4Qi9arvSenNQWne1TcRwhCL1HwLI21bEqdpj8/rA==", + "dev": true, + "requires": { + "safer-buffer": ">= 2.1.2 < 3" + } + } + } + }, + "react-is": { + "version": "16.13.1", + "resolved": "https://registry.npmjs.org/react-is/-/react-is-16.13.1.tgz", + "integrity": "sha512-24e6ynE2H+OKt4kqsOvNd8kBpV65zoxbA4BVsEOB3ARVWQki/DHzaUoC5KuON/BiccDaCCTZBuOcfZs70kR8bQ==", + "dev": true + }, + "read-chunk": { + "version": "3.2.0", + "resolved": "https://registry.npmjs.org/read-chunk/-/read-chunk-3.2.0.tgz", + "integrity": "sha512-CEjy9LCzhmD7nUpJ1oVOE6s/hBkejlcJEgLQHVnQznOSilOPb+kpKktlLfFDK3/WP43+F80xkUTM2VOkYoSYvQ==", + "dev": true, + "requires": { + "pify": "^4.0.1", + "with-open-file": "^0.1.6" + }, + "dependencies": { + "pify": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/pify/-/pify-4.0.1.tgz", + "integrity": "sha512-uB80kBFb/tfd68bVleG9T5GGsGPjJrLAUpR5PZIrhBnIaRTQRjqdJSsIKkOP6OAIFbj7GOrcudc5pNjZ+geV2g==", + "dev": true + } + } + }, + "read-package-json": { + "version": "2.1.2", + "resolved": "https://registry.npmjs.org/read-package-json/-/read-package-json-2.1.2.tgz", + "integrity": "sha512-D1KmuLQr6ZSJS0tW8hf3WGpRlwszJOXZ3E8Yd/DNRaM5d+1wVRZdHlpGBLAuovjr28LbWvjpWkBHMxpRGGjzNA==", + "dev": true, + "requires": { + "glob": "^7.1.1", + "json-parse-even-better-errors": "^2.3.0", + "normalize-package-data": "^2.0.0", + "npm-normalize-package-bin": "^1.0.0" + } + }, + "read-package-tree": { + "version": "5.3.1", + "resolved": "https://registry.npmjs.org/read-package-tree/-/read-package-tree-5.3.1.tgz", + "integrity": "sha512-mLUDsD5JVtlZxjSlPPx1RETkNjjvQYuweKwNVt1Sn8kP5Jh44pvYuUHCp6xSVDZWbNxVxG5lyZJ921aJH61sTw==", + "dev": true, + "requires": { + "read-package-json": "^2.0.0", + "readdir-scoped-modules": "^1.0.0", + "util-promisify": "^2.1.0" + } + }, + "read-pkg": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/read-pkg/-/read-pkg-1.1.0.tgz", + "integrity": "sha1-9f+qXs0pyzHAR0vKfXVra7KePyg=", + "dev": true, + "requires": { + "load-json-file": "^1.0.0", + "normalize-package-data": "^2.3.2", + "path-type": "^1.0.0" + } + }, + "read-pkg-up": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/read-pkg-up/-/read-pkg-up-1.0.1.tgz", + "integrity": "sha1-nWPBMnbAZZGNV/ACpX9AobZD+wI=", + "dev": true, + "requires": { + "find-up": "^1.0.0", + "read-pkg": "^1.0.0" + }, + "dependencies": { + "find-up": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/find-up/-/find-up-1.1.2.tgz", + "integrity": "sha1-ay6YIrGizgpgq2TWEOzK1TyyTQ8=", + "dev": true, + "requires": { + "path-exists": "^2.0.0", + "pinkie-promise": "^2.0.0" + } + }, + "path-exists": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/path-exists/-/path-exists-2.1.0.tgz", + "integrity": "sha1-D+tsZPD8UY2adU3V77YscCJ2H0s=", + "dev": true, + "requires": { + "pinkie-promise": "^2.0.0" + } + } + } + }, + "readable-stream": { + "version": "2.3.7", + "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-2.3.7.tgz", + "integrity": "sha512-Ebho8K4jIbHAxnuxi7o42OrZgF/ZTNcsZj6nRKyUmkhLFq8CHItp/fy6hQZuZmP/n3yZ9VBUbp4zz/mX8hmYPw==", + "dev": true, + "requires": { + "core-util-is": "~1.0.0", + "inherits": "~2.0.3", + "isarray": "~1.0.0", + "process-nextick-args": "~2.0.0", + "safe-buffer": "~5.1.1", + "string_decoder": "~1.1.1", + "util-deprecate": "~1.0.1" + } + }, + "readdir-scoped-modules": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/readdir-scoped-modules/-/readdir-scoped-modules-1.1.0.tgz", + "integrity": "sha512-asaikDeqAQg7JifRsZn1NJZXo9E+VwlyCfbkZhwyISinqk5zNS6266HS5kah6P0SaQKGF6SkNnZVHUzHFYxYDw==", + "dev": true, + "requires": { + "debuglog": "^1.0.1", + "dezalgo": "^1.0.0", + "graceful-fs": "^4.1.2", + "once": "^1.3.0" + } + }, + "readdirp": { + "version": "3.5.0", + "resolved": "https://registry.npmjs.org/readdirp/-/readdirp-3.5.0.tgz", + "integrity": "sha512-cMhu7c/8rdhkHXWsY+osBhfSy0JikwpHK/5+imo+LpeasTF8ouErHrlYkwT0++njiyuDvc7OFY5T3ukvZ8qmFQ==", + "dev": true, + "requires": { + "picomatch": "^2.2.1" + } + }, + "rechoir": { + "version": "0.6.2", + "resolved": "https://registry.npmjs.org/rechoir/-/rechoir-0.6.2.tgz", + "integrity": "sha1-hSBLVNuoLVdC4oyWdW70OvUOM4Q=", + "dev": true, + "requires": { + "resolve": "^1.1.6" + } + }, + "redent": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/redent/-/redent-3.0.0.tgz", + "integrity": "sha512-6tDA8g98We0zd0GvVeMT9arEOnTw9qM03L9cJXaCjrip1OO764RDBLBfrB4cwzNGDj5OA5ioymC9GkizgWJDUg==", + "dev": true, + "requires": { + "indent-string": "^4.0.0", + "strip-indent": "^3.0.0" + } + }, + "reflect-metadata": { + "version": "0.1.13", + "resolved": "https://registry.npmjs.org/reflect-metadata/-/reflect-metadata-0.1.13.tgz", + "integrity": "sha512-Ts1Y/anZELhSsjMcU605fU9RE4Oi3p5ORujwbIKXfWa+0Zxs510Qrmrce5/Jowq3cHSZSJqBjypxmHarc+vEWg==", + "dev": true + }, + "regenerate": { + "version": "1.4.2", + "resolved": "https://registry.npmjs.org/regenerate/-/regenerate-1.4.2.tgz", + "integrity": "sha512-zrceR/XhGYU/d/opr2EKO7aRHUeiBI8qjtfHqADTwZd6Szfy16la6kqD0MIUs5z5hx6AaKa+PixpPrR289+I0A==", + "dev": true + }, + "regenerate-unicode-properties": { + "version": "8.2.0", + "resolved": "https://registry.npmjs.org/regenerate-unicode-properties/-/regenerate-unicode-properties-8.2.0.tgz", + "integrity": "sha512-F9DjY1vKLo/tPePDycuH3dn9H1OTPIkVD9Kz4LODu+F2C75mgjAJ7x/gwy6ZcSNRAAkhNlJSOHRe8k3p+K9WhA==", + "dev": true, + "requires": { + "regenerate": "^1.4.0" + } + }, + "regenerator-runtime": { + "version": "0.13.7", + "resolved": "https://registry.npmjs.org/regenerator-runtime/-/regenerator-runtime-0.13.7.tgz", + "integrity": "sha512-a54FxoJDIr27pgf7IgeQGxmqUNYrcV338lf/6gH456HZ/PhX+5BcwHXG9ajESmwe6WRO0tAzRUrRmNONWgkrew==", + "dev": true + }, + "regenerator-transform": { + "version": "0.14.5", + "resolved": "https://registry.npmjs.org/regenerator-transform/-/regenerator-transform-0.14.5.tgz", + "integrity": "sha512-eOf6vka5IO151Jfsw2NO9WpGX58W6wWmefK3I1zEGr0lOD0u8rwPaNqQL1aRxUaxLeKO3ArNh3VYg1KbaD+FFw==", + "dev": true, + "requires": { + "@babel/runtime": "^7.8.4" + } + }, + "regex-not": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/regex-not/-/regex-not-1.0.2.tgz", + "integrity": "sha512-J6SDjUgDxQj5NusnOtdFxDwN/+HWykR8GELwctJ7mdqhcyy1xEc4SRFHUXvxTp661YaVKAjfRLZ9cCqS6tn32A==", + "dev": true, + "requires": { + "extend-shallow": "^3.0.2", + "safe-regex": "^1.1.0" + } + }, + "regexp-to-ast": { + "version": "0.5.0", + "resolved": "https://registry.npmjs.org/regexp-to-ast/-/regexp-to-ast-0.5.0.tgz", + "integrity": "sha512-tlbJqcMHnPKI9zSrystikWKwHkBqu2a/Sgw01h3zFjvYrMxEDYHzzoMZnUrbIfpTFEsoRnnviOXNCzFiSc54Qw==", + "dev": true + }, + "regexp.prototype.flags": { + "version": "1.3.0", + "resolved": "https://registry.npmjs.org/regexp.prototype.flags/-/regexp.prototype.flags-1.3.0.tgz", + "integrity": "sha512-2+Q0C5g951OlYlJz6yu5/M33IcsESLlLfsyIaLJaG4FA2r4yP8MvVMJUUP/fVBkSpbbbZlS5gynbEWLipiiXiQ==", + "dev": true, + "requires": { + "define-properties": "^1.1.3", + "es-abstract": "^1.17.0-next.1" + } + }, + "regexpp": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/regexpp/-/regexpp-3.1.0.tgz", + "integrity": "sha512-ZOIzd8yVsQQA7j8GCSlPGXwg5PfmA1mrq0JP4nGhh54LaKN3xdai/vHUDu74pKwV8OxseMS65u2NImosQcSD0Q==", + "dev": true + }, + "regexpu-core": { + "version": "4.7.1", + "resolved": "https://registry.npmjs.org/regexpu-core/-/regexpu-core-4.7.1.tgz", + "integrity": "sha512-ywH2VUraA44DZQuRKzARmw6S66mr48pQVva4LBeRhcOltJ6hExvWly5ZjFLYo67xbIxb6W1q4bAGtgfEl20zfQ==", + "dev": true, + "requires": { + "regenerate": "^1.4.0", + "regenerate-unicode-properties": "^8.2.0", + "regjsgen": "^0.5.1", + "regjsparser": "^0.6.4", + "unicode-match-property-ecmascript": "^1.0.4", + "unicode-match-property-value-ecmascript": "^1.2.0" + } + }, + "regjsgen": { + "version": "0.5.2", + "resolved": "https://registry.npmjs.org/regjsgen/-/regjsgen-0.5.2.tgz", + "integrity": "sha512-OFFT3MfrH90xIW8OOSyUrk6QHD5E9JOTeGodiJeBS3J6IwlgzJMNE/1bZklWz5oTg+9dCMyEetclvCVXOPoN3A==", + "dev": true + }, + "regjsparser": { + "version": "0.6.4", + "resolved": "https://registry.npmjs.org/regjsparser/-/regjsparser-0.6.4.tgz", + "integrity": "sha512-64O87/dPDgfk8/RQqC4gkZoGyyWFIEUTTh80CU6CWuK5vkCGyekIx+oKcEIYtP/RAxSQltCZHCNu/mdd7fqlJw==", + "dev": true, + "requires": { + "jsesc": "~0.5.0" + }, + "dependencies": { + "jsesc": { + "version": "0.5.0", + "resolved": "https://registry.npmjs.org/jsesc/-/jsesc-0.5.0.tgz", + "integrity": "sha1-597mbjXW/Bb3EP6R1c9p9w8IkR0=", + "dev": true + } + } + }, + "relateurl": { + "version": "0.2.7", + "resolved": "https://registry.npmjs.org/relateurl/-/relateurl-0.2.7.tgz", + "integrity": "sha1-VNvzd+UUQKypCkzSdGANP/LYiKk=", + "dev": true + }, + "remove-trailing-separator": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/remove-trailing-separator/-/remove-trailing-separator-1.1.0.tgz", + "integrity": "sha1-wkvOKig62tW8P1jg1IJJuSN52O8=", + "dev": true + }, + "renderkid": { + "version": "2.0.4", + "resolved": "https://registry.npmjs.org/renderkid/-/renderkid-2.0.4.tgz", + "integrity": "sha512-K2eXrSOJdq+HuKzlcjOlGoOarUu5SDguDEhE7+Ah4zuOWL40j8A/oHvLlLob9PSTNvVnBd+/q0Er1QfpEuem5g==", + "dev": true, + "requires": { + "css-select": "^1.1.0", + "dom-converter": "^0.2", + "htmlparser2": "^3.3.0", + "lodash": "^4.17.20", + "strip-ansi": "^3.0.0" + }, + "dependencies": { + "ansi-regex": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-2.1.1.tgz", + "integrity": "sha1-w7M6te42DYbg5ijwRorn7yfWVN8=", + "dev": true + }, + "dom-serializer": { + "version": "0.2.2", + "resolved": "https://registry.npmjs.org/dom-serializer/-/dom-serializer-0.2.2.tgz", + "integrity": "sha512-2/xPb3ORsQ42nHYiSunXkDjPLBaEj/xTwUO4B7XCZQTRk7EBtTOPaygh10YAAh2OI1Qrp6NWfpAhzswj0ydt9g==", + "dev": true, + "requires": { + "domelementtype": "^2.0.1", + "entities": "^2.0.0" + }, + "dependencies": { + "domelementtype": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/domelementtype/-/domelementtype-2.0.2.tgz", + "integrity": "sha512-wFwTwCVebUrMgGeAwRL/NhZtHAUyT9n9yg4IMDwf10+6iCMxSkVq9MGCVEH+QZWo1nNidy8kNvwmv4zWHDTqvA==", + "dev": true + }, + "entities": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/entities/-/entities-2.1.0.tgz", + "integrity": "sha512-hCx1oky9PFrJ611mf0ifBLBRW8lUUVRlFolb5gWRfIELabBlbp9xZvrqZLZAs+NxFnbfQoeGd8wDkygjg7U85w==", + "dev": true + } + } + }, + "domelementtype": { + "version": "1.3.1", + "resolved": "https://registry.npmjs.org/domelementtype/-/domelementtype-1.3.1.tgz", + "integrity": "sha512-BSKB+TSpMpFI/HOxCNr1O8aMOTZ8hT3pM3GQ0w/mWRmkhEDSFJkkyzz4XQsBV44BChwGkrDfMyjVD0eA2aFV3w==", + "dev": true + }, + "domhandler": { + "version": "2.4.2", + "resolved": "https://registry.npmjs.org/domhandler/-/domhandler-2.4.2.tgz", + "integrity": "sha512-JiK04h0Ht5u/80fdLMCEmV4zkNh2BcoMFBmZ/91WtYZ8qVXSKjiw7fXMgFPnHcSZgOo3XdinHvmnDUeMf5R4wA==", + "dev": true, + "requires": { + "domelementtype": "1" + } + }, + "domutils": { + "version": "1.7.0", + "resolved": "https://registry.npmjs.org/domutils/-/domutils-1.7.0.tgz", + "integrity": "sha512-Lgd2XcJ/NjEw+7tFvfKxOzCYKZsdct5lczQ2ZaQY8Djz7pfAD3Gbp8ySJWtreII/vDlMVmxwa6pHmdxIYgttDg==", + "dev": true, + "requires": { + "dom-serializer": "0", + "domelementtype": "1" + } + }, + "entities": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/entities/-/entities-1.1.2.tgz", + "integrity": "sha512-f2LZMYl1Fzu7YSBKg+RoROelpOaNrcGmE9AZubeDfrCEia483oW4MI4VyFd5VNHIgQ/7qm1I0wUHK1eJnn2y2w==", + "dev": true + }, + "htmlparser2": { + "version": "3.10.1", + "resolved": "https://registry.npmjs.org/htmlparser2/-/htmlparser2-3.10.1.tgz", + "integrity": "sha512-IgieNijUMbkDovyoKObU1DUhm1iwNYE/fuifEoEHfd1oZKZDaONBSkal7Y01shxsM49R4XaMdGez3WnF9UfiCQ==", + "dev": true, + "requires": { + "domelementtype": "^1.3.1", + "domhandler": "^2.3.0", + "domutils": "^1.5.1", + "entities": "^1.1.1", + "inherits": "^2.0.1", + "readable-stream": "^3.1.1" + } + }, + "readable-stream": { + "version": "3.6.0", + "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-3.6.0.tgz", + "integrity": "sha512-BViHy7LKeTz4oNnkcLJ+lVSL6vpiFeX6/d3oSH8zCW7UxP2onchk+vTGB143xuFjHS3deTgkKoXXymXqymiIdA==", + "dev": true, + "requires": { + "inherits": "^2.0.3", + "string_decoder": "^1.1.1", + "util-deprecate": "^1.0.1" + } + }, + "strip-ansi": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-3.0.1.tgz", + "integrity": "sha1-ajhfuIU9lS1f8F0Oiq+UJ43GPc8=", + "dev": true, + "requires": { + "ansi-regex": "^2.0.0" + } + } + } + }, + "repeat-element": { + "version": "1.1.3", + "resolved": "https://registry.npmjs.org/repeat-element/-/repeat-element-1.1.3.tgz", + "integrity": "sha512-ahGq0ZnV5m5XtZLMb+vP76kcAM5nkLqk0lpqAuojSKGgQtn4eRi4ZZGm2olo2zKFH+sMsWaqOCW1dqAnOru72g==", + "dev": true + }, + "repeat-string": { + "version": "1.6.1", + "resolved": "https://registry.npmjs.org/repeat-string/-/repeat-string-1.6.1.tgz", + "integrity": "sha1-jcrkcOHIirwtYA//Sndihtp15jc=", + "dev": true + }, + "replace-ext": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/replace-ext/-/replace-ext-1.0.1.tgz", + "integrity": "sha512-yD5BHCe7quCgBph4rMQ+0KkIRKwWCrHDOX1p1Gp6HwjPM5kVoCdKGNhN7ydqqsX6lJEnQDKZ/tFMiEdQ1dvPEw==", + "dev": true + }, + "request": { + "version": "2.88.2", + "resolved": "https://registry.npmjs.org/request/-/request-2.88.2.tgz", + "integrity": "sha512-MsvtOrfG9ZcrOwAW+Qi+F6HbD0CWXEh9ou77uOb7FM2WPhwT7smM833PzanhJLsgXjN89Ir6V2PczXNnMpwKhw==", + "dev": true, + "requires": { + "aws-sign2": "~0.7.0", + "aws4": "^1.8.0", + "caseless": "~0.12.0", + "combined-stream": "~1.0.6", + "extend": "~3.0.2", + "forever-agent": "~0.6.1", + "form-data": "~2.3.2", + "har-validator": "~5.1.3", + "http-signature": "~1.2.0", + "is-typedarray": "~1.0.0", + "isstream": "~0.1.2", + "json-stringify-safe": "~5.0.1", + "mime-types": "~2.1.19", + "oauth-sign": "~0.9.0", + "performance-now": "^2.1.0", + "qs": "~6.5.2", + "safe-buffer": "^5.1.2", + "tough-cookie": "~2.5.0", + "tunnel-agent": "^0.6.0", + "uuid": "^3.3.2" + }, + "dependencies": { + "uuid": { + "version": "3.4.0", + "resolved": "https://registry.npmjs.org/uuid/-/uuid-3.4.0.tgz", + "integrity": "sha512-HjSDRw6gZE5JMggctHBcjVak08+KEVhSIiDzFnT9S9aegmp85S/bReBVTb4QTFaRNptJ9kuYaNhnbNEOkbKb/A==", + "dev": true + } + } + }, + "request-promise-core": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/request-promise-core/-/request-promise-core-1.1.4.tgz", + "integrity": "sha512-TTbAfBBRdWD7aNNOoVOBH4pN/KigV6LyapYNNlAPA8JwbovRti1E88m3sYAwsLi5ryhPKsE9APwnjFTgdUjTpw==", + "dev": true, + "requires": { + "lodash": "^4.17.19" + } + }, + "request-promise-native": { + "version": "1.0.9", + "resolved": "https://registry.npmjs.org/request-promise-native/-/request-promise-native-1.0.9.tgz", + "integrity": "sha512-wcW+sIUiWnKgNY0dqCpOZkUbF/I+YPi+f09JZIDa39Ec+q82CpSYniDp+ISgTTbKmnpJWASeJBPZmoxH84wt3g==", + "dev": true, + "requires": { + "request-promise-core": "1.1.4", + "stealthy-require": "^1.1.1", + "tough-cookie": "^2.3.3" + } + }, + "require-directory": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/require-directory/-/require-directory-2.1.1.tgz", + "integrity": "sha1-jGStX9MNqxyXbiNE/+f3kqam30I=" + }, + "require-main-filename": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/require-main-filename/-/require-main-filename-2.0.0.tgz", + "integrity": "sha512-NKN5kMDylKuldxYLSUfrbo5Tuzh4hd+2E8NPPX02mZtn1VuREQToYe/ZdlJy+J3uCpfaiGF05e7B8W0iXbQHmg==" + }, + "requires-port": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/requires-port/-/requires-port-1.0.0.tgz", + "integrity": "sha1-kl0mAdOaxIXgkc8NpcbmlNw9yv8=", + "dev": true + }, + "resolve": { + "version": "1.19.0", + "resolved": "https://registry.npmjs.org/resolve/-/resolve-1.19.0.tgz", + "integrity": "sha512-rArEXAgsBG4UgRGcynxWIWKFvh/XZCcS8UJdHhwy91zwAvCZIbcs+vAbflgBnNjYMs/i/i+/Ux6IZhML1yPvxg==", + "requires": { + "is-core-module": "^2.1.0", + "path-parse": "^1.0.6" + } + }, + "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-dir": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/resolve-dir/-/resolve-dir-1.0.1.tgz", + "integrity": "sha1-eaQGRMNivoLybv/nOcm7U4IEb0M=", + "dev": true, + "requires": { + "expand-tilde": "^2.0.0", + "global-modules": "^1.0.0" + }, + "dependencies": { + "global-modules": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/global-modules/-/global-modules-1.0.0.tgz", + "integrity": "sha512-sKzpEkf11GpOFuw0Zzjzmt4B4UZwjOcG757PPvrfhxcLFbq0wpsgpOqxpxtxFiCG4DtG93M6XRVbF2oGdev7bg==", + "dev": true, + "requires": { + "global-prefix": "^1.0.1", + "is-windows": "^1.0.1", + "resolve-dir": "^1.0.0" + } + } + } + }, + "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 + }, + "resolve-url": { + "version": "0.2.1", + "resolved": "https://registry.npmjs.org/resolve-url/-/resolve-url-0.2.1.tgz", + "integrity": "sha1-LGN/53yJOv0qZj/iGqkIAGjiBSo=", + "dev": true + }, + "resp-modifier": { + "version": "6.0.2", + "resolved": "https://registry.npmjs.org/resp-modifier/-/resp-modifier-6.0.2.tgz", + "integrity": "sha1-sSTeXE+6/LpUH0j/pzlw9KpFa08=", + "dev": true, + "requires": { + "debug": "^2.2.0", + "minimatch": "^3.0.2" + }, + "dependencies": { + "debug": { + "version": "2.6.9", + "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", + "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", + "dev": true, + "requires": { + "ms": "2.0.0" + } + }, + "ms": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", + "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=", + "dev": true + } + } + }, + "restore-cursor": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/restore-cursor/-/restore-cursor-3.1.0.tgz", + "integrity": "sha512-l+sSefzHpj5qimhFSE5a8nufZYAM3sBSVMAPtYkmC+4EH2anSGaEMXSD0izRQbu9nfyQ9y5JrVmp7E8oZrUjvA==", + "dev": true, + "requires": { + "onetime": "^5.1.0", + "signal-exit": "^3.0.2" + } + }, + "ret": { + "version": "0.1.15", + "resolved": "https://registry.npmjs.org/ret/-/ret-0.1.15.tgz", + "integrity": "sha512-TTlYpa+OL+vMMNG24xSlQGEJ3B/RzEfUlLct7b5G/ytav+wPrplCpVMFuwzXbkecJrb6IYo1iFb0S9v37754mg==", + "dev": true + }, + "retry": { + "version": "0.10.1", + "resolved": "https://registry.npmjs.org/retry/-/retry-0.10.1.tgz", + "integrity": "sha1-52OI0heZLCUnUCQdPTlW/tmNj/Q=", + "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 + }, + "rgb-regex": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/rgb-regex/-/rgb-regex-1.0.1.tgz", + "integrity": "sha1-wODWiC3w4jviVKR16O3UGRX+rrE=", + "dev": true + }, + "rgba-regex": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/rgba-regex/-/rgba-regex-1.0.0.tgz", + "integrity": "sha1-QzdOLiyglosO8VI0YLfXMP8i7rM=", + "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" + }, + "dependencies": { + "glob": { + "version": "7.1.6", + "resolved": "https://registry.npmjs.org/glob/-/glob-7.1.6.tgz", + "integrity": "sha512-LwaxwyZ72Lk7vZINtNNrywX0ZuLyStrdDtabefZKAY5ZGJhVtgdznluResxNmPitE0SAO+O26sWTHeKSI2wMBA==", + "dev": true, + "requires": { + "fs.realpath": "^1.0.0", + "inflight": "^1.0.4", + "inherits": "2", + "minimatch": "^3.0.4", + "once": "^1.3.0", + "path-is-absolute": "^1.0.0" + } + } + } + }, + "ripemd160": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/ripemd160/-/ripemd160-2.0.2.tgz", + "integrity": "sha512-ii4iagi25WusVoiC4B4lq7pbXfAp3D9v5CwfkY33vffw2+pkDjY1D8GaN7spsxvCSx8dkPqOZCEZyfxcmJG2IA==", + "dev": true, + "requires": { + "hash-base": "^3.0.0", + "inherits": "^2.0.1" + } + }, + "rollup": { + "version": "1.32.1", + "resolved": "https://registry.npmjs.org/rollup/-/rollup-1.32.1.tgz", + "integrity": "sha512-/2HA0Ec70TvQnXdzynFffkjA6XN+1e2pEv/uKS5Ulca40g2L7KuOE3riasHoNVHOsFD5KKZgDsMk1CP3Tw9s+A==", + "dev": true, + "requires": { + "@types/estree": "*", + "@types/node": "*", + "acorn": "^7.1.0" + } + }, + "rollup-plugin-babel": { + "version": "4.4.0", + "resolved": "https://registry.npmjs.org/rollup-plugin-babel/-/rollup-plugin-babel-4.4.0.tgz", + "integrity": "sha512-Lek/TYp1+7g7I+uMfJnnSJ7YWoD58ajo6Oarhlex7lvUce+RCKRuGRSgztDO3/MF/PuGKmUL5iTHKf208UNszw==", + "dev": true, + "requires": { + "@babel/helper-module-imports": "^7.0.0", + "rollup-pluginutils": "^2.8.1" + } + }, + "rollup-plugin-terser": { + "version": "5.3.1", + "resolved": "https://registry.npmjs.org/rollup-plugin-terser/-/rollup-plugin-terser-5.3.1.tgz", + "integrity": "sha512-1pkwkervMJQGFYvM9nscrUoncPwiKR/K+bHdjv6PFgRo3cgPHoRT83y2Aa3GvINj4539S15t/tpFPb775TDs6w==", + "dev": true, + "requires": { + "@babel/code-frame": "^7.5.5", + "jest-worker": "^24.9.0", + "rollup-pluginutils": "^2.8.2", + "serialize-javascript": "^4.0.0", + "terser": "^4.6.2" + }, + "dependencies": { + "jest-worker": { + "version": "24.9.0", + "resolved": "https://registry.npmjs.org/jest-worker/-/jest-worker-24.9.0.tgz", + "integrity": "sha512-51PE4haMSXcHohnSMdM42anbvZANYTqMrr52tVKPqqsPJMzoP6FYYDVqahX/HrAoKEKz3uUPzSvKs9A3qR4iVw==", + "dev": true, + "requires": { + "merge-stream": "^2.0.0", + "supports-color": "^6.1.0" + } + }, + "serialize-javascript": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/serialize-javascript/-/serialize-javascript-4.0.0.tgz", + "integrity": "sha512-GaNA54380uFefWghODBWEGisLZFj00nS5ACs6yHa9nLqlLpVLO8ChDGeKRjZnV4Nh4n0Qi7nhYZD/9fCPzEqkw==", + "dev": true, + "requires": { + "randombytes": "^2.1.0" + } + }, + "supports-color": { + "version": "6.1.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-6.1.0.tgz", + "integrity": "sha512-qe1jfm1Mg7Nq/NSh6XE24gPXROEVsWHxC1LIx//XNlD9iw7YZQGjZNjYN7xGaEG6iKdA8EtNFW6R0gjnVXp+wQ==", + "dev": true, + "requires": { + "has-flag": "^3.0.0" + } + } + } + }, + "rollup-pluginutils": { + "version": "2.8.2", + "resolved": "https://registry.npmjs.org/rollup-pluginutils/-/rollup-pluginutils-2.8.2.tgz", + "integrity": "sha512-EEp9NhnUkwY8aif6bxgovPHMoMoNr2FulJziTndpt5H9RdwC47GSGuII9XxpSdzVGM0GWrNPHV6ie1LTNJPaLQ==", + "dev": true, + "requires": { + "estree-walker": "^0.6.1" + }, + "dependencies": { + "estree-walker": { + "version": "0.6.1", + "resolved": "https://registry.npmjs.org/estree-walker/-/estree-walker-0.6.1.tgz", + "integrity": "sha512-SqmZANLWS0mnatqbSfRP5g8OXZC12Fgg1IwNtLsyHDzJizORW4khDfjPqJZsemPWBB2uqykUah5YpQ6epsqC/w==", + "dev": true + } + } + }, + "rsvp": { + "version": "4.8.5", + "resolved": "https://registry.npmjs.org/rsvp/-/rsvp-4.8.5.tgz", + "integrity": "sha512-nfMOlASu9OnRJo1mbEk2cz0D56a1MBNrJ7orjRZQG10XDyuvwksKbuXNp6qa+kbn839HwjwhBzhFmdsaEAfauA==", + "dev": true + }, + "run-async": { + "version": "2.4.1", + "resolved": "https://registry.npmjs.org/run-async/-/run-async-2.4.1.tgz", + "integrity": "sha512-tvVnVv01b8c1RrA6Ep7JkStj85Guv/YrMcwqYQnwjsAS2cTmmPGBBjAjpCW7RrSodNSoE2/qg9O4bceNvUuDgQ==", + "dev": true + }, + "run-parallel": { + "version": "1.1.10", + "resolved": "https://registry.npmjs.org/run-parallel/-/run-parallel-1.1.10.tgz", + "integrity": "sha512-zb/1OuZ6flOlH6tQyMPUrE3x3Ulxjlo9WIVXR4yVYi4H9UXQaeIsPbLn2R3O3vQCnDKkAl2qHiuocKKX4Tz/Sw==", + "dev": true + }, + "run-queue": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/run-queue/-/run-queue-1.0.3.tgz", + "integrity": "sha1-6Eg5bwV9Ij8kOGkkYY4laUFh7Ec=", + "dev": true, + "requires": { + "aproba": "^1.1.1" + } + }, + "rx": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/rx/-/rx-4.1.0.tgz", + "integrity": "sha1-pfE/957zt0D+MKqAP7CfmIBdR4I=", + "dev": true + }, + "rxjs": { + "version": "6.5.5", + "resolved": "https://registry.npmjs.org/rxjs/-/rxjs-6.5.5.tgz", + "integrity": "sha512-WfQI+1gohdf0Dai/Bbmk5L5ItH5tYqm3ki2c5GdWhKjalzjg93N3avFjVStyZZz+A2Em+ZxKH5bNghw9UeylGQ==", + "requires": { + "tslib": "^1.9.0" + }, + "dependencies": { + "tslib": { + "version": "1.14.1", + "resolved": "https://registry.npmjs.org/tslib/-/tslib-1.14.1.tgz", + "integrity": "sha512-Xni35NKzjgMrwevysHTCArtLDpPvye8zV/0E4EyYn43P7/7qvQwPh9BGkHewbMulVntbigmcT7rdX3BNo9wRJg==" + } + } + }, + "safe-buffer": { + "version": "5.1.2", + "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.2.tgz", + "integrity": "sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g==" + }, + "safe-regex": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/safe-regex/-/safe-regex-1.1.0.tgz", + "integrity": "sha1-QKNmnzsHfR6UPURinhV91IAjvy4=", + "dev": true, + "requires": { + "ret": "~0.1.10" + } + }, + "safer-buffer": { + "version": "2.1.2", + "resolved": "https://registry.npmjs.org/safer-buffer/-/safer-buffer-2.1.2.tgz", + "integrity": "sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg==", + "dev": true + }, + "sane": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/sane/-/sane-4.1.0.tgz", + "integrity": "sha512-hhbzAgTIX8O7SHfp2c8/kREfEn4qO/9q8C9beyY6+tvZ87EpoZ3i1RIEvp27YBswnNbY9mWd6paKVmKbAgLfZA==", + "dev": true, + "requires": { + "@cnakazawa/watch": "^1.0.3", + "anymatch": "^2.0.0", + "capture-exit": "^2.0.0", + "exec-sh": "^0.3.2", + "execa": "^1.0.0", + "fb-watchman": "^2.0.0", + "micromatch": "^3.1.4", + "minimist": "^1.1.1", + "walker": "~1.0.5" + }, + "dependencies": { + "anymatch": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/anymatch/-/anymatch-2.0.0.tgz", + "integrity": "sha512-5teOsQWABXHHBFP9y3skS5P3d/WfWXpv3FUpy+LorMrNYaT9pI4oLMQX7jzQ2KklNpGpWHzdCXTDT2Y3XGlZBw==", + "dev": true, + "requires": { + "micromatch": "^3.1.4", + "normalize-path": "^2.1.1" + } + }, + "normalize-path": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/normalize-path/-/normalize-path-2.1.1.tgz", + "integrity": "sha1-GrKLVW4Zg2Oowab35vogE3/mrtk=", + "dev": true, + "requires": { + "remove-trailing-separator": "^1.0.1" + } + } + } + }, + "sass": { + "version": "1.26.9", + "resolved": "https://registry.npmjs.org/sass/-/sass-1.26.9.tgz", + "integrity": "sha512-t8AkRVi+xvba4yZiLWkJdgJHBFCB3Dh4johniQkPy9ywkgFHNasXFEFP+RG/F6LhQ+aoE4aX+IorIWQjS0esVw==", + "dev": true, + "requires": { + "chokidar": ">=2.0.0 <4.0.0" + } + }, + "sass-loader": { + "version": "8.0.2", + "resolved": "https://registry.npmjs.org/sass-loader/-/sass-loader-8.0.2.tgz", + "integrity": "sha512-7o4dbSK8/Ol2KflEmSco4jTjQoV988bM82P9CZdmo9hR3RLnvNc0ufMNdMrB0caq38JQ/FgF4/7RcbcfKzxoFQ==", + "dev": true, + "requires": { + "clone-deep": "^4.0.1", + "loader-utils": "^1.2.3", + "neo-async": "^2.6.1", + "schema-utils": "^2.6.1", + "semver": "^6.3.0" + }, + "dependencies": { + "json5": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/json5/-/json5-1.0.1.tgz", + "integrity": "sha512-aKS4WQjPenRxiQsC93MNfjx+nbF4PAdYzmd/1JIj8HYzqfbu86beTuNgXDzPknWk0n0uARlyewZo4s++ES36Ow==", + "dev": true, + "requires": { + "minimist": "^1.2.0" + } + }, + "loader-utils": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/loader-utils/-/loader-utils-1.4.0.tgz", + "integrity": "sha512-qH0WSMBtn/oHuwjy/NucEgbx5dbxxnxup9s4PVXJUDHZBQY+s0NWA9rJf53RBnQZxfch7euUui7hpoAPvALZdA==", + "dev": true, + "requires": { + "big.js": "^5.2.2", + "emojis-list": "^3.0.0", + "json5": "^1.0.1" + } + }, + "semver": { + "version": "6.3.0", + "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.0.tgz", + "integrity": "sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw==", + "dev": true + } + } + }, + "sax": { + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/sax/-/sax-1.2.1.tgz", + "integrity": "sha1-e45lYZCyKOgaZq6nSEgNgozS03o=", + "dev": true + }, + "saxes": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/saxes/-/saxes-5.0.1.tgz", + "integrity": "sha512-5LBh1Tls8c9xgGjw3QrMwETmTMVk0oFgvrFSvWx62llR2hcEInrKNZ2GZCCuuy2lvWrdl5jhbpeqc5hRYKFOcw==", + "dev": true, + "requires": { + "xmlchars": "^2.2.0" + } + }, + "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" + }, + "dependencies": { + "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" + } + } + } + }, + "scoped-regex": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/scoped-regex/-/scoped-regex-1.0.0.tgz", + "integrity": "sha1-o0a7Gs1CB65wvXwMfKnlZra63bg=", + "dev": true + }, + "select-hose": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/select-hose/-/select-hose-2.0.0.tgz", + "integrity": "sha1-Yl2GWPhlr0Psliv8N2o3NZpJlMo=", + "dev": true + }, + "selfsigned": { + "version": "1.10.8", + "resolved": "https://registry.npmjs.org/selfsigned/-/selfsigned-1.10.8.tgz", + "integrity": "sha512-2P4PtieJeEwVgTU9QEcwIRDQ/mXJLX8/+I3ur+Pg16nS8oNbrGxEso9NyYWy8NAmXiNl4dlAp5MwoNeCWzON4w==", + "dev": true, + "requires": { + "node-forge": "^0.10.0" + } + }, + "semver": { + "version": "5.7.1", + "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.1.tgz", + "integrity": "sha512-sauaDf/PZdVgrLTNYHRtpXa1iRiKcaebiKQ1BJdpQlWH2lCvexQdX55snPFyK7QzpudqbCI0qXFfOasHdyNDGQ==" + }, + "semver-compare": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/semver-compare/-/semver-compare-1.0.0.tgz", + "integrity": "sha1-De4hahyUGrN+nvsXiPavxf9VN/w=", + "dev": true + }, + "semver-dsl": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/semver-dsl/-/semver-dsl-1.0.1.tgz", + "integrity": "sha1-02eN5VVeimH2Ke7QJTZq5fJzQKA=", + "dev": true, + "requires": { + "semver": "^5.3.0" + } + }, + "semver-intersect": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/semver-intersect/-/semver-intersect-1.4.0.tgz", + "integrity": "sha512-d8fvGg5ycKAq0+I6nfWeCx6ffaWJCsBYU0H2Rq56+/zFePYfT8mXkB3tWBSjR5BerkHNZ5eTPIk1/LBYas35xQ==", + "dev": true, + "requires": { + "semver": "^5.0.0" + } + }, + "semver-regex": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/semver-regex/-/semver-regex-2.0.0.tgz", + "integrity": "sha512-mUdIBBvdn0PLOeP3TEkMH7HHeUP3GjsXCwKarjv/kGmUFOYg1VqEemKhoQpWMu6X2I8kHeuVdGibLGkVK+/5Qw==", + "dev": true + }, + "send": { + "version": "0.16.2", + "resolved": "https://registry.npmjs.org/send/-/send-0.16.2.tgz", + "integrity": "sha512-E64YFPUssFHEFBvpbbjr44NCLtI1AohxQ8ZSiJjQLskAdKuriYEP6VyGEsRDH8ScozGpkaX1BGvhanqCwkcEZw==", + "dev": true, + "requires": { + "debug": "2.6.9", + "depd": "~1.1.2", + "destroy": "~1.0.4", + "encodeurl": "~1.0.2", + "escape-html": "~1.0.3", + "etag": "~1.8.1", + "fresh": "0.5.2", + "http-errors": "~1.6.2", + "mime": "1.4.1", + "ms": "2.0.0", + "on-finished": "~2.3.0", + "range-parser": "~1.2.0", + "statuses": "~1.4.0" + }, + "dependencies": { + "debug": { + "version": "2.6.9", + "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", + "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", + "dev": true, + "requires": { + "ms": "2.0.0" + } + }, + "http-errors": { + "version": "1.6.3", + "resolved": "https://registry.npmjs.org/http-errors/-/http-errors-1.6.3.tgz", + "integrity": "sha1-i1VoC7S+KDoLW/TqLjhYC+HZMg0=", + "dev": true, + "requires": { + "depd": "~1.1.2", + "inherits": "2.0.3", + "setprototypeof": "1.1.0", + "statuses": ">= 1.4.0 < 2" + } + }, + "inherits": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.3.tgz", + "integrity": "sha1-Yzwsg+PaQqUC9SRmAiSA9CCCYd4=", + "dev": true + }, + "ms": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", + "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=", + "dev": true + }, + "setprototypeof": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/setprototypeof/-/setprototypeof-1.1.0.tgz", + "integrity": "sha512-BvE/TwpZX4FXExxOxZyRGQQv651MSwmWKZGqvmPcRIjDqWub67kTKuIMx43cZZrS/cBBzwBcNDWoFxt2XEFIpQ==", + "dev": true + }, + "statuses": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/statuses/-/statuses-1.4.0.tgz", + "integrity": "sha512-zhSCtt8v2NDrRlPQpCNtw/heZLtfUDqxBM1udqikb/Hbk52LK4nQSwr10u77iopCW5LsyHpuXS0GnEc48mLeew==", + "dev": true + } + } + }, + "serialize-javascript": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/serialize-javascript/-/serialize-javascript-3.1.0.tgz", + "integrity": "sha512-JIJT1DGiWmIKhzRsG91aS6Ze4sFUrYbltlkg2onR5OrnNM02Kl/hnY/T4FN2omvyeBbQmMJv+K4cPOpGzOTFBg==", + "dev": true, + "requires": { + "randombytes": "^2.1.0" + } + }, + "serve-index": { + "version": "1.9.1", + "resolved": "https://registry.npmjs.org/serve-index/-/serve-index-1.9.1.tgz", + "integrity": "sha1-03aNabHn2C5c4FD/9bRTvqEqkjk=", + "dev": true, + "requires": { + "accepts": "~1.3.4", + "batch": "0.6.1", + "debug": "2.6.9", + "escape-html": "~1.0.3", + "http-errors": "~1.6.2", + "mime-types": "~2.1.17", + "parseurl": "~1.3.2" + }, + "dependencies": { + "debug": { + "version": "2.6.9", + "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", + "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", + "dev": true, + "requires": { + "ms": "2.0.0" + } + }, + "http-errors": { + "version": "1.6.3", + "resolved": "https://registry.npmjs.org/http-errors/-/http-errors-1.6.3.tgz", + "integrity": "sha1-i1VoC7S+KDoLW/TqLjhYC+HZMg0=", + "dev": true, + "requires": { + "depd": "~1.1.2", + "inherits": "2.0.3", + "setprototypeof": "1.1.0", + "statuses": ">= 1.4.0 < 2" + } + }, + "inherits": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.3.tgz", + "integrity": "sha1-Yzwsg+PaQqUC9SRmAiSA9CCCYd4=", + "dev": true + }, + "ms": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", + "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=", + "dev": true + }, + "setprototypeof": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/setprototypeof/-/setprototypeof-1.1.0.tgz", + "integrity": "sha512-BvE/TwpZX4FXExxOxZyRGQQv651MSwmWKZGqvmPcRIjDqWub67kTKuIMx43cZZrS/cBBzwBcNDWoFxt2XEFIpQ==", + "dev": true + }, + "statuses": { + "version": "1.5.0", + "resolved": "https://registry.npmjs.org/statuses/-/statuses-1.5.0.tgz", + "integrity": "sha1-Fhx9rBd2Wf2YEfQ3cfqZOBR4Yow=", + "dev": true + } + } + }, + "serve-static": { + "version": "1.13.2", + "resolved": "https://registry.npmjs.org/serve-static/-/serve-static-1.13.2.tgz", + "integrity": "sha512-p/tdJrO4U387R9oMjb1oj7qSMaMfmOyd4j9hOFoxZe2baQszgHcSWjuya/CiT5kgZZKRudHNOA0pYXOl8rQ5nw==", + "dev": true, + "requires": { + "encodeurl": "~1.0.2", + "escape-html": "~1.0.3", + "parseurl": "~1.3.2", + "send": "0.16.2" + } + }, + "server-destroy": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/server-destroy/-/server-destroy-1.0.1.tgz", + "integrity": "sha1-8Tv5KOQrnD55OD5hzDmYtdFObN0=", + "dev": true + }, + "set-blocking": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/set-blocking/-/set-blocking-2.0.0.tgz", + "integrity": "sha1-BF+XgtARrppoA93TgrJDkrPYkPc=" + }, + "set-getter": { + "version": "0.1.0", + "resolved": "https://registry.npmjs.org/set-getter/-/set-getter-0.1.0.tgz", + "integrity": "sha1-12nBgsnVpR9AkUXy+6guXoboA3Y=", + "dev": true, + "requires": { + "to-object-path": "^0.3.0" + } + }, + "set-value": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/set-value/-/set-value-2.0.1.tgz", + "integrity": "sha512-JxHc1weCN68wRY0fhCoXpyK55m/XPHafOmK4UWD7m2CI14GMcFypt4w/0+NV5f/ZMby2F6S2wwA7fgynh9gWSw==", + "dev": true, + "requires": { + "extend-shallow": "^2.0.1", + "is-extendable": "^0.1.1", + "is-plain-object": "^2.0.3", + "split-string": "^3.0.1" + }, + "dependencies": { + "extend-shallow": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/extend-shallow/-/extend-shallow-2.0.1.tgz", + "integrity": "sha1-Ua99YUrZqfYQ6huvu5idaxxWiQ8=", + "dev": true, + "requires": { + "is-extendable": "^0.1.0" + } + } + } + }, + "setimmediate": { + "version": "1.0.5", + "resolved": "https://registry.npmjs.org/setimmediate/-/setimmediate-1.0.5.tgz", + "integrity": "sha1-KQy7Iy4waULX1+qbg3Mqt4VvgoU=", + "dev": true + }, + "setprototypeof": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/setprototypeof/-/setprototypeof-1.1.1.tgz", + "integrity": "sha512-JvdAWfbXeIGaZ9cILp38HntZSFSo3mWg6xGcJJsd+d4aRMOqauag1C63dJfDw7OaMYwEbHMOxEZ1lqVRYP2OAw==", + "dev": true + }, + "sha.js": { + "version": "2.4.11", + "resolved": "https://registry.npmjs.org/sha.js/-/sha.js-2.4.11.tgz", + "integrity": "sha512-QMEp5B7cftE7APOjk5Y6xgrbWu+WkLVQwk8JNjZ8nKRciZaByEW6MubieAiToS7+dwvrjGhH8jRXz3MVd0AYqQ==", + "dev": true, + "requires": { + "inherits": "^2.0.1", + "safe-buffer": "^5.0.1" + } + }, + "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": "1.2.0", + "resolved": "https://registry.npmjs.org/shebang-command/-/shebang-command-1.2.0.tgz", + "integrity": "sha1-RKrGW2lbAzmJaMOfNj/uXer98eo=", + "dev": true, + "requires": { + "shebang-regex": "^1.0.0" + } + }, + "shebang-regex": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/shebang-regex/-/shebang-regex-1.0.0.tgz", + "integrity": "sha1-2kL0l0DAtC2yypcoVxyxkMmO/qM=", + "dev": true + }, + "shelljs": { + "version": "0.8.4", + "resolved": "https://registry.npmjs.org/shelljs/-/shelljs-0.8.4.tgz", + "integrity": "sha512-7gk3UZ9kOfPLIAbslLzyWeGiEqx9e3rxwZM0KE6EL8GlGwjym9Mrlx5/p33bWTu9YG6vcS4MBxYZDHYr5lr8BQ==", + "dev": true, + "requires": { + "glob": "^7.0.0", + "interpret": "^1.0.0", + "rechoir": "^0.6.2" + } + }, + "shellwords": { + "version": "0.1.1", + "resolved": "https://registry.npmjs.org/shellwords/-/shellwords-0.1.1.tgz", + "integrity": "sha512-vFwSUfQvqybiICwZY5+DAWIPLKsWO31Q91JSKl3UYv+K5c2QRPzn0qzec6QPu1Qc9eHYItiP3NdJqNVqetYAww==", + "dev": true + }, + "signal-exit": { + "version": "3.0.3", + "resolved": "https://registry.npmjs.org/signal-exit/-/signal-exit-3.0.3.tgz", + "integrity": "sha512-VUJ49FC8U1OxwZLxIbTTrDvLnf/6TDgxZcK8wxR8zs13xpx7xbG60ndBlhNrFi2EMuFRoeDoJO7wthSLq42EjA==", + "dev": true + }, + "simple-git": { + "version": "1.132.0", + "resolved": "https://registry.npmjs.org/simple-git/-/simple-git-1.132.0.tgz", + "integrity": "sha512-xauHm1YqCTom1sC9eOjfq3/9RKiUA9iPnxBbrY2DdL8l4ADMu0jjM5l5lphQP5YWNqAL2aXC/OeuQ76vHtW5fg==", + "dev": true, + "requires": { + "debug": "^4.0.1" + } + }, + "simple-progress-webpack-plugin": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/simple-progress-webpack-plugin/-/simple-progress-webpack-plugin-1.1.2.tgz", + "integrity": "sha512-bNQfb3qSqbtsfxg6d0dGechUUJH2lZqKG5+bj2aoJmEA0rSzcm+2JVfC2YgkDABfuGItZ/O5ttt6BssWZW4SNg==", + "dev": true, + "requires": { + "chalk": "2.3.x", + "figures": "2.0.x", + "log-update": "2.3.x" + }, + "dependencies": { + "chalk": { + "version": "2.3.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-2.3.2.tgz", + "integrity": "sha512-ZM4j2/ld/YZDc3Ma8PgN7gyAk+kHMMMyzLNryCPGhWrsfAuDVeuid5bpRFTDgMH9JBK2lA4dyyAkkZYF/WcqDQ==", + "dev": true, + "requires": { + "ansi-styles": "^3.2.1", + "escape-string-regexp": "^1.0.5", + "supports-color": "^5.3.0" + } + }, + "figures": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/figures/-/figures-2.0.0.tgz", + "integrity": "sha1-OrGi0qYsi/tDGgyUy3l6L84nyWI=", + "dev": true, + "requires": { + "escape-string-regexp": "^1.0.5" + } + } + } + }, + "simple-swizzle": { + "version": "0.2.2", + "resolved": "https://registry.npmjs.org/simple-swizzle/-/simple-swizzle-0.2.2.tgz", + "integrity": "sha1-pNprY1/8zMoz9w0Xy5JZLeleVXo=", + "dev": true, + "requires": { + "is-arrayish": "^0.3.1" + }, + "dependencies": { + "is-arrayish": { + "version": "0.3.2", + "resolved": "https://registry.npmjs.org/is-arrayish/-/is-arrayish-0.3.2.tgz", + "integrity": "sha512-eVRqCvVlZbuw3GrM63ovNSNAeA1K16kaR/LRY/92w0zxQ5/1YzwblUX652i4Xs9RwAGjW9d9y6X88t8OaAJfWQ==", + "dev": true + } + } + }, + "sisteransi": { + "version": "1.0.5", + "resolved": "https://registry.npmjs.org/sisteransi/-/sisteransi-1.0.5.tgz", + "integrity": "sha512-bLGGlR1QxBcynn2d5YmDX4MGjlZvy2MRBDRNHLJ8VI6l6+9FUiyTFNJ0IveOSP0bcXgVDPRcfGqA0pjaqUpfVg==", + "dev": true + }, + "slash": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/slash/-/slash-3.0.0.tgz", + "integrity": "sha512-g9Q1haeby36OSStwb4ntCGGGaKsaVSjQ68fBxoQcutl5fS1vuY18H3wSt3jFyFtrkx+Kz0V1G85A4MyAdDMi2Q==", + "dev": true + }, + "slice-ansi": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/slice-ansi/-/slice-ansi-2.1.0.tgz", + "integrity": "sha512-Qu+VC3EwYLldKa1fCxuuvULvSJOKEgk9pi8dZeCVK7TqBfUNTH4sFkk4joj8afVSfAYgJoSOetjx9QWOJ5mYoQ==", + "dev": true, + "requires": { + "ansi-styles": "^3.2.0", + "astral-regex": "^1.0.0", + "is-fullwidth-code-point": "^2.0.0" + }, + "dependencies": { + "is-fullwidth-code-point": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-2.0.0.tgz", + "integrity": "sha1-o7MKXE8ZkYMWeqq5O+764937ZU8=", + "dev": true + } + } + }, + "smart-buffer": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/smart-buffer/-/smart-buffer-4.1.0.tgz", + "integrity": "sha512-iVICrxOzCynf/SNaBQCw34eM9jROU/s5rzIhpOvzhzuYHfJR/DhZfDkXiZSgKXfgv26HT3Yni3AV/DGw0cGnnw==", + "dev": true + }, + "snapdragon": { + "version": "0.8.2", + "resolved": "https://registry.npmjs.org/snapdragon/-/snapdragon-0.8.2.tgz", + "integrity": "sha512-FtyOnWN/wCHTVXOMwvSv26d+ko5vWlIDD6zoUJ7LW8vh+ZBC8QdljveRP+crNrtBwioEUWy/4dMtbBjA4ioNlg==", + "dev": true, + "requires": { + "base": "^0.11.1", + "debug": "^2.2.0", + "define-property": "^0.2.5", + "extend-shallow": "^2.0.1", + "map-cache": "^0.2.2", + "source-map": "^0.5.6", + "source-map-resolve": "^0.5.0", + "use": "^3.1.0" + }, + "dependencies": { + "debug": { + "version": "2.6.9", + "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", + "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", + "dev": true, + "requires": { + "ms": "2.0.0" + } + }, + "define-property": { + "version": "0.2.5", + "resolved": "https://registry.npmjs.org/define-property/-/define-property-0.2.5.tgz", + "integrity": "sha1-w1se+RjsPJkPmlvFe+BKrOxcgRY=", + "dev": true, + "requires": { + "is-descriptor": "^0.1.0" + } + }, + "extend-shallow": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/extend-shallow/-/extend-shallow-2.0.1.tgz", + "integrity": "sha1-Ua99YUrZqfYQ6huvu5idaxxWiQ8=", + "dev": true, + "requires": { + "is-extendable": "^0.1.0" + } + }, + "ms": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", + "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=", + "dev": true + } + } + }, + "snapdragon-node": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/snapdragon-node/-/snapdragon-node-2.1.1.tgz", + "integrity": "sha512-O27l4xaMYt/RSQ5TR3vpWCAB5Kb/czIcqUFOM/C4fYcLnbZUc1PkjTAMjof2pBWaSTwOUd6qUHcFGVGj7aIwnw==", + "dev": true, + "requires": { + "define-property": "^1.0.0", + "isobject": "^3.0.0", + "snapdragon-util": "^3.0.1" + }, + "dependencies": { + "define-property": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/define-property/-/define-property-1.0.0.tgz", + "integrity": "sha1-dp66rz9KY6rTr56NMEybvnm/sOY=", + "dev": true, + "requires": { + "is-descriptor": "^1.0.0" + } + }, + "is-accessor-descriptor": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/is-accessor-descriptor/-/is-accessor-descriptor-1.0.0.tgz", + "integrity": "sha512-m5hnHTkcVsPfqx3AKlyttIPb7J+XykHvJP2B9bZDjlhLIoEq4XoK64Vg7boZlVWYK6LUY94dYPEE7Lh0ZkZKcQ==", + "dev": true, + "requires": { + "kind-of": "^6.0.0" + } + }, + "is-data-descriptor": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/is-data-descriptor/-/is-data-descriptor-1.0.0.tgz", + "integrity": "sha512-jbRXy1FmtAoCjQkVmIVYwuuqDFUbaOeDjmed1tOGPrsMhtJA4rD9tkgA0F1qJ3gRFRXcHYVkdeaP50Q5rE/jLQ==", + "dev": true, + "requires": { + "kind-of": "^6.0.0" + } + }, + "is-descriptor": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/is-descriptor/-/is-descriptor-1.0.2.tgz", + "integrity": "sha512-2eis5WqQGV7peooDyLmNEPUrps9+SXX5c9pL3xEB+4e9HnGuDa7mB7kHxHw4CbqS9k1T2hOH3miL8n8WtiYVtg==", + "dev": true, + "requires": { + "is-accessor-descriptor": "^1.0.0", + "is-data-descriptor": "^1.0.0", + "kind-of": "^6.0.2" + } + } + } + }, + "snapdragon-util": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/snapdragon-util/-/snapdragon-util-3.0.1.tgz", + "integrity": "sha512-mbKkMdQKsjX4BAL4bRYTj21edOf8cN7XHdYUJEe+Zn99hVEYcMvKPct1IqNe7+AZPirn8BCDOQBHQZknqmKlZQ==", + "dev": true, + "requires": { + "kind-of": "^3.2.0" + }, + "dependencies": { + "kind-of": { + "version": "3.2.2", + "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-3.2.2.tgz", + "integrity": "sha1-MeohpzS6ubuw8yRm2JOupR5KPGQ=", + "dev": true, + "requires": { + "is-buffer": "^1.1.5" + } + } + } + }, + "socket.io": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/socket.io/-/socket.io-2.1.1.tgz", + "integrity": "sha512-rORqq9c+7W0DAK3cleWNSyfv/qKXV99hV4tZe+gGLfBECw3XEhBy7x85F3wypA9688LKjtwO9pX9L33/xQI8yA==", + "dev": true, + "requires": { + "debug": "~3.1.0", + "engine.io": "~3.2.0", + "has-binary2": "~1.0.2", + "socket.io-adapter": "~1.1.0", + "socket.io-client": "2.1.1", + "socket.io-parser": "~3.2.0" + }, + "dependencies": { + "base64-arraybuffer": { + "version": "0.1.5", + "resolved": "https://registry.npmjs.org/base64-arraybuffer/-/base64-arraybuffer-0.1.5.tgz", + "integrity": "sha1-c5JncZI7Whl0etZmqlzUv5xunOg=", + "dev": true + }, + "component-emitter": { + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/component-emitter/-/component-emitter-1.2.1.tgz", + "integrity": "sha1-E3kY1teCg/ffemt8WmPhQOaUJeY=", + "dev": true + }, + "debug": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/debug/-/debug-3.1.0.tgz", + "integrity": "sha512-OX8XqP7/1a9cqkxYw2yXss15f26NKWBpDXQd0/uK/KPqdQhxbPa994hnzjcE2VqQpDslf55723cKPUOGSmMY3g==", + "dev": true, + "requires": { + "ms": "2.0.0" + } + }, + "engine.io-client": { + "version": "3.2.1", + "resolved": "https://registry.npmjs.org/engine.io-client/-/engine.io-client-3.2.1.tgz", + "integrity": "sha512-y5AbkytWeM4jQr7m/koQLc5AxpRKC1hEVUb/s1FUAWEJq5AzJJ4NLvzuKPuxtDi5Mq755WuDvZ6Iv2rXj4PTzw==", + "dev": true, + "requires": { + "component-emitter": "1.2.1", + "component-inherit": "0.0.3", + "debug": "~3.1.0", + "engine.io-parser": "~2.1.1", + "has-cors": "1.1.0", + "indexof": "0.0.1", + "parseqs": "0.0.5", + "parseuri": "0.0.5", + "ws": "~3.3.1", + "xmlhttprequest-ssl": "~1.5.4", + "yeast": "0.1.2" + } + }, + "engine.io-parser": { + "version": "2.1.3", + "resolved": "https://registry.npmjs.org/engine.io-parser/-/engine.io-parser-2.1.3.tgz", + "integrity": "sha512-6HXPre2O4Houl7c4g7Ic/XzPnHBvaEmN90vtRO9uLmwtRqQmTOw0QMevL1TOfL2Cpu1VzsaTmMotQgMdkzGkVA==", + "dev": true, + "requires": { + "after": "0.8.2", + "arraybuffer.slice": "~0.0.7", + "base64-arraybuffer": "0.1.5", + "blob": "0.0.5", + "has-binary2": "~1.0.2" + } + }, + "isarray": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/isarray/-/isarray-2.0.1.tgz", + "integrity": "sha1-o32U7ZzaLVmGXJ92/llu4fM4dB4=", + "dev": true + }, + "ms": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", + "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=", + "dev": true + }, + "parseqs": { + "version": "0.0.5", + "resolved": "https://registry.npmjs.org/parseqs/-/parseqs-0.0.5.tgz", + "integrity": "sha1-1SCKNzjkZ2bikbouoXNoSSGouJ0=", + "dev": true, + "requires": { + "better-assert": "~1.0.0" + } + }, + "parseuri": { + "version": "0.0.5", + "resolved": "https://registry.npmjs.org/parseuri/-/parseuri-0.0.5.tgz", + "integrity": "sha1-gCBKUNTbt3m/3G6+J3jZDkvOMgo=", + "dev": true, + "requires": { + "better-assert": "~1.0.0" + } + }, + "socket.io-client": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/socket.io-client/-/socket.io-client-2.1.1.tgz", + "integrity": "sha512-jxnFyhAuFxYfjqIgduQlhzqTcOEQSn+OHKVfAxWaNWa7ecP7xSNk2Dx/3UEsDcY7NcFafxvNvKPmmO7HTwTxGQ==", + "dev": true, + "requires": { + "backo2": "1.0.2", + "base64-arraybuffer": "0.1.5", + "component-bind": "1.0.0", + "component-emitter": "1.2.1", + "debug": "~3.1.0", + "engine.io-client": "~3.2.0", + "has-binary2": "~1.0.2", + "has-cors": "1.1.0", + "indexof": "0.0.1", + "object-component": "0.0.3", + "parseqs": "0.0.5", + "parseuri": "0.0.5", + "socket.io-parser": "~3.2.0", + "to-array": "0.1.4" + } + }, + "socket.io-parser": { + "version": "3.2.0", + "resolved": "https://registry.npmjs.org/socket.io-parser/-/socket.io-parser-3.2.0.tgz", + "integrity": "sha512-FYiBx7rc/KORMJlgsXysflWx/RIvtqZbyGLlHZvjfmPTPeuD/I8MaW7cfFrj5tRltICJdgwflhfZ3NVVbVLFQA==", + "dev": true, + "requires": { + "component-emitter": "1.2.1", + "debug": "~3.1.0", + "isarray": "2.0.1" + } + }, + "ws": { + "version": "3.3.3", + "resolved": "https://registry.npmjs.org/ws/-/ws-3.3.3.tgz", + "integrity": "sha512-nnWLa/NwZSt4KQJu51MYlCcSQ5g7INpOrOMt4XV8j4dqTXdmlUmSHQ8/oLC069ckre0fRsgfvsKwbTdtKLCDkA==", + "dev": true, + "requires": { + "async-limiter": "~1.0.0", + "safe-buffer": "~5.1.0", + "ultron": "~1.1.0" + } + } + } + }, + "socket.io-adapter": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/socket.io-adapter/-/socket.io-adapter-1.1.2.tgz", + "integrity": "sha512-WzZRUj1kUjrTIrUKpZLEzFZ1OLj5FwLlAFQs9kuZJzJi5DKdU7FsWc36SNmA8iDOtwBQyT8FkrriRM8vXLYz8g==", + "dev": true + }, + "socket.io-client": { + "version": "2.3.1", + "resolved": "https://registry.npmjs.org/socket.io-client/-/socket.io-client-2.3.1.tgz", + "integrity": "sha512-YXmXn3pA8abPOY//JtYxou95Ihvzmg8U6kQyolArkIyLd0pgVhrfor/iMsox8cn07WCOOvvuJ6XKegzIucPutQ==", + "dev": true, + "requires": { + "backo2": "1.0.2", + "component-bind": "1.0.0", + "component-emitter": "~1.3.0", + "debug": "~3.1.0", + "engine.io-client": "~3.4.0", + "has-binary2": "~1.0.2", + "indexof": "0.0.1", + "parseqs": "0.0.6", + "parseuri": "0.0.6", + "socket.io-parser": "~3.3.0", + "to-array": "0.1.4" + }, + "dependencies": { + "debug": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/debug/-/debug-3.1.0.tgz", + "integrity": "sha512-OX8XqP7/1a9cqkxYw2yXss15f26NKWBpDXQd0/uK/KPqdQhxbPa994hnzjcE2VqQpDslf55723cKPUOGSmMY3g==", + "dev": true, + "requires": { + "ms": "2.0.0" + } + }, + "ms": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", + "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=", + "dev": true + } + } + }, + "socket.io-parser": { + "version": "3.3.1", + "resolved": "https://registry.npmjs.org/socket.io-parser/-/socket.io-parser-3.3.1.tgz", + "integrity": "sha512-1QLvVAe8dTz+mKmZ07Swxt+LAo4Y1ff50rlyoEx00TQmDFVQYPfcqGvIDJLGaBdhdNCecXtyKpD+EgKGcmmbuQ==", + "dev": true, + "requires": { + "component-emitter": "~1.3.0", + "debug": "~3.1.0", + "isarray": "2.0.1" + }, + "dependencies": { + "debug": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/debug/-/debug-3.1.0.tgz", + "integrity": "sha512-OX8XqP7/1a9cqkxYw2yXss15f26NKWBpDXQd0/uK/KPqdQhxbPa994hnzjcE2VqQpDslf55723cKPUOGSmMY3g==", + "dev": true, + "requires": { + "ms": "2.0.0" + } + }, + "isarray": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/isarray/-/isarray-2.0.1.tgz", + "integrity": "sha1-o32U7ZzaLVmGXJ92/llu4fM4dB4=", + "dev": true + }, + "ms": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", + "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=", + "dev": true + } + } + }, + "sockjs": { + "version": "0.3.20", + "resolved": "https://registry.npmjs.org/sockjs/-/sockjs-0.3.20.tgz", + "integrity": "sha512-SpmVOVpdq0DJc0qArhF3E5xsxvaiqGNb73XfgBpK1y3UD5gs8DSo8aCTsuT5pX8rssdc2NDIzANwP9eCAiSdTA==", + "dev": true, + "requires": { + "faye-websocket": "^0.10.0", + "uuid": "^3.4.0", + "websocket-driver": "0.6.5" + }, + "dependencies": { + "uuid": { + "version": "3.4.0", + "resolved": "https://registry.npmjs.org/uuid/-/uuid-3.4.0.tgz", + "integrity": "sha512-HjSDRw6gZE5JMggctHBcjVak08+KEVhSIiDzFnT9S9aegmp85S/bReBVTb4QTFaRNptJ9kuYaNhnbNEOkbKb/A==", + "dev": true + } + } + }, + "sockjs-client": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/sockjs-client/-/sockjs-client-1.4.0.tgz", + "integrity": "sha512-5zaLyO8/nri5cua0VtOrFXBPK1jbL4+1cebT/mmKA1E1ZXOvJrII75bPu0l0k843G/+iAbhEqzyKr0w/eCCj7g==", + "dev": true, + "requires": { + "debug": "^3.2.5", + "eventsource": "^1.0.7", + "faye-websocket": "~0.11.1", + "inherits": "^2.0.3", + "json3": "^3.3.2", + "url-parse": "^1.4.3" + }, + "dependencies": { + "debug": { + "version": "3.2.7", + "resolved": "https://registry.npmjs.org/debug/-/debug-3.2.7.tgz", + "integrity": "sha512-CFjzYYAi4ThfiQvizrFQevTTXHtnCqWfe7x1AhgEscTz6ZbLbfoLRLPugTQyBth6f8ZERVUSyWHFD/7Wu4t1XQ==", + "dev": true, + "requires": { + "ms": "^2.1.1" + } + }, + "faye-websocket": { + "version": "0.11.3", + "resolved": "https://registry.npmjs.org/faye-websocket/-/faye-websocket-0.11.3.tgz", + "integrity": "sha512-D2y4bovYpzziGgbHYtGCMjlJM36vAl/y+xUyn1C+FVx8szd1E+86KwVw6XvYSzOP8iMpm1X0I4xJD+QtUb36OA==", + "dev": true, + "requires": { + "websocket-driver": ">=0.5.1" + } + } + } + }, + "socks": { + "version": "2.3.3", + "resolved": "https://registry.npmjs.org/socks/-/socks-2.3.3.tgz", + "integrity": "sha512-o5t52PCNtVdiOvzMry7wU4aOqYWL0PeCXRWBEiJow4/i/wr+wpsJQ9awEu1EonLIqsfGd5qSgDdxEOvCdmBEpA==", + "dev": true, + "requires": { + "ip": "1.1.5", + "smart-buffer": "^4.1.0" + } + }, + "socks-proxy-agent": { + "version": "4.0.2", + "resolved": "https://registry.npmjs.org/socks-proxy-agent/-/socks-proxy-agent-4.0.2.tgz", + "integrity": "sha512-NT6syHhI9LmuEMSK6Kd2V7gNv5KFZoLE7V5udWmn0de+3Mkj3UMA/AJPLyeNUVmElCurSHtUdM3ETpR3z770Wg==", + "dev": true, + "requires": { + "agent-base": "~4.2.1", + "socks": "~2.3.2" + }, + "dependencies": { + "agent-base": { + "version": "4.2.1", + "resolved": "https://registry.npmjs.org/agent-base/-/agent-base-4.2.1.tgz", + "integrity": "sha512-JVwXMr9nHYTUXsBFKUqhJwvlcYU/blreOEUkhNR2eXZIvwd+c+o5V4MgDPKWnMS/56awN3TRzIP+KoPn+roQtg==", + "dev": true, + "requires": { + "es6-promisify": "^5.0.0" + } + } + } + }, + "sort-keys": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/sort-keys/-/sort-keys-1.1.2.tgz", + "integrity": "sha1-RBttTTRnmPG05J6JIK37oOVD+a0=", + "dev": true, + "requires": { + "is-plain-obj": "^1.0.0" + } + }, + "source-list-map": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/source-list-map/-/source-list-map-2.0.1.tgz", + "integrity": "sha512-qnQ7gVMxGNxsiL4lEuJwe/To8UnK7fAnmbGEEH8RpLouuKbeEm0lhbQVFIrNSuB+G7tVrAlVsZgETT5nljf+Iw==", + "dev": true + }, + "source-map": { + "version": "0.5.7", + "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.5.7.tgz", + "integrity": "sha1-igOdLRAh0i0eoUyA2OpGi6LvP8w=" + }, + "source-map-resolve": { + "version": "0.5.3", + "resolved": "https://registry.npmjs.org/source-map-resolve/-/source-map-resolve-0.5.3.tgz", + "integrity": "sha512-Htz+RnsXWk5+P2slx5Jh3Q66vhQj1Cllm0zvnaY98+NFx+Dv2CF/f5O/t8x+KaNdrdIAsruNzoh/KpialbqAnw==", + "dev": true, + "requires": { + "atob": "^2.1.2", + "decode-uri-component": "^0.2.0", + "resolve-url": "^0.2.1", + "source-map-url": "^0.4.0", + "urix": "^0.1.0" + } + }, + "source-map-support": { + "version": "0.5.19", + "resolved": "https://registry.npmjs.org/source-map-support/-/source-map-support-0.5.19.tgz", + "integrity": "sha512-Wonm7zOCIJzBGQdB+thsPar0kYuCIzYvxZwlBa87yi/Mdjv7Tip2cyVbLj5o0cFPN4EVkuTwb3GDDyUx2DGnGw==", + "dev": true, + "requires": { + "buffer-from": "^1.0.0", + "source-map": "^0.6.0" + }, + "dependencies": { + "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-url": { + "version": "0.4.0", + "resolved": "https://registry.npmjs.org/source-map-url/-/source-map-url-0.4.0.tgz", + "integrity": "sha1-PpNdfd1zYxuXZZlW1VEo6HtQhKM=", + "dev": true + }, + "sourcemap-codec": { + "version": "1.4.8", + "resolved": "https://registry.npmjs.org/sourcemap-codec/-/sourcemap-codec-1.4.8.tgz", + "integrity": "sha512-9NykojV5Uih4lgo5So5dtw+f0JgJX30KCNI8gwhz2J9A15wD0Ml6tjHKwf6fTSa6fAdVBdZeNOs9eJ71qCk8vA==", + "dev": true + }, + "spawn-sync": { + "version": "1.0.15", + "resolved": "https://registry.npmjs.org/spawn-sync/-/spawn-sync-1.0.15.tgz", + "integrity": "sha1-sAeZVX63+wyDdsKdROih6mfldHY=", + "dev": true, + "requires": { + "concat-stream": "^1.4.7", + "os-shim": "^0.1.2" + } + }, + "spdx-correct": { + "version": "3.1.1", + "resolved": "https://registry.npmjs.org/spdx-correct/-/spdx-correct-3.1.1.tgz", + "integrity": "sha512-cOYcUWwhCuHCXi49RhFRCyJEK3iPj1Ziz9DpViV3tbZOwXD49QzIN3MpOLJNxh2qwq2lJJZaKMVw9qNi4jTC0w==", + "dev": true, + "requires": { + "spdx-expression-parse": "^3.0.0", + "spdx-license-ids": "^3.0.0" + } + }, + "spdx-exceptions": { + "version": "2.3.0", + "resolved": "https://registry.npmjs.org/spdx-exceptions/-/spdx-exceptions-2.3.0.tgz", + "integrity": "sha512-/tTrYOC7PPI1nUAgx34hUpqXuyJG+DTHJTnIULG4rDygi4xu/tfgmq1e1cIRwRzwZgo4NLySi+ricLkZkw4i5A==", + "dev": true + }, + "spdx-expression-parse": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/spdx-expression-parse/-/spdx-expression-parse-3.0.1.tgz", + "integrity": "sha512-cbqHunsQWnJNE6KhVSMsMeH5H/L9EpymbzqTQ3uLwNCLZ1Q481oWaofqH7nO6V07xlXwY6PhQdQ2IedWx/ZK4Q==", + "dev": true, + "requires": { + "spdx-exceptions": "^2.1.0", + "spdx-license-ids": "^3.0.0" + } + }, + "spdx-license-ids": { + "version": "3.0.6", + "resolved": "https://registry.npmjs.org/spdx-license-ids/-/spdx-license-ids-3.0.6.tgz", + "integrity": "sha512-+orQK83kyMva3WyPf59k1+Y525csj5JejicWut55zeTWANuN17qSiSLUXWtzHeNWORSvT7GLDJ/E/XiIWoXBTw==", + "dev": true + }, + "spdy": { + "version": "4.0.2", + "resolved": "https://registry.npmjs.org/spdy/-/spdy-4.0.2.tgz", + "integrity": "sha512-r46gZQZQV+Kl9oItvl1JZZqJKGr+oEkB08A6BzkiR7593/7IbtuncXHd2YoYeTsG4157ZssMu9KYvUHLcjcDoA==", + "dev": true, + "requires": { + "debug": "^4.1.0", + "handle-thing": "^2.0.0", + "http-deceiver": "^1.2.7", + "select-hose": "^2.0.0", + "spdy-transport": "^3.0.0" + } + }, + "spdy-transport": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/spdy-transport/-/spdy-transport-3.0.0.tgz", + "integrity": "sha512-hsLVFE5SjA6TCisWeJXFKniGGOpBgMLmerfO2aCyCU5s7nJ/rpAepqmFifv/GCbSbueEeAJJnmSQ2rKC/g8Fcw==", + "dev": true, + "requires": { + "debug": "^4.1.0", + "detect-node": "^2.0.4", + "hpack.js": "^2.1.6", + "obuf": "^1.1.2", + "readable-stream": "^3.0.6", + "wbuf": "^1.7.3" + }, + "dependencies": { + "readable-stream": { + "version": "3.6.0", + "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-3.6.0.tgz", + "integrity": "sha512-BViHy7LKeTz4oNnkcLJ+lVSL6vpiFeX6/d3oSH8zCW7UxP2onchk+vTGB143xuFjHS3deTgkKoXXymXqymiIdA==", + "dev": true, + "requires": { + "inherits": "^2.0.3", + "string_decoder": "^1.1.1", + "util-deprecate": "^1.0.1" + } + } + } + }, + "split-string": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/split-string/-/split-string-3.1.0.tgz", + "integrity": "sha512-NzNVhJDYpwceVVii8/Hu6DKfD2G+NrQHlS/V/qgv763EYudVwEcMQNxd2lh+0VrUByXN/oJkl5grOhYWvQUYiw==", + "dev": true, + "requires": { + "extend-shallow": "^3.0.0" + } + }, + "sprintf-js": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/sprintf-js/-/sprintf-js-1.1.2.tgz", + "integrity": "sha512-VE0SOVEHCk7Qc8ulkWw3ntAzXuqf7S2lvwQaDLRnUeIEaKNQJzV6BwmLKhOqT61aGhfUMrXeaBk+oDGCzvhcug==", + "dev": true + }, + "sshpk": { + "version": "1.16.1", + "resolved": "https://registry.npmjs.org/sshpk/-/sshpk-1.16.1.tgz", + "integrity": "sha512-HXXqVUq7+pcKeLqqZj6mHFUMvXtOJt1uoUx09pFW6011inTMxqI8BA8PM95myrIyyKwdnzjdFjLiE6KBPVtJIg==", + "dev": true, + "requires": { + "asn1": "~0.2.3", + "assert-plus": "^1.0.0", + "bcrypt-pbkdf": "^1.0.0", + "dashdash": "^1.12.0", + "ecc-jsbn": "~0.1.1", + "getpass": "^0.1.1", + "jsbn": "~0.1.0", + "safer-buffer": "^2.0.2", + "tweetnacl": "~0.14.0" + } + }, + "ssri": { + "version": "6.0.1", + "resolved": "https://registry.npmjs.org/ssri/-/ssri-6.0.1.tgz", + "integrity": "sha512-3Wge10hNcT1Kur4PDFwEieXSCMCJs/7WvSACcrMYrNp+b8kDL1/0wJch5Ni2WrtwEa2IO8OsVfeKIciKCDx/QA==", + "dev": true, + "requires": { + "figgy-pudding": "^3.5.1" + } + }, + "stable": { + "version": "0.1.8", + "resolved": "https://registry.npmjs.org/stable/-/stable-0.1.8.tgz", + "integrity": "sha512-ji9qxRnOVfcuLDySj9qzhGSEFVobyt1kIOSkj1qZzYLzq7Tos/oUUWvotUPQLlrsidqsK6tBH89Bc9kL5zHA6w==", + "dev": true + }, + "stack-trace": { + "version": "0.0.10", + "resolved": "https://registry.npmjs.org/stack-trace/-/stack-trace-0.0.10.tgz", + "integrity": "sha1-VHxws0fo0ytOEI6hoqFZ5f3eGcA=", + "dev": true + }, + "stack-utils": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/stack-utils/-/stack-utils-2.0.3.tgz", + "integrity": "sha512-gL//fkxfWUsIlFL2Tl42Cl6+HFALEaB1FU76I/Fy+oZjRreP7OPMXFlGbxM7NQsI0ZpUfw76sHnv0WNYuTb7Iw==", + "dev": true, + "requires": { + "escape-string-regexp": "^2.0.0" + }, + "dependencies": { + "escape-string-regexp": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-2.0.0.tgz", + "integrity": "sha512-UpzcLCXolUWcNu5HtVMHYdXJjArjsF9C0aNnquZYY4uW/Vu0miy5YoWvbV345HauVvcAUnpRuhMMcqTcGOY2+w==", + "dev": true + } + } + }, + "stackframe": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/stackframe/-/stackframe-1.2.0.tgz", + "integrity": "sha512-GrdeshiRmS1YLMYgzF16olf2jJ/IzxXY9lhKOskuVziubpTYcYqyOwYeJKzQkwy7uN0fYSsbsC4RQaXf9LCrYA==", + "dev": true + }, + "staged-git-files": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/staged-git-files/-/staged-git-files-1.1.2.tgz", + "integrity": "sha512-0Eyrk6uXW6tg9PYkhi/V/J4zHp33aNyi2hOCmhFLqLTIhbgqWn5jlSzI+IU0VqrZq6+DbHcabQl/WP6P3BG0QA==", + "dev": true + }, + "static-extend": { + "version": "0.1.2", + "resolved": "https://registry.npmjs.org/static-extend/-/static-extend-0.1.2.tgz", + "integrity": "sha1-YICcOcv/VTNyJv1eC1IPNB8ftcY=", + "dev": true, + "requires": { + "define-property": "^0.2.5", + "object-copy": "^0.1.0" + }, + "dependencies": { + "define-property": { + "version": "0.2.5", + "resolved": "https://registry.npmjs.org/define-property/-/define-property-0.2.5.tgz", + "integrity": "sha1-w1se+RjsPJkPmlvFe+BKrOxcgRY=", + "dev": true, + "requires": { + "is-descriptor": "^0.1.0" + } + } + } + }, + "statuses": { + "version": "1.3.1", + "resolved": "https://registry.npmjs.org/statuses/-/statuses-1.3.1.tgz", + "integrity": "sha1-+vUbnrdKrvOzrPStX2Gr8ky3uT4=", + "dev": true + }, + "stealthy-require": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/stealthy-require/-/stealthy-require-1.1.1.tgz", + "integrity": "sha1-NbCYdbT/SfJqd35QmzCQoyJr8ks=", + "dev": true + }, + "stream-browserify": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/stream-browserify/-/stream-browserify-2.0.2.tgz", + "integrity": "sha512-nX6hmklHs/gr2FuxYDltq8fJA1GDlxKQCz8O/IM4atRqBH8OORmBNgfvW5gG10GT/qQ9u0CzIvr2X5Pkt6ntqg==", + "dev": true, + "requires": { + "inherits": "~2.0.1", + "readable-stream": "^2.0.2" + } + }, + "stream-each": { + "version": "1.2.3", + "resolved": "https://registry.npmjs.org/stream-each/-/stream-each-1.2.3.tgz", + "integrity": "sha512-vlMC2f8I2u/bZGqkdfLQW/13Zihpej/7PmSiMQsbYddxuTsJp8vRe2x2FvVExZg7FaOds43ROAuFJwPR4MTZLw==", + "dev": true, + "requires": { + "end-of-stream": "^1.1.0", + "stream-shift": "^1.0.0" + } + }, + "stream-http": { + "version": "2.8.3", + "resolved": "https://registry.npmjs.org/stream-http/-/stream-http-2.8.3.tgz", + "integrity": "sha512-+TSkfINHDo4J+ZobQLWiMouQYB+UVYFttRA94FpEzzJ7ZdqcL4uUUQ7WkdkI4DSozGmgBUE/a47L+38PenXhUw==", + "dev": true, + "requires": { + "builtin-status-codes": "^3.0.0", + "inherits": "^2.0.1", + "readable-stream": "^2.3.6", + "to-arraybuffer": "^1.0.0", + "xtend": "^4.0.0" + } + }, + "stream-shift": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/stream-shift/-/stream-shift-1.0.1.tgz", + "integrity": "sha512-AiisoFqQ0vbGcZgQPY1cdP2I76glaVA/RauYR4G4thNFgkTqr90yXTo4LYX60Jl+sIlPNHHdGSwo01AvbKUSVQ==", + "dev": true + }, + "stream-throttle": { + "version": "0.1.3", + "resolved": "https://registry.npmjs.org/stream-throttle/-/stream-throttle-0.1.3.tgz", + "integrity": "sha1-rdV8jXzHOoFjDTHNVdOWHPr7qcM=", + "dev": true, + "requires": { + "commander": "^2.2.0", + "limiter": "^1.0.5" + } + }, + "streamfilter": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/streamfilter/-/streamfilter-3.0.0.tgz", + "integrity": "sha512-kvKNfXCmUyC8lAXSSHCIXBUlo/lhsLcCU/OmzACZYpRUdtKIH68xYhm/+HI15jFJYtNJGYtCgn2wmIiExY1VwA==", + "dev": true, + "requires": { + "readable-stream": "^3.0.6" + }, + "dependencies": { + "readable-stream": { + "version": "3.6.0", + "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-3.6.0.tgz", + "integrity": "sha512-BViHy7LKeTz4oNnkcLJ+lVSL6vpiFeX6/d3oSH8zCW7UxP2onchk+vTGB143xuFjHS3deTgkKoXXymXqymiIdA==", + "dev": true, + "requires": { + "inherits": "^2.0.3", + "string_decoder": "^1.1.1", + "util-deprecate": "^1.0.1" + } + } + } + }, + "strict-uri-encode": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/strict-uri-encode/-/strict-uri-encode-1.1.0.tgz", + "integrity": "sha1-J5siXfHVgrH1TmWt3UNS4Y+qBxM=", + "dev": true + }, + "string-argv": { + "version": "0.0.2", + "resolved": "https://registry.npmjs.org/string-argv/-/string-argv-0.0.2.tgz", + "integrity": "sha1-2sMECGkMIfPDYwo/86BYd73L1zY=", + "dev": true + }, + "string-length": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/string-length/-/string-length-4.0.1.tgz", + "integrity": "sha512-PKyXUd0LK0ePjSOnWn34V2uD6acUWev9uy0Ft05k0E8xRW+SKcA0F7eMr7h5xlzfn+4O3N+55rduYyet3Jk+jw==", + "dev": true, + "requires": { + "char-regex": "^1.0.2", + "strip-ansi": "^6.0.0" + } + }, + "string-template": { + "version": "0.2.1", + "resolved": "https://registry.npmjs.org/string-template/-/string-template-0.2.1.tgz", + "integrity": "sha1-QpMuWYo1LQH8IuwzZ9nYTuxsmt0=", + "dev": true + }, + "string-width": { + "version": "4.2.0", + "resolved": "https://registry.npmjs.org/string-width/-/string-width-4.2.0.tgz", + "integrity": "sha512-zUz5JD+tgqtuDjMhwIg5uFVV3dtqZ9yQJlZVfq4I01/K5Paj5UHj7VyrQOJvzawSVlKpObApbfD0Ed6yJc+1eg==", + "requires": { + "emoji-regex": "^8.0.0", + "is-fullwidth-code-point": "^3.0.0", + "strip-ansi": "^6.0.0" + } + }, + "string.prototype.trimend": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/string.prototype.trimend/-/string.prototype.trimend-1.0.3.tgz", + "integrity": "sha512-ayH0pB+uf0U28CtjlLvL7NaohvR1amUvVZk+y3DYb0Ey2PUV5zPkkKy9+U1ndVEIXO8hNg18eIv9Jntbii+dKw==", + "dev": true, + "requires": { + "call-bind": "^1.0.0", + "define-properties": "^1.1.3" + } + }, + "string.prototype.trimstart": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/string.prototype.trimstart/-/string.prototype.trimstart-1.0.3.tgz", + "integrity": "sha512-oBIBUy5lea5tt0ovtOFiEQaBkoBBkyJhZXzJYrSmDo5IUUqbOPvVezuRs/agBIdZ2p2Eo1FD6bD9USyBLfl3xg==", + "dev": true, + "requires": { + "call-bind": "^1.0.0", + "define-properties": "^1.1.3" + } + }, + "string_decoder": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/string_decoder/-/string_decoder-1.1.1.tgz", + "integrity": "sha512-n/ShnvDi6FHbbVfviro+WojiFzv+s8MPMHBczVePfUpDJLwoLT0ht1l4YwBCbi8pJAveEEdnkHyPyTP/mzRfwg==", + "dev": true, + "requires": { + "safe-buffer": "~5.1.0" + } + }, + "stringify-object": { + "version": "3.3.0", + "resolved": "https://registry.npmjs.org/stringify-object/-/stringify-object-3.3.0.tgz", + "integrity": "sha512-rHqiFh1elqCQ9WPLIC8I0Q/g/wj5J1eMkyoiD6eoQApWHP0FtlK7rqnhmabL5VUY9JQCcqwwvlOaSuutekgyrw==", + "dev": true, + "requires": { + "get-own-enumerable-property-symbols": "^3.0.0", + "is-obj": "^1.0.1", + "is-regexp": "^1.0.0" + }, + "dependencies": { + "is-obj": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/is-obj/-/is-obj-1.0.1.tgz", + "integrity": "sha1-PkcprB9f3gJc19g6iW2rn09n2w8=", + "dev": true + } + } + }, + "strip-ansi": { + "version": "6.0.0", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.0.tgz", + "integrity": "sha512-AuvKTrTfQNYNIctbR1K/YGTR1756GycPsg7b9bdV9Duqur4gv6aKqHXah67Z8ImS7WEz5QVcOtlfW2rZEugt6w==", + "requires": { + "ansi-regex": "^5.0.0" + } + }, + "strip-bom": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/strip-bom/-/strip-bom-2.0.0.tgz", + "integrity": "sha1-YhmoVhZSBJHzV4i9vxRHqZx+aw4=", + "dev": true, + "requires": { + "is-utf8": "^0.2.0" + } + }, + "strip-bom-buf": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/strip-bom-buf/-/strip-bom-buf-1.0.0.tgz", + "integrity": "sha1-HLRar1dTD0yvhsf3UXnSyaUd1XI=", + "dev": true, + "requires": { + "is-utf8": "^0.2.1" + } + }, + "strip-bom-stream": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/strip-bom-stream/-/strip-bom-stream-2.0.0.tgz", + "integrity": "sha1-+H217yYT9paKpUWr/h7HKLaoKco=", + "dev": true, + "requires": { + "first-chunk-stream": "^2.0.0", + "strip-bom": "^2.0.0" + } + }, + "strip-comments": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/strip-comments/-/strip-comments-1.0.2.tgz", + "integrity": "sha512-kL97alc47hoyIQSV165tTt9rG5dn4w1dNnBhOQ3bOU1Nc1hel09jnXANaHJ7vzHLd4Ju8kseDGzlev96pghLFw==", + "dev": true, + "requires": { + "babel-extract-comments": "^1.0.0", + "babel-plugin-transform-object-rest-spread": "^6.26.0" + } + }, + "strip-eof": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/strip-eof/-/strip-eof-1.0.0.tgz", + "integrity": "sha1-u0P/VZim6wXYm1n80SnJgzE2Br8=", + "dev": true + }, + "strip-final-newline": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/strip-final-newline/-/strip-final-newline-2.0.0.tgz", + "integrity": "sha512-BrpvfNAE3dcvq7ll3xVumzjKjZQ5tI1sEUIKr3Uoks0XUl45St3FlatVqef9prk4jRDzhW6WZg+3bk93y6pLjA==", + "dev": true + }, + "strip-indent": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/strip-indent/-/strip-indent-3.0.0.tgz", + "integrity": "sha512-laJTa3Jb+VQpaC6DseHhF7dXVqHTfJPCRDaEbid/drOhgitgYku/letMUqOXFoWV0zIIUbjpdH2t+tYj4bQMRQ==", + "dev": true, + "requires": { + "min-indent": "^1.0.0" + } + }, + "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 + }, + "style-loader": { + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/style-loader/-/style-loader-1.2.1.tgz", + "integrity": "sha512-ByHSTQvHLkWE9Ir5+lGbVOXhxX10fbprhLvdg96wedFZb4NDekDPxVKv5Fwmio+QcMlkkNfuK+5W1peQ5CUhZg==", + "dev": true, + "requires": { + "loader-utils": "^2.0.0", + "schema-utils": "^2.6.6" + } + }, + "stylehacks": { + "version": "4.0.3", + "resolved": "https://registry.npmjs.org/stylehacks/-/stylehacks-4.0.3.tgz", + "integrity": "sha512-7GlLk9JwlElY4Y6a/rmbH2MhVlTyVmiJd1PfTCqFaIBEGMYNsrO/v3SeGTdhBThLg4Z+NbOk/qFMwCa+J+3p/g==", + "dev": true, + "requires": { + "browserslist": "^4.0.0", + "postcss": "^7.0.0", + "postcss-selector-parser": "^3.0.0" + }, + "dependencies": { + "postcss-selector-parser": { + "version": "3.1.2", + "resolved": "https://registry.npmjs.org/postcss-selector-parser/-/postcss-selector-parser-3.1.2.tgz", + "integrity": "sha512-h7fJ/5uWuRVyOtkO45pnt1Ih40CEleeyCHzipqAZO2e5H20g25Y48uYnFUiShvY4rZWNJ/Bib/KVPmanaCtOhA==", + "dev": true, + "requires": { + "dot-prop": "^5.2.0", + "indexes-of": "^1.0.1", + "uniq": "^1.0.1" + } + } + } + }, + "supports-color": { + "version": "5.5.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-5.5.0.tgz", + "integrity": "sha512-QjVjwdXIt408MIiAqCX4oUKsgU2EqAGzs2Ppkm4aQYbjm+ZEWEcW4SfFNTr4uMNZma0ey4f5lgLrkB0aX0QMow==", + "requires": { + "has-flag": "^3.0.0" + } + }, + "supports-hyperlinks": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/supports-hyperlinks/-/supports-hyperlinks-2.1.0.tgz", + "integrity": "sha512-zoE5/e+dnEijk6ASB6/qrK+oYdm2do1hjoLWrqUC/8WEIW1gbxFcKuBof7sW8ArN6e+AYvsE8HBGiVRWL/F5CA==", + "dev": true, + "requires": { + "has-flag": "^4.0.0", + "supports-color": "^7.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": "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" + } + } + } + }, + "svgo": { + "version": "1.3.2", + "resolved": "https://registry.npmjs.org/svgo/-/svgo-1.3.2.tgz", + "integrity": "sha512-yhy/sQYxR5BkC98CY7o31VGsg014AKLEPxdfhora76l36hD9Rdy5NZA/Ocn6yayNPgSamYdtX2rFJdcv07AYVw==", + "dev": true, + "requires": { + "chalk": "^2.4.1", + "coa": "^2.0.2", + "css-select": "^2.0.0", + "css-select-base-adapter": "^0.1.1", + "css-tree": "1.0.0-alpha.37", + "csso": "^4.0.2", + "js-yaml": "^3.13.1", + "mkdirp": "~0.5.1", + "object.values": "^1.1.0", + "sax": "~1.2.4", + "stable": "^0.1.8", + "unquote": "~1.1.1", + "util.promisify": "~1.0.0" + }, + "dependencies": { + "css-select": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/css-select/-/css-select-2.1.0.tgz", + "integrity": "sha512-Dqk7LQKpwLoH3VovzZnkzegqNSuAziQyNZUcrdDM401iY+R5NkGBXGmtO05/yaXQziALuPogeG0b7UAgjnTJTQ==", + "dev": true, + "requires": { + "boolbase": "^1.0.0", + "css-what": "^3.2.1", + "domutils": "^1.7.0", + "nth-check": "^1.0.2" + } + }, + "css-what": { + "version": "3.4.2", + "resolved": "https://registry.npmjs.org/css-what/-/css-what-3.4.2.tgz", + "integrity": "sha512-ACUm3L0/jiZTqfzRM3Hi9Q8eZqd6IK37mMWPLz9PJxkLWllYeRf+EHUSHYEtFop2Eqytaq1FizFVh7XfBnXCDQ==", + "dev": true + }, + "dom-serializer": { + "version": "0.2.2", + "resolved": "https://registry.npmjs.org/dom-serializer/-/dom-serializer-0.2.2.tgz", + "integrity": "sha512-2/xPb3ORsQ42nHYiSunXkDjPLBaEj/xTwUO4B7XCZQTRk7EBtTOPaygh10YAAh2OI1Qrp6NWfpAhzswj0ydt9g==", + "dev": true, + "requires": { + "domelementtype": "^2.0.1", + "entities": "^2.0.0" + }, + "dependencies": { + "domelementtype": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/domelementtype/-/domelementtype-2.0.2.tgz", + "integrity": "sha512-wFwTwCVebUrMgGeAwRL/NhZtHAUyT9n9yg4IMDwf10+6iCMxSkVq9MGCVEH+QZWo1nNidy8kNvwmv4zWHDTqvA==", + "dev": true + } + } + }, + "domelementtype": { + "version": "1.3.1", + "resolved": "https://registry.npmjs.org/domelementtype/-/domelementtype-1.3.1.tgz", + "integrity": "sha512-BSKB+TSpMpFI/HOxCNr1O8aMOTZ8hT3pM3GQ0w/mWRmkhEDSFJkkyzz4XQsBV44BChwGkrDfMyjVD0eA2aFV3w==", + "dev": true + }, + "domutils": { + "version": "1.7.0", + "resolved": "https://registry.npmjs.org/domutils/-/domutils-1.7.0.tgz", + "integrity": "sha512-Lgd2XcJ/NjEw+7tFvfKxOzCYKZsdct5lczQ2ZaQY8Djz7pfAD3Gbp8ySJWtreII/vDlMVmxwa6pHmdxIYgttDg==", + "dev": true, + "requires": { + "dom-serializer": "0", + "domelementtype": "1" + } + }, + "sax": { + "version": "1.2.4", + "resolved": "https://registry.npmjs.org/sax/-/sax-1.2.4.tgz", + "integrity": "sha512-NqVDv9TpANUjFm0N8uM5GxL36UgKi9/atZw+x7YFnQ8ckwFGKrl4xX4yWtrey3UJm5nP1kUbnYgLopqWNSRhWw==", + "dev": true + } + } + }, + "swagger-ui-dist": { + "version": "3.25.1", + "resolved": "https://registry.npmjs.org/swagger-ui-dist/-/swagger-ui-dist-3.25.1.tgz", + "integrity": "sha512-Sw/K95j1pT9TZtLKiHDEml7YqcXC9thTTQjxrvNgd9j1KzOIxpo/5lhHuUMAN/hxVAHetzmcBcQaBjywRXog8w==" + }, + "symbol-observable": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/symbol-observable/-/symbol-observable-1.2.0.tgz", + "integrity": "sha512-e900nM8RRtGhlV36KGEU9k65K3mPb1WV70OdjfxlG2EAuM1noi/E/BaW/uMhL7bPEssK8QV57vN3esixjUvcXQ==", + "dev": true + }, + "symbol-tree": { + "version": "3.2.4", + "resolved": "https://registry.npmjs.org/symbol-tree/-/symbol-tree-3.2.4.tgz", + "integrity": "sha512-9QNk5KwDF+Bvz+PyObkmSYjI5ksVUYtjW7AU22r2NKcfLJcXp96hkDWU3+XndOsUb+AQ9QhfzfCT2O+CNWT5Tw==", + "dev": true + }, + "synchronous-promise": { + "version": "2.0.15", + "resolved": "https://registry.npmjs.org/synchronous-promise/-/synchronous-promise-2.0.15.tgz", + "integrity": "sha512-k8uzYIkIVwmT+TcglpdN50pS2y1BDcUnBPK9iJeGu0Pl1lOI8pD6wtzgw91Pjpe+RxtTncw32tLxs/R0yNL2Mg==", + "dev": true + }, + "table": { + "version": "5.4.6", + "resolved": "https://registry.npmjs.org/table/-/table-5.4.6.tgz", + "integrity": "sha512-wmEc8m4fjnob4gt5riFRtTu/6+4rSe12TpAELNSqHMfF3IqnA+CH37USM6/YR3qRZv7e56kAEAtd6nKZaxe0Ug==", + "dev": true, + "requires": { + "ajv": "^6.10.2", + "lodash": "^4.17.14", + "slice-ansi": "^2.1.0", + "string-width": "^3.0.0" + }, + "dependencies": { + "ansi-regex": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-4.1.0.tgz", + "integrity": "sha512-1apePfXM1UOSqw0o9IiFAovVz9M5S1Dg+4TrDwfMewQ6p/rmMueb7tWZjQ1rx4Loy1ArBggoqGpfqqdI4rondg==", + "dev": true + }, + "emoji-regex": { + "version": "7.0.3", + "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-7.0.3.tgz", + "integrity": "sha512-CwBLREIQ7LvYFB0WyRvwhq5N5qPhc6PMjD6bYggFlI5YyDgl+0vxq5VHbMOFqLg7hfWzmu8T5Z1QofhmTIhItA==", + "dev": true + }, + "is-fullwidth-code-point": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-2.0.0.tgz", + "integrity": "sha1-o7MKXE8ZkYMWeqq5O+764937ZU8=", + "dev": true + }, + "string-width": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/string-width/-/string-width-3.1.0.tgz", + "integrity": "sha512-vafcv6KjVZKSgz06oM/H6GDBrAtz8vdhQakGjFIvNrHA6y3HCF1CInLy+QLq8dTJPQ1b+KDUqDFctkdRW44e1w==", + "dev": true, + "requires": { + "emoji-regex": "^7.0.1", + "is-fullwidth-code-point": "^2.0.0", + "strip-ansi": "^5.1.0" + } + }, + "strip-ansi": { + "version": "5.2.0", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-5.2.0.tgz", + "integrity": "sha512-DuRs1gKbBqsMKIZlrffwlug8MHkcnpjs5VPmL1PAh+mA30U0DTotfDZ0d2UUsXpPmPmMMJ6W773MaA3J+lbiWA==", + "dev": true, + "requires": { + "ansi-regex": "^4.1.0" + } + } + } + }, + "tabtab": { + "version": "2.2.2", + "resolved": "https://registry.npmjs.org/tabtab/-/tabtab-2.2.2.tgz", + "integrity": "sha1-egR/FDsBC0y9MfhX6ClhUSy/ThQ=", + "dev": true, + "requires": { + "debug": "^2.2.0", + "inquirer": "^1.0.2", + "lodash.difference": "^4.5.0", + "lodash.uniq": "^4.5.0", + "minimist": "^1.2.0", + "mkdirp": "^0.5.1", + "npmlog": "^2.0.3", + "object-assign": "^4.1.0" + }, + "dependencies": { + "ansi-escapes": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/ansi-escapes/-/ansi-escapes-1.4.0.tgz", + "integrity": "sha1-06ioOzGapneTZisT52HHkRQiMG4=", + "dev": true + }, + "ansi-regex": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-2.1.1.tgz", + "integrity": "sha1-w7M6te42DYbg5ijwRorn7yfWVN8=", + "dev": true + }, + "ansi-styles": { + "version": "2.2.1", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-2.2.1.tgz", + "integrity": "sha1-tDLdM1i2NM914eRmQ2gkBTPB3b4=", + "dev": true + }, + "chalk": { + "version": "1.1.3", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-1.1.3.tgz", + "integrity": "sha1-qBFcVeSnAv5NFQq9OHKCKn4J/Jg=", + "dev": true, + "requires": { + "ansi-styles": "^2.2.1", + "escape-string-regexp": "^1.0.2", + "has-ansi": "^2.0.0", + "strip-ansi": "^3.0.0", + "supports-color": "^2.0.0" + } + }, + "cli-cursor": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/cli-cursor/-/cli-cursor-1.0.2.tgz", + "integrity": "sha1-ZNo/fValRBLll5S9Ytw1KV6PKYc=", + "dev": true, + "requires": { + "restore-cursor": "^1.0.1" + } + }, + "debug": { + "version": "2.6.9", + "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", + "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", + "dev": true, + "requires": { + "ms": "2.0.0" + } + }, + "external-editor": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/external-editor/-/external-editor-1.1.1.tgz", + "integrity": "sha1-Etew24UPf/fnCBuvQAVwAGDEYAs=", + "dev": true, + "requires": { + "extend": "^3.0.0", + "spawn-sync": "^1.0.15", + "tmp": "^0.0.29" + } + }, + "figures": { + "version": "1.7.0", + "resolved": "https://registry.npmjs.org/figures/-/figures-1.7.0.tgz", + "integrity": "sha1-y+Hjr/zxzUS4DK3+0o3Hk6lwHS4=", + "dev": true, + "requires": { + "escape-string-regexp": "^1.0.5", + "object-assign": "^4.1.0" + } + }, + "inquirer": { + "version": "1.2.3", + "resolved": "https://registry.npmjs.org/inquirer/-/inquirer-1.2.3.tgz", + "integrity": "sha1-TexvMvN+97sLLtPx0aXD9UUHSRg=", + "dev": true, + "requires": { + "ansi-escapes": "^1.1.0", + "chalk": "^1.0.0", + "cli-cursor": "^1.0.1", + "cli-width": "^2.0.0", + "external-editor": "^1.1.0", + "figures": "^1.3.5", + "lodash": "^4.3.0", + "mute-stream": "0.0.6", + "pinkie-promise": "^2.0.0", + "run-async": "^2.2.0", + "rx": "^4.1.0", + "string-width": "^1.0.1", + "strip-ansi": "^3.0.0", + "through": "^2.3.6" + } + }, + "is-fullwidth-code-point": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-1.0.0.tgz", + "integrity": "sha1-754xOG8DGn8NZDr4L95QxFfvAMs=", + "dev": true, + "requires": { + "number-is-nan": "^1.0.0" + } + }, + "ms": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", + "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=", + "dev": true + }, + "mute-stream": { + "version": "0.0.6", + "resolved": "https://registry.npmjs.org/mute-stream/-/mute-stream-0.0.6.tgz", + "integrity": "sha1-SJYrGeFp/R38JAs/HnMXYnu8R9s=", + "dev": true + }, + "onetime": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/onetime/-/onetime-1.1.0.tgz", + "integrity": "sha1-ofeDj4MUxRbwXs78vEzP4EtO14k=", + "dev": true + }, + "restore-cursor": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/restore-cursor/-/restore-cursor-1.0.1.tgz", + "integrity": "sha1-NGYfRohjJ/7SmRR5FSJS35LapUE=", + "dev": true, + "requires": { + "exit-hook": "^1.0.0", + "onetime": "^1.0.0" + } + }, + "string-width": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/string-width/-/string-width-1.0.2.tgz", + "integrity": "sha1-EYvfW4zcUaKn5w0hHgfisLmxB9M=", + "dev": true, + "requires": { + "code-point-at": "^1.0.0", + "is-fullwidth-code-point": "^1.0.0", + "strip-ansi": "^3.0.0" + } + }, + "strip-ansi": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-3.0.1.tgz", + "integrity": "sha1-ajhfuIU9lS1f8F0Oiq+UJ43GPc8=", + "dev": true, + "requires": { + "ansi-regex": "^2.0.0" + } + }, + "supports-color": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-2.0.0.tgz", + "integrity": "sha1-U10EXOa2Nj+kARcIRimZXp3zJMc=", + "dev": true + }, + "tmp": { + "version": "0.0.29", + "resolved": "https://registry.npmjs.org/tmp/-/tmp-0.0.29.tgz", + "integrity": "sha1-8lEl/w3Z2jzLDC3Tce4SiLuRKMA=", + "dev": true, + "requires": { + "os-tmpdir": "~1.0.1" + } + } + } + }, + "tapable": { + "version": "1.1.3", + "resolved": "https://registry.npmjs.org/tapable/-/tapable-1.1.3.tgz", + "integrity": "sha512-4WK/bYZmj8xLr+HUCODHGF1ZFzsYffasLUgEiMBY4fgtltdO6B4WJtlSbPaDTLpYTcGVwM2qLnFTICEcNxs3kA==", + "dev": true + }, + "tar": { + "version": "4.4.13", + "resolved": "https://registry.npmjs.org/tar/-/tar-4.4.13.tgz", + "integrity": "sha512-w2VwSrBoHa5BsSyH+KxEqeQBAllHhccyMFVHtGtdMpF4W7IRWfZjFiQceJPChOeTsSDVUpER2T8FA93pr0L+QA==", + "dev": true, + "requires": { + "chownr": "^1.1.1", + "fs-minipass": "^1.2.5", + "minipass": "^2.8.6", + "minizlib": "^1.2.1", + "mkdirp": "^0.5.0", + "safe-buffer": "^5.1.2", + "yallist": "^3.0.3" + }, + "dependencies": { + "yallist": { + "version": "3.1.1", + "resolved": "https://registry.npmjs.org/yallist/-/yallist-3.1.1.tgz", + "integrity": "sha512-a4UGQaWPH59mOXUYnAG2ewncQS4i4F43Tv3JoAM+s2VDAmS9NsK8GpDMLrCHPksFT7h3K6TOoUNn2pb7RoXx4g==", + "dev": true + } + } + }, + "temp-dir": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/temp-dir/-/temp-dir-1.0.0.tgz", + "integrity": "sha1-CnwOom06Oa+n4OvqnB/AvE2qAR0=", + "dev": true + }, + "tempy": { + "version": "0.3.0", + "resolved": "https://registry.npmjs.org/tempy/-/tempy-0.3.0.tgz", + "integrity": "sha512-WrH/pui8YCwmeiAoxV+lpRH9HpRtgBhSR2ViBPgpGb/wnYDzp21R4MN45fsCGvLROvY67o3byhJRYRONJyImVQ==", + "dev": true, + "requires": { + "temp-dir": "^1.0.0", + "type-fest": "^0.3.1", + "unique-string": "^1.0.0" + }, + "dependencies": { + "type-fest": { + "version": "0.3.1", + "resolved": "https://registry.npmjs.org/type-fest/-/type-fest-0.3.1.tgz", + "integrity": "sha512-cUGJnCdr4STbePCgqNFbpVNCepa+kAVohJs1sLhxzdH+gnEoOd8VhbYa7pD3zZYGiURWM2xzEII3fQcRizDkYQ==", + "dev": true + } + } + }, + "terminal-link": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/terminal-link/-/terminal-link-2.1.1.tgz", + "integrity": "sha512-un0FmiRUQNr5PJqy9kP7c40F5BOfpGlYTrxonDChEZB7pzZxRNp/bt+ymiy9/npwXya9KH99nJ/GXFIiUkYGFQ==", + "dev": true, + "requires": { + "ansi-escapes": "^4.2.1", + "supports-hyperlinks": "^2.0.0" + } + }, + "terser": { + "version": "4.8.0", + "resolved": "https://registry.npmjs.org/terser/-/terser-4.8.0.tgz", + "integrity": "sha512-EAPipTNeWsb/3wLPeup1tVPaXfIaU68xMnVdPafIL1TV05OhASArYyIfFvnvJCNrR2NIOvDVNNTFRa+Re2MWyw==", + "dev": true, + "requires": { + "commander": "^2.20.0", + "source-map": "~0.6.1", + "source-map-support": "~0.5.12" + }, + "dependencies": { + "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 + } + } + }, + "terser-webpack-plugin": { + "version": "3.0.6", + "resolved": "https://registry.npmjs.org/terser-webpack-plugin/-/terser-webpack-plugin-3.0.6.tgz", + "integrity": "sha512-z3HLOOPUHkCNGkeEHqqiMAIy1pjpHwS1o+i6Zn0Ws3EAvHJj46737efNNEvJ0Vx9BdDQM83d56qySDJOSORA0A==", + "dev": true, + "requires": { + "cacache": "^15.0.4", + "find-cache-dir": "^3.3.1", + "jest-worker": "^26.0.0", + "p-limit": "^3.0.1", + "schema-utils": "^2.6.6", + "serialize-javascript": "^4.0.0", + "source-map": "^0.6.1", + "terser": "^4.8.0", + "webpack-sources": "^1.4.3" + }, + "dependencies": { + "cacache": { + "version": "15.0.5", + "resolved": "https://registry.npmjs.org/cacache/-/cacache-15.0.5.tgz", + "integrity": "sha512-lloiL22n7sOjEEXdL8NAjTgv9a1u43xICE9/203qonkZUCj5X1UEWIdf2/Y0d6QcCtMzbKQyhrcDbdvlZTs/+A==", + "dev": true, + "requires": { + "@npmcli/move-file": "^1.0.1", + "chownr": "^2.0.0", + "fs-minipass": "^2.0.0", + "glob": "^7.1.4", + "infer-owner": "^1.0.4", + "lru-cache": "^6.0.0", + "minipass": "^3.1.1", + "minipass-collect": "^1.0.2", + "minipass-flush": "^1.0.5", + "minipass-pipeline": "^1.2.2", + "mkdirp": "^1.0.3", + "p-map": "^4.0.0", + "promise-inflight": "^1.0.1", + "rimraf": "^3.0.2", + "ssri": "^8.0.0", + "tar": "^6.0.2", + "unique-filename": "^1.1.1" + } + }, + "chownr": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/chownr/-/chownr-2.0.0.tgz", + "integrity": "sha512-bIomtDF5KGpdogkLd9VspvFzk9KfpyyGlS8YFVZl7TGPBHL5snIOnxeshwVgPteQ9b4Eydl+pVbIyE1DcvCWgQ==", + "dev": true + }, + "fs-minipass": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/fs-minipass/-/fs-minipass-2.1.0.tgz", + "integrity": "sha512-V/JgOLFCS+R6Vcq0slCuaeWEdNC3ouDlJMNIsacH2VtALiu9mV4LPrHc5cDl8k5aw6J8jwgWWpiTo5RYhmIzvg==", + "dev": true, + "requires": { + "minipass": "^3.0.0" + } + }, + "glob": { + "version": "7.1.6", + "resolved": "https://registry.npmjs.org/glob/-/glob-7.1.6.tgz", + "integrity": "sha512-LwaxwyZ72Lk7vZINtNNrywX0ZuLyStrdDtabefZKAY5ZGJhVtgdznluResxNmPitE0SAO+O26sWTHeKSI2wMBA==", + "dev": true, + "requires": { + "fs.realpath": "^1.0.0", + "inflight": "^1.0.4", + "inherits": "2", + "minimatch": "^3.0.4", + "once": "^1.3.0", + "path-is-absolute": "^1.0.0" + } + }, + "minipass": { + "version": "3.1.3", + "resolved": "https://registry.npmjs.org/minipass/-/minipass-3.1.3.tgz", + "integrity": "sha512-Mgd2GdMVzY+x3IJ+oHnVM+KG3lA5c8tnabyJKmHSaG2kAGpudxuOf8ToDkhumF7UzME7DecbQE9uOZhNm7PuJg==", + "dev": true, + "requires": { + "yallist": "^4.0.0" + } + }, + "minizlib": { + "version": "2.1.2", + "resolved": "https://registry.npmjs.org/minizlib/-/minizlib-2.1.2.tgz", + "integrity": "sha512-bAxsR8BVfj60DWXHE3u30oHzfl4G7khkSuPW+qvpd7jFRHm7dLxOjUk1EHACJ/hxLY8phGJ0YhYHZo7jil7Qdg==", + "dev": true, + "requires": { + "minipass": "^3.0.0", + "yallist": "^4.0.0" + } + }, + "mkdirp": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/mkdirp/-/mkdirp-1.0.4.tgz", + "integrity": "sha512-vVqVZQyf3WLx2Shd0qJ9xuvqgAyKPLAiqITEtqW0oIUjzo3PePDd6fW9iFz30ef7Ysp/oiWqbhszeGWW2T6Gzw==", + "dev": true + }, + "p-limit": { + "version": "3.0.2", + "resolved": "https://registry.npmjs.org/p-limit/-/p-limit-3.0.2.tgz", + "integrity": "sha512-iwqZSOoWIW+Ew4kAGUlN16J4M7OB3ysMLSZtnhmqx7njIHFPlxWBX8xo3lVTyFVq6mI/lL9qt2IsN1sHwaxJkg==", + "dev": true, + "requires": { + "p-try": "^2.0.0" + } + }, + "serialize-javascript": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/serialize-javascript/-/serialize-javascript-4.0.0.tgz", + "integrity": "sha512-GaNA54380uFefWghODBWEGisLZFj00nS5ACs6yHa9nLqlLpVLO8ChDGeKRjZnV4Nh4n0Qi7nhYZD/9fCPzEqkw==", + "dev": true, + "requires": { + "randombytes": "^2.1.0" + } + }, + "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 + }, + "ssri": { + "version": "8.0.0", + "resolved": "https://registry.npmjs.org/ssri/-/ssri-8.0.0.tgz", + "integrity": "sha512-aq/pz989nxVYwn16Tsbj1TqFpD5LLrQxHf5zaHuieFV+R0Bbr4y8qUsOA45hXT/N4/9UNXTarBjnjVmjSOVaAA==", + "dev": true, + "requires": { + "minipass": "^3.1.1" + } + }, + "tar": { + "version": "6.0.5", + "resolved": "https://registry.npmjs.org/tar/-/tar-6.0.5.tgz", + "integrity": "sha512-0b4HOimQHj9nXNEAA7zWwMM91Zhhba3pspja6sQbgTpynOJf+bkjBnfybNYzbpLbnwXnbyB4LOREvlyXLkCHSg==", + "dev": true, + "requires": { + "chownr": "^2.0.0", + "fs-minipass": "^2.0.0", + "minipass": "^3.0.0", + "minizlib": "^2.1.1", + "mkdirp": "^1.0.3", + "yallist": "^4.0.0" + } + } + } + }, + "test": { + "version": "0.6.0", + "resolved": "https://registry.npmjs.org/test/-/test-0.6.0.tgz", + "integrity": "sha1-WYasRF7Bd1QyJRLRBLoyyKY+k44=", + "dev": true, + "requires": { + "ansi-font": "0.0.2" + } + }, + "test-exclude": { + "version": "6.0.0", + "resolved": "https://registry.npmjs.org/test-exclude/-/test-exclude-6.0.0.tgz", + "integrity": "sha512-cAGWPIyOHU6zlmg88jwm7VRyXnMN7iV68OGAbYDk/Mh/xC/pzVPlQtY6ngoIH/5/tciuhGfvESU8GrHrcxD56w==", + "dev": true, + "requires": { + "@istanbuljs/schema": "^0.1.2", + "glob": "^7.1.4", + "minimatch": "^3.0.4" + }, + "dependencies": { + "glob": { + "version": "7.1.6", + "resolved": "https://registry.npmjs.org/glob/-/glob-7.1.6.tgz", + "integrity": "sha512-LwaxwyZ72Lk7vZINtNNrywX0ZuLyStrdDtabefZKAY5ZGJhVtgdznluResxNmPitE0SAO+O26sWTHeKSI2wMBA==", + "dev": true, + "requires": { + "fs.realpath": "^1.0.0", + "inflight": "^1.0.4", + "inherits": "2", + "minimatch": "^3.0.4", + "once": "^1.3.0", + "path-is-absolute": "^1.0.0" + } + } + } + }, + "text-hex": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/text-hex/-/text-hex-1.0.0.tgz", + "integrity": "sha512-uuVGNWzgJ4yhRaNSiubPY7OjISw4sw4E5Uv0wbjp+OzcbmVU/rsT8ujgcXJhn9ypzsgr5vlzpPqP+MBBKcGvbg==", + "dev": true + }, + "text-table": { + "version": "0.2.0", + "resolved": "https://registry.npmjs.org/text-table/-/text-table-0.2.0.tgz", + "integrity": "sha1-f17oI66AUgfACvLfSoTsP8+lcLQ=", + "dev": true + }, + "textextensions": { + "version": "2.6.0", + "resolved": "https://registry.npmjs.org/textextensions/-/textextensions-2.6.0.tgz", + "integrity": "sha512-49WtAWS+tcsy93dRt6P0P3AMD2m5PvXRhuEA0kaXos5ZLlujtYmpmFsB+QvWUSxE1ZsstmYXfQ7L40+EcQgpAQ==", + "dev": true + }, + "tfunk": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/tfunk/-/tfunk-4.0.0.tgz", + "integrity": "sha512-eJQ0dGfDIzWNiFNYFVjJ+Ezl/GmwHaFTBTjrtqNPW0S7cuVDBrZrmzUz6VkMeCR4DZFqhd4YtLwsw3i2wYHswQ==", + "dev": true, + "requires": { + "chalk": "^1.1.3", + "dlv": "^1.1.3" + }, + "dependencies": { + "ansi-regex": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-2.1.1.tgz", + "integrity": "sha1-w7M6te42DYbg5ijwRorn7yfWVN8=", + "dev": true + }, + "ansi-styles": { + "version": "2.2.1", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-2.2.1.tgz", + "integrity": "sha1-tDLdM1i2NM914eRmQ2gkBTPB3b4=", + "dev": true + }, + "chalk": { + "version": "1.1.3", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-1.1.3.tgz", + "integrity": "sha1-qBFcVeSnAv5NFQq9OHKCKn4J/Jg=", + "dev": true, + "requires": { + "ansi-styles": "^2.2.1", + "escape-string-regexp": "^1.0.2", + "has-ansi": "^2.0.0", + "strip-ansi": "^3.0.0", + "supports-color": "^2.0.0" + } + }, + "strip-ansi": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-3.0.1.tgz", + "integrity": "sha1-ajhfuIU9lS1f8F0Oiq+UJ43GPc8=", + "dev": true, + "requires": { + "ansi-regex": "^2.0.0" + } + }, + "supports-color": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-2.0.0.tgz", + "integrity": "sha1-U10EXOa2Nj+kARcIRimZXp3zJMc=", + "dev": true + } + } + }, + "then-request": { + "version": "6.0.2", + "resolved": "https://registry.npmjs.org/then-request/-/then-request-6.0.2.tgz", + "integrity": "sha512-3ZBiG7JvP3wbDzA9iNY5zJQcHL4jn/0BWtXIkagfz7QgOL/LqjCEOBQuJNZfu0XYnv5JhKh+cDxCPM4ILrqruA==", + "dev": true, + "requires": { + "@types/concat-stream": "^1.6.0", + "@types/form-data": "0.0.33", + "@types/node": "^8.0.0", + "@types/qs": "^6.2.31", + "caseless": "~0.12.0", + "concat-stream": "^1.6.0", + "form-data": "^2.2.0", + "http-basic": "^8.1.1", + "http-response-object": "^3.0.1", + "promise": "^8.0.0", + "qs": "^6.4.0" + }, + "dependencies": { + "@types/node": { + "version": "8.10.66", + "resolved": "https://registry.npmjs.org/@types/node/-/node-8.10.66.tgz", + "integrity": "sha512-tktOkFUA4kXx2hhhrB8bIFb5TbwzS4uOhKEmwiD+NoiL0qtP2OQ9mFldbgD4dV1djrlBYP6eBuQZiWjuHUpqFw==", + "dev": true + } + } + }, + "thread-loader": { + "version": "2.1.3", + "resolved": "https://registry.npmjs.org/thread-loader/-/thread-loader-2.1.3.tgz", + "integrity": "sha512-wNrVKH2Lcf8ZrWxDF/khdlLlsTMczdcwPA9VEK4c2exlEPynYWxi9op3nPTo5lAnDIkE0rQEB3VBP+4Zncc9Hg==", + "dev": true, + "requires": { + "loader-runner": "^2.3.1", + "loader-utils": "^1.1.0", + "neo-async": "^2.6.0" + }, + "dependencies": { + "json5": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/json5/-/json5-1.0.1.tgz", + "integrity": "sha512-aKS4WQjPenRxiQsC93MNfjx+nbF4PAdYzmd/1JIj8HYzqfbu86beTuNgXDzPknWk0n0uARlyewZo4s++ES36Ow==", + "dev": true, + "requires": { + "minimist": "^1.2.0" + } + }, + "loader-utils": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/loader-utils/-/loader-utils-1.4.0.tgz", + "integrity": "sha512-qH0WSMBtn/oHuwjy/NucEgbx5dbxxnxup9s4PVXJUDHZBQY+s0NWA9rJf53RBnQZxfch7euUui7hpoAPvALZdA==", + "dev": true, + "requires": { + "big.js": "^5.2.2", + "emojis-list": "^3.0.0", + "json5": "^1.0.1" + } + } + } + }, + "throat": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/throat/-/throat-5.0.0.tgz", + "integrity": "sha512-fcwX4mndzpLQKBS1DVYhGAcYaYt7vsHNIvQV+WXMvnow5cgjPphq5CaayLaGsjRdSCKZFNGt7/GYAuXaNOiYCA==", + "dev": true + }, + "through": { + "version": "2.3.8", + "resolved": "https://registry.npmjs.org/through/-/through-2.3.8.tgz", + "integrity": "sha1-DdTJ/6q8NXlgsbckEV1+Doai4fU=", + "dev": true + }, + "through2": { + "version": "2.0.5", + "resolved": "https://registry.npmjs.org/through2/-/through2-2.0.5.tgz", + "integrity": "sha512-/mrRod8xqpA+IHSLyGCQ2s8SPHiCDEeQJSep1jqLYeEUClOFG2Qsh+4FU6G9VeqpZnGW/Su8LQGc4YKni5rYSQ==", + "dev": true, + "requires": { + "readable-stream": "~2.3.6", + "xtend": "~4.0.1" + } + }, + "thunky": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/thunky/-/thunky-1.1.0.tgz", + "integrity": "sha512-eHY7nBftgThBqOyHGVN+l8gF0BucP09fMo0oO/Lb0w1OF80dJv+lDVpXG60WMQvkcxAkNybKsrEIE3ZtKGmPrA==", + "dev": true + }, + "timed-out": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/timed-out/-/timed-out-4.0.1.tgz", + "integrity": "sha1-8y6srFoXW+ol1/q1Zas+2HQe9W8=", + "dev": true + }, + "timers-browserify": { + "version": "2.0.12", + "resolved": "https://registry.npmjs.org/timers-browserify/-/timers-browserify-2.0.12.tgz", + "integrity": "sha512-9phl76Cqm6FhSX9Xe1ZUAMLtm1BLkKj2Qd5ApyWkXzsMRaA7dgr81kf4wJmQf/hAvg8EEyJxDo3du/0KlhPiKQ==", + "dev": true, + "requires": { + "setimmediate": "^1.0.4" + } + }, + "timsort": { + "version": "0.3.0", + "resolved": "https://registry.npmjs.org/timsort/-/timsort-0.3.0.tgz", + "integrity": "sha1-QFQRqOfmM5/mTbmiNN4R3DHgK9Q=", + "dev": true + }, + "tmp": { + "version": "0.0.33", + "resolved": "https://registry.npmjs.org/tmp/-/tmp-0.0.33.tgz", + "integrity": "sha512-jRCJlojKnZ3addtTOjdIqoRuPEKBvNXcGYqzO6zWZX8KfKEpnGY5jfggJQ3EjKuu8D4bJRr0y+cYJFmYbImXGw==", + "dev": true, + "requires": { + "os-tmpdir": "~1.0.2" + } + }, + "tmpl": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/tmpl/-/tmpl-1.0.4.tgz", + "integrity": "sha1-I2QN17QtAEM5ERQIIOXPRA5SHdE=", + "dev": true + }, + "to-array": { + "version": "0.1.4", + "resolved": "https://registry.npmjs.org/to-array/-/to-array-0.1.4.tgz", + "integrity": "sha1-F+bBH3PdTz10zaek/zI46a2b+JA=", + "dev": true + }, + "to-arraybuffer": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/to-arraybuffer/-/to-arraybuffer-1.0.1.tgz", + "integrity": "sha1-fSKbH8xjfkZsoIEYCDanqr/4P0M=", + "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": "sha1-3F5pjL0HkmW8c+A3doGk5Og/YW4=" + }, + "to-object-path": { + "version": "0.3.0", + "resolved": "https://registry.npmjs.org/to-object-path/-/to-object-path-0.3.0.tgz", + "integrity": "sha1-KXWIt7Dn4KwI4E5nL4XB9JmeF68=", + "dev": true, + "requires": { + "kind-of": "^3.0.2" + }, + "dependencies": { + "kind-of": { + "version": "3.2.2", + "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-3.2.2.tgz", + "integrity": "sha1-MeohpzS6ubuw8yRm2JOupR5KPGQ=", + "dev": true, + "requires": { + "is-buffer": "^1.1.5" + } + } + } + }, + "to-regex": { + "version": "3.0.2", + "resolved": "https://registry.npmjs.org/to-regex/-/to-regex-3.0.2.tgz", + "integrity": "sha512-FWtleNAtZ/Ki2qtqej2CXTOayOH9bHDQF+Q48VpWyDXjbYxA4Yz8iDB31zXOBUlOHHKidDbqGVrTUvQMPmBGBw==", + "dev": true, + "requires": { + "define-property": "^2.0.2", + "extend-shallow": "^3.0.2", + "regex-not": "^1.0.2", + "safe-regex": "^1.1.0" + } + }, + "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" + } + }, + "to-string-loader": { + "version": "1.1.6", + "resolved": "https://registry.npmjs.org/to-string-loader/-/to-string-loader-1.1.6.tgz", + "integrity": "sha512-VNg62//PS1WfNwrK3n7t6wtK5Vdtx/qeYLLEioW46VMlYUwAYT6wnfB+OwS2FMTCalIHu0tk79D3RXX8ttmZTQ==", + "dev": true, + "requires": { + "loader-utils": "^1.0.0" + }, + "dependencies": { + "json5": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/json5/-/json5-1.0.1.tgz", + "integrity": "sha512-aKS4WQjPenRxiQsC93MNfjx+nbF4PAdYzmd/1JIj8HYzqfbu86beTuNgXDzPknWk0n0uARlyewZo4s++ES36Ow==", + "dev": true, + "requires": { + "minimist": "^1.2.0" + } + }, + "loader-utils": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/loader-utils/-/loader-utils-1.4.0.tgz", + "integrity": "sha512-qH0WSMBtn/oHuwjy/NucEgbx5dbxxnxup9s4PVXJUDHZBQY+s0NWA9rJf53RBnQZxfch7euUui7hpoAPvALZdA==", + "dev": true, + "requires": { + "big.js": "^5.2.2", + "emojis-list": "^3.0.0", + "json5": "^1.0.1" + } + } + } + }, + "toidentifier": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/toidentifier/-/toidentifier-1.0.0.tgz", + "integrity": "sha512-yaOH/Pk/VEhBWWTlhI+qXxDFXlejDGcQipMlyxda9nthulaxLZUNcUqFxokp0vcYnvteJln5FNQDRrxj3YcbVw==", + "dev": true + }, + "toposort": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/toposort/-/toposort-2.0.2.tgz", + "integrity": "sha1-riF2gXXRVZ1IvvNUILL0li8JwzA=", + "dev": true + }, + "tough-cookie": { + "version": "2.5.0", + "resolved": "https://registry.npmjs.org/tough-cookie/-/tough-cookie-2.5.0.tgz", + "integrity": "sha512-nlLsUzgm1kfLXSXfRZMc1KLAugd4hqJHDTvc2hDIwS3mZAfMEuMbc03SujMF+GEcpaX/qboeycw6iO8JwVv2+g==", + "dev": true, + "requires": { + "psl": "^1.1.28", + "punycode": "^2.1.1" + } + }, + "tr46": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/tr46/-/tr46-2.0.2.tgz", + "integrity": "sha512-3n1qG+/5kg+jrbTzwAykB5yRYtQCTqOGKq5U5PE3b0a1/mzo6snDhjGS0zJVJunO0NrT3Dg1MLy5TjWP/UJppg==", + "dev": true, + "requires": { + "punycode": "^2.1.1" + } + }, + "trim-newlines": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/trim-newlines/-/trim-newlines-3.0.0.tgz", + "integrity": "sha512-C4+gOpvmxaSMKuEf9Qc134F1ZuOHVXKRbtEflf4NTtuuJDEIJ9p5PXsalL8SkeRw+qit1Mo+yuvMPAKwWg/1hA==", + "dev": true + }, + "triple-beam": { + "version": "1.3.0", + "resolved": "https://registry.npmjs.org/triple-beam/-/triple-beam-1.3.0.tgz", + "integrity": "sha512-XrHUvV5HpdLmIj4uVMxHggLbFSZYIn7HEWsqePZcI50pco+MPqJ50wMGY794X7AOOhxOBAjbkqfAbEe/QMp2Lw==", + "dev": true + }, + "tryer": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/tryer/-/tryer-1.0.1.tgz", + "integrity": "sha512-c3zayb8/kWWpycWYg87P71E1S1ZL6b6IJxfb5fvsUgsf0S2MVGaDhDXXjDMpdCpfWXqptc+4mXwmiy1ypXqRAA==", + "dev": true + }, + "ts-jest": { + "version": "26.4.4", + "resolved": "https://registry.npmjs.org/ts-jest/-/ts-jest-26.4.4.tgz", + "integrity": "sha512-3lFWKbLxJm34QxyVNNCgXX1u4o/RV0myvA2y2Bxm46iGIjKlaY0own9gIckbjZJPn+WaJEnfPPJ20HHGpoq4yg==", + "dev": true, + "requires": { + "@types/jest": "26.x", + "bs-logger": "0.x", + "buffer-from": "1.x", + "fast-json-stable-stringify": "2.x", + "jest-util": "^26.1.0", + "json5": "2.x", + "lodash.memoize": "4.x", + "make-error": "1.x", + "mkdirp": "1.x", + "semver": "7.x", + "yargs-parser": "20.x" + }, + "dependencies": { + "mkdirp": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/mkdirp/-/mkdirp-1.0.4.tgz", + "integrity": "sha512-vVqVZQyf3WLx2Shd0qJ9xuvqgAyKPLAiqITEtqW0oIUjzo3PePDd6fW9iFz30ef7Ysp/oiWqbhszeGWW2T6Gzw==", + "dev": true + }, + "semver": { + "version": "7.3.2", + "resolved": "https://registry.npmjs.org/semver/-/semver-7.3.2.tgz", + "integrity": "sha512-OrOb32TeeambH6UrhtShmF7CRDqhL6/5XpPNp2DuRH6+9QLw/orhp72j87v8Qa1ScDkvrrBNpZcDejAirJmfXQ==", + "dev": true + }, + "yargs-parser": { + "version": "20.2.4", + "resolved": "https://registry.npmjs.org/yargs-parser/-/yargs-parser-20.2.4.tgz", + "integrity": "sha512-WOkpgNhPTlE73h4VFAFsOnomJVaovO8VqLDzy5saChRBFQFBoMYirowyW+Q9HB4HFF4Z7VZTiG3iSzJJA29yRA==", + "dev": true + } + } + }, + "ts-loader": { + "version": "7.0.5", + "resolved": "https://registry.npmjs.org/ts-loader/-/ts-loader-7.0.5.tgz", + "integrity": "sha512-zXypEIT6k3oTc+OZNx/cqElrsbBtYqDknf48OZos0NQ3RTt045fBIU8RRSu+suObBzYB355aIPGOe/3kj9h7Ig==", + "dev": true, + "requires": { + "chalk": "^2.3.0", + "enhanced-resolve": "^4.0.0", + "loader-utils": "^1.0.2", + "micromatch": "^4.0.0", + "semver": "^6.0.0" + }, + "dependencies": { + "json5": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/json5/-/json5-1.0.1.tgz", + "integrity": "sha512-aKS4WQjPenRxiQsC93MNfjx+nbF4PAdYzmd/1JIj8HYzqfbu86beTuNgXDzPknWk0n0uARlyewZo4s++ES36Ow==", + "dev": true, + "requires": { + "minimist": "^1.2.0" + } + }, + "loader-utils": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/loader-utils/-/loader-utils-1.4.0.tgz", + "integrity": "sha512-qH0WSMBtn/oHuwjy/NucEgbx5dbxxnxup9s4PVXJUDHZBQY+s0NWA9rJf53RBnQZxfch7euUui7hpoAPvALZdA==", + "dev": true, + "requires": { + "big.js": "^5.2.2", + "emojis-list": "^3.0.0", + "json5": "^1.0.1" + } + }, + "micromatch": { + "version": "4.0.2", + "resolved": "https://registry.npmjs.org/micromatch/-/micromatch-4.0.2.tgz", + "integrity": "sha512-y7FpHSbMUMoyPbYUSzO6PaZ6FyRnQOpHuKwbo1G+Knck95XVU4QAiKdGEnj5wwoS7PlOgthX/09u5iFJ+aYf5Q==", + "dev": true, + "requires": { + "braces": "^3.0.1", + "picomatch": "^2.0.5" + } + }, + "semver": { + "version": "6.3.0", + "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.0.tgz", + "integrity": "sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw==", + "dev": true + } + } + }, + "tslib": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/tslib/-/tslib-2.0.0.tgz", + "integrity": "sha512-lTqkx847PI7xEDYJntxZH89L2/aXInsyF2luSafe/+0fHOMjlBNXdH6th7f70qxLDhul7KZK0zC8V5ZIyHl0/g==" + }, + "tslint": { + "version": "6.1.2", + "resolved": "https://registry.npmjs.org/tslint/-/tslint-6.1.2.tgz", + "integrity": "sha512-UyNrLdK3E0fQG/xWNqAFAC5ugtFyPO4JJR1KyyfQAyzR8W0fTRrC91A8Wej4BntFzcvETdCSDa/4PnNYJQLYiA==", + "dev": true, + "requires": { + "@babel/code-frame": "^7.0.0", + "builtin-modules": "^1.1.1", + "chalk": "^2.3.0", + "commander": "^2.12.1", + "diff": "^4.0.1", + "glob": "^7.1.1", + "js-yaml": "^3.13.1", + "minimatch": "^3.0.4", + "mkdirp": "^0.5.3", + "resolve": "^1.3.2", + "semver": "^5.3.0", + "tslib": "^1.10.0", + "tsutils": "^2.29.0" + }, + "dependencies": { + "diff": { + "version": "4.0.2", + "resolved": "https://registry.npmjs.org/diff/-/diff-4.0.2.tgz", + "integrity": "sha512-58lmxKSA4BNyLz+HHMUzlOEpg09FV+ev6ZMe3vJihgdxzgcwZ8VoEEPmALCZG9LmqfVoNMMKpttIYTVG6uDY7A==", + "dev": true + }, + "tslib": { + "version": "1.14.1", + "resolved": "https://registry.npmjs.org/tslib/-/tslib-1.14.1.tgz", + "integrity": "sha512-Xni35NKzjgMrwevysHTCArtLDpPvye8zV/0E4EyYn43P7/7qvQwPh9BGkHewbMulVntbigmcT7rdX3BNo9wRJg==", + "dev": true + }, + "tsutils": { + "version": "2.29.0", + "resolved": "https://registry.npmjs.org/tsutils/-/tsutils-2.29.0.tgz", + "integrity": "sha512-g5JVHCIJwzfISaXpXE1qvNalca5Jwob6FjI4AoPlqMusJ6ftFE7IkkFoMhVLRgK+4Kx3gkzb8UZK5t5yTTvEmA==", + "dev": true, + "requires": { + "tslib": "^1.8.1" + } + } + } + }, + "tsutils": { + "version": "3.17.1", + "resolved": "https://registry.npmjs.org/tsutils/-/tsutils-3.17.1.tgz", + "integrity": "sha512-kzeQ5B8H3w60nFY2g8cJIuH7JDpsALXySGtwGJ0p2LSjLgay3NdIpqq5SoOBe46bKDW2iq25irHCr8wjomUS2g==", + "dev": true, + "requires": { + "tslib": "^1.8.1" + }, + "dependencies": { + "tslib": { + "version": "1.14.1", + "resolved": "https://registry.npmjs.org/tslib/-/tslib-1.14.1.tgz", + "integrity": "sha512-Xni35NKzjgMrwevysHTCArtLDpPvye8zV/0E4EyYn43P7/7qvQwPh9BGkHewbMulVntbigmcT7rdX3BNo9wRJg==", + "dev": true + } + } + }, + "tty-browserify": { + "version": "0.0.0", + "resolved": "https://registry.npmjs.org/tty-browserify/-/tty-browserify-0.0.0.tgz", + "integrity": "sha1-oVe6QC2iTpv5V/mqadUk7tQpAaY=", + "dev": true + }, + "tunnel-agent": { + "version": "0.6.0", + "resolved": "https://registry.npmjs.org/tunnel-agent/-/tunnel-agent-0.6.0.tgz", + "integrity": "sha1-J6XeoGs2sEoKmWZ3SykIaPD8QP0=", + "dev": true, + "requires": { + "safe-buffer": "^5.0.1" + } + }, + "tweetnacl": { + "version": "0.14.5", + "resolved": "https://registry.npmjs.org/tweetnacl/-/tweetnacl-0.14.5.tgz", + "integrity": "sha1-WuaBd/GS1EViadEIr6k/+HQ/T2Q=", + "dev": true + }, + "type-check": { + "version": "0.3.2", + "resolved": "https://registry.npmjs.org/type-check/-/type-check-0.3.2.tgz", + "integrity": "sha1-WITKtRLPHTVeP7eE8wgEsrUg23I=", + "dev": true, + "requires": { + "prelude-ls": "~1.1.2" + } + }, + "type-detect": { + "version": "4.0.8", + "resolved": "https://registry.npmjs.org/type-detect/-/type-detect-4.0.8.tgz", + "integrity": "sha512-0fr/mIH1dlO+x7TlcMy+bIDqKPsw/70tVyeHW787goQjhmqaZe10uwLujubK9q9Lg6Fiho1KUKDYz0Z7k7g5/g==", + "dev": true + }, + "type-fest": { + "version": "0.11.0", + "resolved": "https://registry.npmjs.org/type-fest/-/type-fest-0.11.0.tgz", + "integrity": "sha512-OdjXJxnCN1AvyLSzeKIgXTXxV+99ZuXl3Hpo9XpJAv9MBcHrrJOQ5kV7ypXOuQie+AmWG25hLbiKdwYTifzcfQ==", + "dev": true + }, + "type-is": { + "version": "1.6.18", + "resolved": "https://registry.npmjs.org/type-is/-/type-is-1.6.18.tgz", + "integrity": "sha512-TkRKr9sUTxEH8MdfuCSP7VizJyzRNMjj2J2do2Jr3Kym598JVdEksuzPQCnlFPW4ky9Q+iA+ma9BGm06XQBy8g==", + "dev": true, + "requires": { + "media-typer": "0.3.0", + "mime-types": "~2.1.24" + } + }, + "typedarray": { + "version": "0.0.6", + "resolved": "https://registry.npmjs.org/typedarray/-/typedarray-0.0.6.tgz", + "integrity": "sha1-hnrHTjhkGHsdPUfZlqeOxciDB3c=", + "dev": true + }, + "typedarray-to-buffer": { + "version": "3.1.5", + "resolved": "https://registry.npmjs.org/typedarray-to-buffer/-/typedarray-to-buffer-3.1.5.tgz", + "integrity": "sha512-zdu8XMNEDepKKR+XYOXAVPtWui0ly0NtohUscw+UmaHiAWT8hrV1rr//H6V+0DvJ3OQ19S979M0laLfX8rm82Q==", + "dev": true, + "requires": { + "is-typedarray": "^1.0.0" + } + }, + "typescript": { + "version": "3.9.5", + "resolved": "https://registry.npmjs.org/typescript/-/typescript-3.9.5.tgz", + "integrity": "sha512-hSAifV3k+i6lEoCJ2k6R2Z/rp/H3+8sdmcn5NrS3/3kE7+RyZXm9aqvxWqjEXHAd8b0pShatpcdMTvEdvAJltQ==", + "dev": true + }, + "ua-parser-js": { + "version": "0.7.17", + "resolved": "https://registry.npmjs.org/ua-parser-js/-/ua-parser-js-0.7.17.tgz", + "integrity": "sha512-uRdSdu1oA1rncCQL7sCj8vSyZkgtL7faaw9Tc9rZ3mGgraQ7+Pdx7w5mnOSF3gw9ZNG6oc+KXfkon3bKuROm0g==", + "dev": true + }, + "ultron": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/ultron/-/ultron-1.1.1.tgz", + "integrity": "sha512-UIEXBNeYmKptWH6z8ZnqTeS8fV74zG0/eRU9VGkpzz+LIJNs8W/zM/L+7ctCkRrgbNnnR0xxw4bKOr0cW0N0Og==", + "dev": true + }, + "unicode-canonical-property-names-ecmascript": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/unicode-canonical-property-names-ecmascript/-/unicode-canonical-property-names-ecmascript-1.0.4.tgz", + "integrity": "sha512-jDrNnXWHd4oHiTZnx/ZG7gtUTVp+gCcTTKr8L0HjlwphROEW3+Him+IpvC+xcJEFegapiMZyZe02CyuOnRmbnQ==", + "dev": true + }, + "unicode-match-property-ecmascript": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/unicode-match-property-ecmascript/-/unicode-match-property-ecmascript-1.0.4.tgz", + "integrity": "sha512-L4Qoh15vTfntsn4P1zqnHulG0LdXgjSO035fEpdtp6YxXhMT51Q6vgM5lYdG/5X3MjS+k/Y9Xw4SFCY9IkR0rg==", + "dev": true, + "requires": { + "unicode-canonical-property-names-ecmascript": "^1.0.4", + "unicode-property-aliases-ecmascript": "^1.0.4" + } + }, + "unicode-match-property-value-ecmascript": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/unicode-match-property-value-ecmascript/-/unicode-match-property-value-ecmascript-1.2.0.tgz", + "integrity": "sha512-wjuQHGQVofmSJv1uVISKLE5zO2rNGzM/KCYZch/QQvez7C1hUhBIuZ701fYXExuufJFMPhv2SyL8CyoIfMLbIQ==", + "dev": true + }, + "unicode-property-aliases-ecmascript": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/unicode-property-aliases-ecmascript/-/unicode-property-aliases-ecmascript-1.1.0.tgz", + "integrity": "sha512-PqSoPh/pWetQ2phoj5RLiaqIk4kCNwoV3CI+LfGmWLKI3rE3kl1h59XpX2BjgDrmbxD9ARtQobPGU1SguCYuQg==", + "dev": true + }, + "union-value": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/union-value/-/union-value-1.0.1.tgz", + "integrity": "sha512-tJfXmxMeWYnczCVs7XAEvIV7ieppALdyepWMkHkwciRpZraG/xwT+s2JN8+pr1+8jCRf80FFzvr+MpQeeoF4Xg==", + "dev": true, + "requires": { + "arr-union": "^3.1.0", + "get-value": "^2.0.6", + "is-extendable": "^0.1.1", + "set-value": "^2.0.1" + } + }, + "uniq": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/uniq/-/uniq-1.0.1.tgz", + "integrity": "sha1-sxxa6CVIRKOoKBVBzisEuGWnNP8=", + "dev": true + }, + "uniqs": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/uniqs/-/uniqs-2.0.0.tgz", + "integrity": "sha1-/+3ks2slKQaW5uFl1KWe25mOawI=", + "dev": true + }, + "unique-filename": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/unique-filename/-/unique-filename-1.1.1.tgz", + "integrity": "sha512-Vmp0jIp2ln35UTXuryvjzkjGdRyf9b2lTXuSYUiPmzRcl3FDtYqAwOnTJkAngD9SWhnoJzDbTKwaOrZ+STtxNQ==", + "dev": true, + "requires": { + "unique-slug": "^2.0.0" + } + }, + "unique-slug": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/unique-slug/-/unique-slug-2.0.2.tgz", + "integrity": "sha512-zoWr9ObaxALD3DOPfjPSqxt4fnZiWblxHIgeWqW8x7UqDzEtHEQLzji2cuJYQFCU6KmoJikOYAZlrTHHebjx2w==", + "dev": true, + "requires": { + "imurmurhash": "^0.1.4" + } + }, + "unique-string": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/unique-string/-/unique-string-1.0.0.tgz", + "integrity": "sha1-nhBXzKhRq7kzmPizOuGHuZyuwRo=", + "dev": true, + "requires": { + "crypto-random-string": "^1.0.0" + } + }, + "universal-analytics": { + "version": "0.4.20", + "resolved": "https://registry.npmjs.org/universal-analytics/-/universal-analytics-0.4.20.tgz", + "integrity": "sha512-gE91dtMvNkjO+kWsPstHRtSwHXz0l2axqptGYp5ceg4MsuurloM0PU3pdOfpb5zBXUvyjT4PwhWK2m39uczZuw==", + "dev": true, + "requires": { + "debug": "^3.0.0", + "request": "^2.88.0", + "uuid": "^3.0.0" + }, + "dependencies": { + "debug": { + "version": "3.2.7", + "resolved": "https://registry.npmjs.org/debug/-/debug-3.2.7.tgz", + "integrity": "sha512-CFjzYYAi4ThfiQvizrFQevTTXHtnCqWfe7x1AhgEscTz6ZbLbfoLRLPugTQyBth6f8ZERVUSyWHFD/7Wu4t1XQ==", + "dev": true, + "requires": { + "ms": "^2.1.1" + } + }, + "uuid": { + "version": "3.4.0", + "resolved": "https://registry.npmjs.org/uuid/-/uuid-3.4.0.tgz", + "integrity": "sha512-HjSDRw6gZE5JMggctHBcjVak08+KEVhSIiDzFnT9S9aegmp85S/bReBVTb4QTFaRNptJ9kuYaNhnbNEOkbKb/A==", + "dev": true + } + } + }, + "universalify": { + "version": "0.1.2", + "resolved": "https://registry.npmjs.org/universalify/-/universalify-0.1.2.tgz", + "integrity": "sha512-rBJeI5CXAlmy1pV+617WB9J63U6XcazHHF2f2dbJix4XzpUF0RS3Zbj0FGIOCAva5P/d/GBOYaACQ1w+0azUkg==", + "dev": true + }, + "unpipe": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/unpipe/-/unpipe-1.0.0.tgz", + "integrity": "sha1-sr9O6FFKrmFltIF4KdIbLvSZBOw=", + "dev": true + }, + "unquote": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/unquote/-/unquote-1.1.1.tgz", + "integrity": "sha1-j97XMk7G6IoP+LkF58CYzcCG1UQ=", + "dev": true + }, + "unset-value": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/unset-value/-/unset-value-1.0.0.tgz", + "integrity": "sha1-g3aHP30jNRef+x5vw6jtDfyKtVk=", + "dev": true, + "requires": { + "has-value": "^0.3.1", + "isobject": "^3.0.0" + }, + "dependencies": { + "has-value": { + "version": "0.3.1", + "resolved": "https://registry.npmjs.org/has-value/-/has-value-0.3.1.tgz", + "integrity": "sha1-ex9YutpiyoJ+wKIHgCVlSEWZXh8=", + "dev": true, + "requires": { + "get-value": "^2.0.3", + "has-values": "^0.1.4", + "isobject": "^2.0.0" + }, + "dependencies": { + "isobject": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/isobject/-/isobject-2.1.0.tgz", + "integrity": "sha1-8GVWEJaj8dou9GJy+BXIQNh+DIk=", + "dev": true, + "requires": { + "isarray": "1.0.0" + } + } + } + }, + "has-values": { + "version": "0.1.4", + "resolved": "https://registry.npmjs.org/has-values/-/has-values-0.1.4.tgz", + "integrity": "sha1-bWHeldkd/Km5oCCJrThL/49it3E=", + "dev": true + } + } + }, + "untildify": { + "version": "3.0.3", + "resolved": "https://registry.npmjs.org/untildify/-/untildify-3.0.3.tgz", + "integrity": "sha512-iSk/J8efr8uPT/Z4eSUywnqyrQU7DSdMfdqK4iWEaUVVmcP5JcnpRqmVMwcwcnmI1ATFNgC5V90u09tBynNFKA==", + "dev": true + }, + "unzip-response": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/unzip-response/-/unzip-response-2.0.1.tgz", + "integrity": "sha1-0vD3N9FrBhXnKmk17QQhRXLVb5c=", + "dev": true + }, + "upath": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/upath/-/upath-1.2.0.tgz", + "integrity": "sha512-aZwGpamFO61g3OlfT7OQCHqhGnW43ieH9WZeP7QxN/G/jS4jfqUkZxoryvJgVPEcrl5NL/ggHsSmLMHuH64Lhg==", + "dev": true + }, + "uri-js": { + "version": "4.4.0", + "resolved": "https://registry.npmjs.org/uri-js/-/uri-js-4.4.0.tgz", + "integrity": "sha512-B0yRTzYdUCCn9n+F4+Gh4yIDtMQcaJsmYBDsTSG8g/OejKBodLQ2IHfN3bM7jUsRXndopT7OIXWdYqc1fjmV6g==", + "dev": true, + "requires": { + "punycode": "^2.1.0" + } + }, + "urix": { + "version": "0.1.0", + "resolved": "https://registry.npmjs.org/urix/-/urix-0.1.0.tgz", + "integrity": "sha1-2pN/emLiH+wf0Y1Js1wpNQZ6bHI=", + "dev": true + }, + "url": { + "version": "0.10.3", + "resolved": "https://registry.npmjs.org/url/-/url-0.10.3.tgz", + "integrity": "sha1-Ah5NnHcF8hu/N9A861h2dAJ3TGQ=", + "dev": true, + "requires": { + "punycode": "1.3.2", + "querystring": "0.2.0" + }, + "dependencies": { + "punycode": { + "version": "1.3.2", + "resolved": "https://registry.npmjs.org/punycode/-/punycode-1.3.2.tgz", + "integrity": "sha1-llOgNvt8HuQjQvIyXM7v6jkmxI0=", + "dev": true + } + } + }, + "url-parse": { + "version": "1.4.7", + "resolved": "https://registry.npmjs.org/url-parse/-/url-parse-1.4.7.tgz", + "integrity": "sha512-d3uaVyzDB9tQoSXFvuSUNFibTd9zxd2bkVrDRvF5TmvWWQwqE4lgYJ5m+x1DbecWkw+LK4RNl2CU1hHuOKPVlg==", + "dev": true, + "requires": { + "querystringify": "^2.1.1", + "requires-port": "^1.0.0" + } + }, + "url-parse-lax": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/url-parse-lax/-/url-parse-lax-1.0.0.tgz", + "integrity": "sha1-evjzA2Rem9eaJy56FKxovAYJ2nM=", + "dev": true, + "requires": { + "prepend-http": "^1.0.1" + } + }, + "use": { + "version": "3.1.1", + "resolved": "https://registry.npmjs.org/use/-/use-3.1.1.tgz", + "integrity": "sha512-cwESVXlO3url9YWlFW/TA9cshCEhtu7IKJ/p5soJ/gGpj7vbvFrAY/eIioQ6Dw23KjZhYgiIo8HOs1nQ2vr/oQ==", + "dev": true + }, + "util": { + "version": "0.11.1", + "resolved": "https://registry.npmjs.org/util/-/util-0.11.1.tgz", + "integrity": "sha512-HShAsny+zS2TZfaXxD9tYj4HQGlBezXZMZuM/S5PKLLoZkShZiGk9o5CzukI1LVHZvjdvZ2Sj1aW/Ndn2NB/HQ==", + "dev": true, + "requires": { + "inherits": "2.0.3" + }, + "dependencies": { + "inherits": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.3.tgz", + "integrity": "sha1-Yzwsg+PaQqUC9SRmAiSA9CCCYd4=", + "dev": true + } + } + }, + "util-deprecate": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/util-deprecate/-/util-deprecate-1.0.2.tgz", + "integrity": "sha1-RQ1Nyfpw3nMnYvvS1KKJgUGaDM8=", + "dev": true + }, + "util-promisify": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/util-promisify/-/util-promisify-2.1.0.tgz", + "integrity": "sha1-PCI2R2xNMsX/PEcAKt18E7moKlM=", + "dev": true, + "requires": { + "object.getownpropertydescriptors": "^2.0.3" + } + }, + "util.promisify": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/util.promisify/-/util.promisify-1.0.0.tgz", + "integrity": "sha512-i+6qA2MPhvoKLuxnJNpXAGhg7HphQOSUq2LKMZD0m15EiskXUkMvKdF4Uui0WYeCUGea+o2cw/ZuwehtfsrNkA==", + "dev": true, + "requires": { + "define-properties": "^1.1.2", + "object.getownpropertydescriptors": "^2.0.3" + } + }, + "utila": { + "version": "0.4.0", + "resolved": "https://registry.npmjs.org/utila/-/utila-0.4.0.tgz", + "integrity": "sha1-ihagXURWV6Oupe7MWxKk+lN5dyw=", + "dev": true + }, + "utils-merge": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/utils-merge/-/utils-merge-1.0.1.tgz", + "integrity": "sha1-n5VxD1CiZ5R7LMwSR0HBAoQn5xM=", + "dev": true + }, + "uuid": { + "version": "8.1.0", + "resolved": "https://registry.npmjs.org/uuid/-/uuid-8.1.0.tgz", + "integrity": "sha512-CI18flHDznR0lq54xBycOVmphdCYnQLKn8abKn7PXUiKUGdEd+/l9LWNJmugXel4hXq7S+RMNl34ecyC9TntWg==", + "dev": true + }, + "v8-compile-cache": { + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/v8-compile-cache/-/v8-compile-cache-2.2.0.tgz", + "integrity": "sha512-gTpR5XQNKFwOd4clxfnhaqvfqMpqEwr4tOtCyz4MtYZX2JYhfr1JvBFKdS+7K/9rfpZR3VLX+YWBbKoxCgS43Q==", + "dev": true + }, + "v8-to-istanbul": { + "version": "7.0.0", + "resolved": "https://registry.npmjs.org/v8-to-istanbul/-/v8-to-istanbul-7.0.0.tgz", + "integrity": "sha512-fLL2rFuQpMtm9r8hrAV2apXX/WqHJ6+IC4/eQVdMDGBUgH/YMV4Gv3duk3kjmyg6uiQWBAA9nJwue4iJUOkHeA==", + "dev": true, + "requires": { + "@types/istanbul-lib-coverage": "^2.0.1", + "convert-source-map": "^1.6.0", + "source-map": "^0.7.3" + }, + "dependencies": { + "source-map": { + "version": "0.7.3", + "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.7.3.tgz", + "integrity": "sha512-CkCj6giN3S+n9qrYiBTX5gystlENnRW5jZeNLHpe6aue+SrHcG5VYwujhW9s4dY31mEGsxBDrHR6oI69fTXsaQ==", + "dev": true + } + } + }, + "validate-npm-package-license": { + "version": "3.0.4", + "resolved": "https://registry.npmjs.org/validate-npm-package-license/-/validate-npm-package-license-3.0.4.tgz", + "integrity": "sha512-DpKm2Ui/xN7/HQKCtpZxoRWBhZ9Z0kqtygG8XCgNQ8ZlDnxuQmWhj566j8fN4Cu3/JmbhsDo7fcAJq4s9h27Ew==", + "dev": true, + "requires": { + "spdx-correct": "^3.0.0", + "spdx-expression-parse": "^3.0.0" + } + }, + "validate-npm-package-name": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/validate-npm-package-name/-/validate-npm-package-name-3.0.0.tgz", + "integrity": "sha1-X6kS2B630MdK/BQN5zF/DKffQ34=", + "dev": true, + "requires": { + "builtins": "^1.0.3" + } + }, + "vary": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/vary/-/vary-1.1.2.tgz", + "integrity": "sha1-IpnwLG3tMNSllhsLn3RSShj2NPw=", + "dev": true + }, + "vendors": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/vendors/-/vendors-1.0.4.tgz", + "integrity": "sha512-/juG65kTL4Cy2su4P8HjtkTxk6VmJDiOPBufWniqQ6wknac6jNiXS9vU+hO3wgusiyqWlzTbVHi0dyJqRONg3w==", + "dev": true + }, + "verror": { + "version": "1.10.0", + "resolved": "https://registry.npmjs.org/verror/-/verror-1.10.0.tgz", + "integrity": "sha1-OhBcoXBTr1XW4nDB+CiGguGNpAA=", + "dev": true, + "requires": { + "assert-plus": "^1.0.0", + "core-util-is": "1.0.2", + "extsprintf": "^1.2.0" + } + }, + "vinyl": { + "version": "2.2.1", + "resolved": "https://registry.npmjs.org/vinyl/-/vinyl-2.2.1.tgz", + "integrity": "sha512-LII3bXRFBZLlezoG5FfZVcXflZgWP/4dCwKtxd5ky9+LOtM4CS3bIRQsmR1KMnMW07jpE8fqR2lcxPZ+8sJIcw==", + "dev": true, + "requires": { + "clone": "^2.1.1", + "clone-buffer": "^1.0.0", + "clone-stats": "^1.0.0", + "cloneable-readable": "^1.0.0", + "remove-trailing-separator": "^1.0.1", + "replace-ext": "^1.0.0" + }, + "dependencies": { + "clone": { + "version": "2.1.2", + "resolved": "https://registry.npmjs.org/clone/-/clone-2.1.2.tgz", + "integrity": "sha1-G39Ln1kfHo+DZwQBYANFoCiHQ18=", + "dev": true + } + } + }, + "vinyl-file": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/vinyl-file/-/vinyl-file-3.0.0.tgz", + "integrity": "sha1-sQTZ5ECf+jJfqt1SBkLQo7SIs2U=", + "dev": true, + "requires": { + "graceful-fs": "^4.1.2", + "pify": "^2.3.0", + "strip-bom-buf": "^1.0.0", + "strip-bom-stream": "^2.0.0", + "vinyl": "^2.0.1" + } + }, + "vm-browserify": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/vm-browserify/-/vm-browserify-1.1.2.tgz", + "integrity": "sha512-2ham8XPWTONajOR0ohOKOHXkm3+gaBmGut3SRuu75xLd/RRaY6vqgh8NBYYk7+RW3u5AtzPQZG8F10LHkl0lAQ==", + "dev": true + }, + "w3c-hr-time": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/w3c-hr-time/-/w3c-hr-time-1.0.2.tgz", + "integrity": "sha512-z8P5DvDNjKDoFIHK7q8r8lackT6l+jo/Ye3HOle7l9nICP9lf1Ci25fy9vHd0JOWewkIFzXIEig3TdKT7JQ5fQ==", + "dev": true, + "requires": { + "browser-process-hrtime": "^1.0.0" + } + }, + "w3c-xmlserializer": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/w3c-xmlserializer/-/w3c-xmlserializer-2.0.0.tgz", + "integrity": "sha512-4tzD0mF8iSiMiNs30BiLO3EpfGLZUT2MSX/G+o7ZywDzliWQ3OPtTZ0PTC3B3ca1UAf4cJMHB+2Bf56EriJuRA==", + "dev": true, + "requires": { + "xml-name-validator": "^3.0.0" + } + }, + "walker": { + "version": "1.0.7", + "resolved": "https://registry.npmjs.org/walker/-/walker-1.0.7.tgz", + "integrity": "sha1-L3+bj9ENZ3JisYqITijRlhjgKPs=", + "dev": true, + "requires": { + "makeerror": "1.0.x" + } + }, + "watchpack": { + "version": "1.7.5", + "resolved": "https://registry.npmjs.org/watchpack/-/watchpack-1.7.5.tgz", + "integrity": "sha512-9P3MWk6SrKjHsGkLT2KHXdQ/9SNkyoJbabxnKOoJepsvJjJG8uYTR3yTPxPQvNDI3w4Nz1xnE0TLHK4RIVe/MQ==", + "dev": true, + "requires": { + "chokidar": "^3.4.1", + "graceful-fs": "^4.1.2", + "neo-async": "^2.5.0", + "watchpack-chokidar2": "^2.0.1" + } + }, + "watchpack-chokidar2": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/watchpack-chokidar2/-/watchpack-chokidar2-2.0.1.tgz", + "integrity": "sha512-nCFfBIPKr5Sh61s4LPpy1Wtfi0HE8isJ3d2Yb5/Ppw2P2B/3eVSEBjKfN0fmHJSK14+31KwMKmcrzs2GM4P0Ww==", + "dev": true, + "optional": true, + "requires": { + "chokidar": "^2.1.8" + }, + "dependencies": { + "anymatch": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/anymatch/-/anymatch-2.0.0.tgz", + "integrity": "sha512-5teOsQWABXHHBFP9y3skS5P3d/WfWXpv3FUpy+LorMrNYaT9pI4oLMQX7jzQ2KklNpGpWHzdCXTDT2Y3XGlZBw==", + "dev": true, + "optional": true, + "requires": { + "micromatch": "^3.1.4", + "normalize-path": "^2.1.1" + }, + "dependencies": { + "normalize-path": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/normalize-path/-/normalize-path-2.1.1.tgz", + "integrity": "sha1-GrKLVW4Zg2Oowab35vogE3/mrtk=", + "dev": true, + "optional": true, + "requires": { + "remove-trailing-separator": "^1.0.1" + } + } + } + }, + "binary-extensions": { + "version": "1.13.1", + "resolved": "https://registry.npmjs.org/binary-extensions/-/binary-extensions-1.13.1.tgz", + "integrity": "sha512-Un7MIEDdUC5gNpcGDV97op1Ywk748MpHcFTHoYs6qnj1Z3j7I53VG3nwZhKzoBZmbdRNnb6WRdFlwl7tSDuZGw==", + "dev": true, + "optional": true + }, + "braces": { + "version": "2.3.2", + "resolved": "https://registry.npmjs.org/braces/-/braces-2.3.2.tgz", + "integrity": "sha512-aNdbnj9P8PjdXU4ybaWLK2IF3jc/EoDYbC7AazW6to3TRsfXxscC9UXOB5iDiEQrkyIbWp2SLQda4+QAa7nc3w==", + "dev": true, + "optional": true, + "requires": { + "arr-flatten": "^1.1.0", + "array-unique": "^0.3.2", + "extend-shallow": "^2.0.1", + "fill-range": "^4.0.0", + "isobject": "^3.0.1", + "repeat-element": "^1.1.2", + "snapdragon": "^0.8.1", + "snapdragon-node": "^2.0.1", + "split-string": "^3.0.2", + "to-regex": "^3.0.1" + } + }, + "chokidar": { + "version": "2.1.8", + "resolved": "https://registry.npmjs.org/chokidar/-/chokidar-2.1.8.tgz", + "integrity": "sha512-ZmZUazfOzf0Nve7duiCKD23PFSCs4JPoYyccjUFF3aQkQadqBhfzhjkwBH2mNOG9cTBwhamM37EIsIkZw3nRgg==", + "dev": true, + "optional": true, + "requires": { + "anymatch": "^2.0.0", + "async-each": "^1.0.1", + "braces": "^2.3.2", + "fsevents": "^1.2.7", + "glob-parent": "^3.1.0", + "inherits": "^2.0.3", + "is-binary-path": "^1.0.0", + "is-glob": "^4.0.0", + "normalize-path": "^3.0.0", + "path-is-absolute": "^1.0.0", + "readdirp": "^2.2.1", + "upath": "^1.1.1" + } + }, + "extend-shallow": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/extend-shallow/-/extend-shallow-2.0.1.tgz", + "integrity": "sha1-Ua99YUrZqfYQ6huvu5idaxxWiQ8=", + "dev": true, + "optional": true, + "requires": { + "is-extendable": "^0.1.0" + } + }, + "fill-range": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/fill-range/-/fill-range-4.0.0.tgz", + "integrity": "sha1-1USBHUKPmOsGpj3EAtJAPDKMOPc=", + "dev": true, + "optional": true, + "requires": { + "extend-shallow": "^2.0.1", + "is-number": "^3.0.0", + "repeat-string": "^1.6.1", + "to-regex-range": "^2.1.0" + } + }, + "fsevents": { + "version": "1.2.13", + "resolved": "https://registry.npmjs.org/fsevents/-/fsevents-1.2.13.tgz", + "integrity": "sha512-oWb1Z6mkHIskLzEJ/XWX0srkpkTQ7vaopMQkyaEIoq0fmtFVxOthb8cCxeT+p3ynTdkk/RZwbgG4brR5BeWECw==", + "dev": true, + "optional": true + }, + "glob-parent": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-3.1.0.tgz", + "integrity": "sha1-nmr2KZ2NO9K9QEMIMr0RPfkGxa4=", + "dev": true, + "optional": true, + "requires": { + "is-glob": "^3.1.0", + "path-dirname": "^1.0.0" + }, + "dependencies": { + "is-glob": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-3.1.0.tgz", + "integrity": "sha1-e6WuJCF4BKxwcHuWkiVnSGzD6Eo=", + "dev": true, + "optional": true, + "requires": { + "is-extglob": "^2.1.0" + } + } + } + }, + "is-binary-path": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/is-binary-path/-/is-binary-path-1.0.1.tgz", + "integrity": "sha1-dfFmQrSA8YenEcgUFh/TpKdlWJg=", + "dev": true, + "optional": true, + "requires": { + "binary-extensions": "^1.0.0" + } + }, + "is-number": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/is-number/-/is-number-3.0.0.tgz", + "integrity": "sha1-JP1iAaR4LPUFYcgQJ2r8fRLXEZU=", + "dev": true, + "optional": true, + "requires": { + "kind-of": "^3.0.2" + } + }, + "kind-of": { + "version": "3.2.2", + "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-3.2.2.tgz", + "integrity": "sha1-MeohpzS6ubuw8yRm2JOupR5KPGQ=", + "dev": true, + "optional": true, + "requires": { + "is-buffer": "^1.1.5" + } + }, + "readdirp": { + "version": "2.2.1", + "resolved": "https://registry.npmjs.org/readdirp/-/readdirp-2.2.1.tgz", + "integrity": "sha512-1JU/8q+VgFZyxwrJ+SVIOsh+KywWGpds3NTqikiKpDMZWScmAYyKIgqkO+ARvNWJfXeXR1zxz7aHF4u4CyH6vQ==", + "dev": true, + "optional": true, + "requires": { + "graceful-fs": "^4.1.11", + "micromatch": "^3.1.10", + "readable-stream": "^2.0.2" + } + }, + "to-regex-range": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/to-regex-range/-/to-regex-range-2.1.1.tgz", + "integrity": "sha1-fIDBe53+vlmeJzZ+DU3VWQFB2zg=", + "dev": true, + "optional": true, + "requires": { + "is-number": "^3.0.0", + "repeat-string": "^1.6.1" + } + } + } + }, + "wbuf": { + "version": "1.7.3", + "resolved": "https://registry.npmjs.org/wbuf/-/wbuf-1.7.3.tgz", + "integrity": "sha512-O84QOnr0icsbFGLS0O3bI5FswxzRr8/gHwWkDlQFskhSPryQXvrTMxjxGP4+iWYoauLoBvfDpkrOauZ+0iZpDA==", + "dev": true, + "requires": { + "minimalistic-assert": "^1.0.0" + } + }, + "wcwidth": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/wcwidth/-/wcwidth-1.0.1.tgz", + "integrity": "sha1-8LDc+RW8X/FSivrbLA4XtTLaL+g=", + "dev": true, + "requires": { + "defaults": "^1.0.3" + } + }, + "webidl-conversions": { + "version": "6.1.0", + "resolved": "https://registry.npmjs.org/webidl-conversions/-/webidl-conversions-6.1.0.tgz", + "integrity": "sha512-qBIvFLGiBpLjfwmYAaHPXsn+ho5xZnGvyGvsarywGNc8VyQJUMHJ8OBKGGrPER0okBeMDaan4mNBlgBROxuI8w==", + "dev": true + }, + "webpack": { + "version": "4.43.0", + "resolved": "https://registry.npmjs.org/webpack/-/webpack-4.43.0.tgz", + "integrity": "sha512-GW1LjnPipFW2Y78OOab8NJlCflB7EFskMih2AHdvjbpKMeDJqEgSx24cXXXiPS65+WSwVyxtDsJH6jGX2czy+g==", + "dev": true, + "requires": { + "@webassemblyjs/ast": "1.9.0", + "@webassemblyjs/helper-module-context": "1.9.0", + "@webassemblyjs/wasm-edit": "1.9.0", + "@webassemblyjs/wasm-parser": "1.9.0", + "acorn": "^6.4.1", + "ajv": "^6.10.2", + "ajv-keywords": "^3.4.1", + "chrome-trace-event": "^1.0.2", + "enhanced-resolve": "^4.1.0", + "eslint-scope": "^4.0.3", + "json-parse-better-errors": "^1.0.2", + "loader-runner": "^2.4.0", + "loader-utils": "^1.2.3", + "memory-fs": "^0.4.1", + "micromatch": "^3.1.10", + "mkdirp": "^0.5.3", + "neo-async": "^2.6.1", + "node-libs-browser": "^2.2.1", + "schema-utils": "^1.0.0", + "tapable": "^1.1.3", + "terser-webpack-plugin": "^1.4.3", + "watchpack": "^1.6.1", + "webpack-sources": "^1.4.1" + }, + "dependencies": { + "acorn": { + "version": "6.4.2", + "resolved": "https://registry.npmjs.org/acorn/-/acorn-6.4.2.tgz", + "integrity": "sha512-XtGIhXwF8YM8bJhGxG5kXgjkEuNGLTkoYqVE+KMR+aspr4KGYmKYg7yUe3KghyQ9yheNwLnjmzh/7+gfDBmHCQ==", + "dev": true + }, + "eslint-scope": { + "version": "4.0.3", + "resolved": "https://registry.npmjs.org/eslint-scope/-/eslint-scope-4.0.3.tgz", + "integrity": "sha512-p7VutNr1O/QrxysMo3E45FjYDTeXBy0iTltPFNSqKAIfjDSXC+4dj+qfyuD8bfAXrW/y6lW3O76VaYNPKfpKrg==", + "dev": true, + "requires": { + "esrecurse": "^4.1.0", + "estraverse": "^4.1.1" + } + }, + "find-cache-dir": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/find-cache-dir/-/find-cache-dir-2.1.0.tgz", + "integrity": "sha512-Tq6PixE0w/VMFfCgbONnkiQIVol/JJL7nRMi20fqzA4NRs9AfeqMGeRdPi3wIhYkxjeBaWh2rxwapn5Tu3IqOQ==", + "dev": true, + "requires": { + "commondir": "^1.0.1", + "make-dir": "^2.0.0", + "pkg-dir": "^3.0.0" + } + }, + "find-up": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/find-up/-/find-up-3.0.0.tgz", + "integrity": "sha512-1yD6RmLI1XBfxugvORwlck6f75tYL+iR0jqwsOrOxMZyGYqUuDhJ0l4AXdO1iX/FTs9cBAMEk1gWSEx1kSbylg==", + "dev": true, + "requires": { + "locate-path": "^3.0.0" + } + }, + "is-wsl": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/is-wsl/-/is-wsl-1.1.0.tgz", + "integrity": "sha1-HxbkqiKwTRM2tmGIpmrzxgDDpm0=", + "dev": true + }, + "json5": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/json5/-/json5-1.0.1.tgz", + "integrity": "sha512-aKS4WQjPenRxiQsC93MNfjx+nbF4PAdYzmd/1JIj8HYzqfbu86beTuNgXDzPknWk0n0uARlyewZo4s++ES36Ow==", + "dev": true, + "requires": { + "minimist": "^1.2.0" + } + }, + "loader-utils": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/loader-utils/-/loader-utils-1.4.0.tgz", + "integrity": "sha512-qH0WSMBtn/oHuwjy/NucEgbx5dbxxnxup9s4PVXJUDHZBQY+s0NWA9rJf53RBnQZxfch7euUui7hpoAPvALZdA==", + "dev": true, + "requires": { + "big.js": "^5.2.2", + "emojis-list": "^3.0.0", + "json5": "^1.0.1" + } + }, + "locate-path": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-3.0.0.tgz", + "integrity": "sha512-7AO748wWnIhNqAuaty2ZWHkQHRSNfPVIsPIfwEOWO22AmaoVrWavlOcMR5nzTLNYvp36X220/maaRsrec1G65A==", + "dev": true, + "requires": { + "p-locate": "^3.0.0", + "path-exists": "^3.0.0" + } + }, + "make-dir": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/make-dir/-/make-dir-2.1.0.tgz", + "integrity": "sha512-LS9X+dc8KLxXCb8dni79fLIIUA5VyZoyjSMCwTluaXA0o27cCK0bhXkpgw+sTXVpPy/lSO57ilRixqk0vDmtRA==", + "dev": true, + "requires": { + "pify": "^4.0.1", + "semver": "^5.6.0" + } + }, + "memory-fs": { + "version": "0.4.1", + "resolved": "https://registry.npmjs.org/memory-fs/-/memory-fs-0.4.1.tgz", + "integrity": "sha1-OpoguEYlI+RHz7x+i7gO1me/xVI=", + "dev": true, + "requires": { + "errno": "^0.1.3", + "readable-stream": "^2.0.1" + } + }, + "p-locate": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-3.0.0.tgz", + "integrity": "sha512-x+12w/To+4GFfgJhBEpiDcLozRJGegY+Ei7/z0tSLkMmxGZNybVMSfWj9aJn8Z5Fc7dBUNJOOVgPv2H7IwulSQ==", + "dev": true, + "requires": { + "p-limit": "^2.0.0" + } + }, + "path-exists": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/path-exists/-/path-exists-3.0.0.tgz", + "integrity": "sha1-zg6+ql94yxiSXqfYENe1mwEP1RU=", + "dev": true + }, + "pify": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/pify/-/pify-4.0.1.tgz", + "integrity": "sha512-uB80kBFb/tfd68bVleG9T5GGsGPjJrLAUpR5PZIrhBnIaRTQRjqdJSsIKkOP6OAIFbj7GOrcudc5pNjZ+geV2g==", + "dev": true + }, + "pkg-dir": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/pkg-dir/-/pkg-dir-3.0.0.tgz", + "integrity": "sha512-/E57AYkoeQ25qkxMj5PBOVgF8Kiu/h7cYS30Z5+R7WaiCCBfLq58ZI/dSeaEKb9WVJV5n/03QwrN3IeWIFllvw==", + "dev": true, + "requires": { + "find-up": "^3.0.0" + } + }, + "schema-utils": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/schema-utils/-/schema-utils-1.0.0.tgz", + "integrity": "sha512-i27Mic4KovM/lnGsy8whRCHhc7VicJajAjTrYg11K9zfZXnYIt4k5F+kZkwjnrhKzLic/HLU4j11mjsz2G/75g==", + "dev": true, + "requires": { + "ajv": "^6.1.0", + "ajv-errors": "^1.0.0", + "ajv-keywords": "^3.1.0" + } + }, + "serialize-javascript": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/serialize-javascript/-/serialize-javascript-4.0.0.tgz", + "integrity": "sha512-GaNA54380uFefWghODBWEGisLZFj00nS5ACs6yHa9nLqlLpVLO8ChDGeKRjZnV4Nh4n0Qi7nhYZD/9fCPzEqkw==", + "dev": true, + "requires": { + "randombytes": "^2.1.0" + } + }, + "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 + }, + "terser-webpack-plugin": { + "version": "1.4.5", + "resolved": "https://registry.npmjs.org/terser-webpack-plugin/-/terser-webpack-plugin-1.4.5.tgz", + "integrity": "sha512-04Rfe496lN8EYruwi6oPQkG0vo8C+HT49X687FZnpPF0qMAIHONI6HEXYPKDOE8e5HjXTyKfqRd/agHtH0kOtw==", + "dev": true, + "requires": { + "cacache": "^12.0.2", + "find-cache-dir": "^2.1.0", + "is-wsl": "^1.1.0", + "schema-utils": "^1.0.0", + "serialize-javascript": "^4.0.0", + "source-map": "^0.6.1", + "terser": "^4.1.2", + "webpack-sources": "^1.4.0", + "worker-farm": "^1.7.0" + } + } + } + }, + "webpack-bundle-analyzer": { + "version": "3.8.0", + "resolved": "https://registry.npmjs.org/webpack-bundle-analyzer/-/webpack-bundle-analyzer-3.8.0.tgz", + "integrity": "sha512-PODQhAYVEourCcOuU+NiYI7WdR8QyELZGgPvB1y2tjbUpbmcQOt5Q7jEK+ttd5se0KSBKD9SXHCEozS++Wllmw==", + "dev": true, + "requires": { + "acorn": "^7.1.1", + "acorn-walk": "^7.1.1", + "bfj": "^6.1.1", + "chalk": "^2.4.1", + "commander": "^2.18.0", + "ejs": "^2.6.1", + "express": "^4.16.3", + "filesize": "^3.6.1", + "gzip-size": "^5.0.0", + "lodash": "^4.17.15", + "mkdirp": "^0.5.1", + "opener": "^1.5.1", + "ws": "^6.0.0" + }, + "dependencies": { + "ejs": { + "version": "2.7.4", + "resolved": "https://registry.npmjs.org/ejs/-/ejs-2.7.4.tgz", + "integrity": "sha512-7vmuyh5+kuUyJKePhQfRQBhXV5Ce+RnaeeQArKu1EAMpL3WbgMt5WG6uQZpEVvYSSsxMXRKOewtDk9RaTKXRlA==", + "dev": true + } + } + }, + "webpack-cli": { + "version": "3.3.12", + "resolved": "https://registry.npmjs.org/webpack-cli/-/webpack-cli-3.3.12.tgz", + "integrity": "sha512-NVWBaz9k839ZH/sinurM+HcDvJOTXwSjYp1ku+5XKeOC03z8v5QitnK/x+lAxGXFyhdayoIf/GOpv85z3/xPag==", + "dev": true, + "requires": { + "chalk": "^2.4.2", + "cross-spawn": "^6.0.5", + "enhanced-resolve": "^4.1.1", + "findup-sync": "^3.0.0", + "global-modules": "^2.0.0", + "import-local": "^2.0.0", + "interpret": "^1.4.0", + "loader-utils": "^1.4.0", + "supports-color": "^6.1.0", + "v8-compile-cache": "^2.1.1", + "yargs": "^13.3.2" + }, + "dependencies": { + "ansi-regex": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-4.1.0.tgz", + "integrity": "sha512-1apePfXM1UOSqw0o9IiFAovVz9M5S1Dg+4TrDwfMewQ6p/rmMueb7tWZjQ1rx4Loy1ArBggoqGpfqqdI4rondg==", + "dev": true + }, + "cliui": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/cliui/-/cliui-5.0.0.tgz", + "integrity": "sha512-PYeGSEmmHM6zvoef2w8TPzlrnNpXIjTipYK780YswmIP9vjxmd6Y2a3CB2Ks6/AU8NHjZugXvo8w3oWM2qnwXA==", + "dev": true, + "requires": { + "string-width": "^3.1.0", + "strip-ansi": "^5.2.0", + "wrap-ansi": "^5.1.0" + } + }, + "emoji-regex": { + "version": "7.0.3", + "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-7.0.3.tgz", + "integrity": "sha512-CwBLREIQ7LvYFB0WyRvwhq5N5qPhc6PMjD6bYggFlI5YyDgl+0vxq5VHbMOFqLg7hfWzmu8T5Z1QofhmTIhItA==", + "dev": true + }, + "find-up": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/find-up/-/find-up-3.0.0.tgz", + "integrity": "sha512-1yD6RmLI1XBfxugvORwlck6f75tYL+iR0jqwsOrOxMZyGYqUuDhJ0l4AXdO1iX/FTs9cBAMEk1gWSEx1kSbylg==", + "dev": true, + "requires": { + "locate-path": "^3.0.0" + } + }, + "import-local": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/import-local/-/import-local-2.0.0.tgz", + "integrity": "sha512-b6s04m3O+s3CGSbqDIyP4R6aAwAeYlVq9+WUWep6iHa8ETRf9yei1U48C5MmfJmV9AiLYYBKPMq/W+/WRpQmCQ==", + "dev": true, + "requires": { + "pkg-dir": "^3.0.0", + "resolve-cwd": "^2.0.0" + } + }, + "is-fullwidth-code-point": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-2.0.0.tgz", + "integrity": "sha1-o7MKXE8ZkYMWeqq5O+764937ZU8=", + "dev": true + }, + "json5": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/json5/-/json5-1.0.1.tgz", + "integrity": "sha512-aKS4WQjPenRxiQsC93MNfjx+nbF4PAdYzmd/1JIj8HYzqfbu86beTuNgXDzPknWk0n0uARlyewZo4s++ES36Ow==", + "dev": true, + "requires": { + "minimist": "^1.2.0" + } + }, + "loader-utils": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/loader-utils/-/loader-utils-1.4.0.tgz", + "integrity": "sha512-qH0WSMBtn/oHuwjy/NucEgbx5dbxxnxup9s4PVXJUDHZBQY+s0NWA9rJf53RBnQZxfch7euUui7hpoAPvALZdA==", + "dev": true, + "requires": { + "big.js": "^5.2.2", + "emojis-list": "^3.0.0", + "json5": "^1.0.1" + } + }, + "locate-path": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-3.0.0.tgz", + "integrity": "sha512-7AO748wWnIhNqAuaty2ZWHkQHRSNfPVIsPIfwEOWO22AmaoVrWavlOcMR5nzTLNYvp36X220/maaRsrec1G65A==", + "dev": true, + "requires": { + "p-locate": "^3.0.0", + "path-exists": "^3.0.0" + } + }, + "p-locate": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-3.0.0.tgz", + "integrity": "sha512-x+12w/To+4GFfgJhBEpiDcLozRJGegY+Ei7/z0tSLkMmxGZNybVMSfWj9aJn8Z5Fc7dBUNJOOVgPv2H7IwulSQ==", + "dev": true, + "requires": { + "p-limit": "^2.0.0" + } + }, + "path-exists": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/path-exists/-/path-exists-3.0.0.tgz", + "integrity": "sha1-zg6+ql94yxiSXqfYENe1mwEP1RU=", + "dev": true + }, + "pkg-dir": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/pkg-dir/-/pkg-dir-3.0.0.tgz", + "integrity": "sha512-/E57AYkoeQ25qkxMj5PBOVgF8Kiu/h7cYS30Z5+R7WaiCCBfLq58ZI/dSeaEKb9WVJV5n/03QwrN3IeWIFllvw==", + "dev": true, + "requires": { + "find-up": "^3.0.0" + } + }, + "resolve-cwd": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/resolve-cwd/-/resolve-cwd-2.0.0.tgz", + "integrity": "sha1-AKn3OHVW4nA46uIyyqNypqWbZlo=", + "dev": true, + "requires": { + "resolve-from": "^3.0.0" + } + }, + "resolve-from": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/resolve-from/-/resolve-from-3.0.0.tgz", + "integrity": "sha1-six699nWiBvItuZTM17rywoYh0g=", + "dev": true + }, + "string-width": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/string-width/-/string-width-3.1.0.tgz", + "integrity": "sha512-vafcv6KjVZKSgz06oM/H6GDBrAtz8vdhQakGjFIvNrHA6y3HCF1CInLy+QLq8dTJPQ1b+KDUqDFctkdRW44e1w==", + "dev": true, + "requires": { + "emoji-regex": "^7.0.1", + "is-fullwidth-code-point": "^2.0.0", + "strip-ansi": "^5.1.0" + } + }, + "strip-ansi": { + "version": "5.2.0", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-5.2.0.tgz", + "integrity": "sha512-DuRs1gKbBqsMKIZlrffwlug8MHkcnpjs5VPmL1PAh+mA30U0DTotfDZ0d2UUsXpPmPmMMJ6W773MaA3J+lbiWA==", + "dev": true, + "requires": { + "ansi-regex": "^4.1.0" + } + }, + "supports-color": { + "version": "6.1.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-6.1.0.tgz", + "integrity": "sha512-qe1jfm1Mg7Nq/NSh6XE24gPXROEVsWHxC1LIx//XNlD9iw7YZQGjZNjYN7xGaEG6iKdA8EtNFW6R0gjnVXp+wQ==", + "dev": true, + "requires": { + "has-flag": "^3.0.0" + } + }, + "wrap-ansi": { + "version": "5.1.0", + "resolved": "https://registry.npmjs.org/wrap-ansi/-/wrap-ansi-5.1.0.tgz", + "integrity": "sha512-QC1/iN/2/RPVJ5jYK8BGttj5z83LmSKmvbvrXPNCLZSEb32KKVDJDl/MOt2N01qU2H/FkzEa9PKto1BqDjtd7Q==", + "dev": true, + "requires": { + "ansi-styles": "^3.2.0", + "string-width": "^3.0.0", + "strip-ansi": "^5.0.0" + } + }, + "yargs": { + "version": "13.3.2", + "resolved": "https://registry.npmjs.org/yargs/-/yargs-13.3.2.tgz", + "integrity": "sha512-AX3Zw5iPruN5ie6xGRIDgqkT+ZhnRlZMLMHAs8tg7nRruy2Nb+i5o9bwghAogtM08q1dpr2LVoS8KSTMYpWXUw==", + "dev": true, + "requires": { + "cliui": "^5.0.0", + "find-up": "^3.0.0", + "get-caller-file": "^2.0.1", + "require-directory": "^2.1.1", + "require-main-filename": "^2.0.0", + "set-blocking": "^2.0.0", + "string-width": "^3.0.0", + "which-module": "^2.0.0", + "y18n": "^4.0.0", + "yargs-parser": "^13.1.2" + } + }, + "yargs-parser": { + "version": "13.1.2", + "resolved": "https://registry.npmjs.org/yargs-parser/-/yargs-parser-13.1.2.tgz", + "integrity": "sha512-3lbsNRf/j+A4QuSZfDRA7HRSfWrzO0YjqTJd5kjAq37Zep1CEgaYmrH9Q3GwPiB9cHyd1Y1UwggGhJGoxipbzg==", + "dev": true, + "requires": { + "camelcase": "^5.0.0", + "decamelize": "^1.2.0" + } + } + } + }, + "webpack-dev-middleware": { + "version": "3.7.2", + "resolved": "https://registry.npmjs.org/webpack-dev-middleware/-/webpack-dev-middleware-3.7.2.tgz", + "integrity": "sha512-1xC42LxbYoqLNAhV6YzTYacicgMZQTqRd27Sim9wn5hJrX3I5nxYy1SxSd4+gjUFsz1dQFj+yEe6zEVmSkeJjw==", + "dev": true, + "requires": { + "memory-fs": "^0.4.1", + "mime": "^2.4.4", + "mkdirp": "^0.5.1", + "range-parser": "^1.2.1", + "webpack-log": "^2.0.0" + }, + "dependencies": { + "memory-fs": { + "version": "0.4.1", + "resolved": "https://registry.npmjs.org/memory-fs/-/memory-fs-0.4.1.tgz", + "integrity": "sha1-OpoguEYlI+RHz7x+i7gO1me/xVI=", + "dev": true, + "requires": { + "errno": "^0.1.3", + "readable-stream": "^2.0.1" + } + }, + "mime": { + "version": "2.4.6", + "resolved": "https://registry.npmjs.org/mime/-/mime-2.4.6.tgz", + "integrity": "sha512-RZKhC3EmpBchfTGBVb8fb+RL2cWyw/32lshnsETttkBAyAUXSGHxbEJWWRXc751DrIxG1q04b8QwMbAwkRPpUA==", + "dev": true + } + } + }, + "webpack-dev-server": { + "version": "3.11.0", + "resolved": "https://registry.npmjs.org/webpack-dev-server/-/webpack-dev-server-3.11.0.tgz", + "integrity": "sha512-PUxZ+oSTxogFQgkTtFndEtJIPNmml7ExwufBZ9L2/Xyyd5PnOL5UreWe5ZT7IU25DSdykL9p1MLQzmLh2ljSeg==", + "dev": true, + "requires": { + "ansi-html": "0.0.7", + "bonjour": "^3.5.0", + "chokidar": "^2.1.8", + "compression": "^1.7.4", + "connect-history-api-fallback": "^1.6.0", + "debug": "^4.1.1", + "del": "^4.1.1", + "express": "^4.17.1", + "html-entities": "^1.3.1", + "http-proxy-middleware": "0.19.1", + "import-local": "^2.0.0", + "internal-ip": "^4.3.0", + "ip": "^1.1.5", + "is-absolute-url": "^3.0.3", + "killable": "^1.0.1", + "loglevel": "^1.6.8", + "opn": "^5.5.0", + "p-retry": "^3.0.1", + "portfinder": "^1.0.26", + "schema-utils": "^1.0.0", + "selfsigned": "^1.10.7", + "semver": "^6.3.0", + "serve-index": "^1.9.1", + "sockjs": "0.3.20", + "sockjs-client": "1.4.0", + "spdy": "^4.0.2", + "strip-ansi": "^3.0.1", + "supports-color": "^6.1.0", + "url": "^0.11.0", + "webpack-dev-middleware": "^3.7.2", + "webpack-log": "^2.0.0", + "ws": "^6.2.1", + "yargs": "^13.3.2" + }, + "dependencies": { + "ansi-regex": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-2.1.1.tgz", + "integrity": "sha1-w7M6te42DYbg5ijwRorn7yfWVN8=", + "dev": true + }, + "anymatch": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/anymatch/-/anymatch-2.0.0.tgz", + "integrity": "sha512-5teOsQWABXHHBFP9y3skS5P3d/WfWXpv3FUpy+LorMrNYaT9pI4oLMQX7jzQ2KklNpGpWHzdCXTDT2Y3XGlZBw==", + "dev": true, + "requires": { + "micromatch": "^3.1.4", + "normalize-path": "^2.1.1" + }, + "dependencies": { + "normalize-path": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/normalize-path/-/normalize-path-2.1.1.tgz", + "integrity": "sha1-GrKLVW4Zg2Oowab35vogE3/mrtk=", + "dev": true, + "requires": { + "remove-trailing-separator": "^1.0.1" + } + } + } + }, + "array-union": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/array-union/-/array-union-1.0.2.tgz", + "integrity": "sha1-mjRBDk9OPaI96jdb5b5w8kd47Dk=", + "dev": true, + "requires": { + "array-uniq": "^1.0.1" + } + }, + "binary-extensions": { + "version": "1.13.1", + "resolved": "https://registry.npmjs.org/binary-extensions/-/binary-extensions-1.13.1.tgz", + "integrity": "sha512-Un7MIEDdUC5gNpcGDV97op1Ywk748MpHcFTHoYs6qnj1Z3j7I53VG3nwZhKzoBZmbdRNnb6WRdFlwl7tSDuZGw==", + "dev": true + }, + "braces": { + "version": "2.3.2", + "resolved": "https://registry.npmjs.org/braces/-/braces-2.3.2.tgz", + "integrity": "sha512-aNdbnj9P8PjdXU4ybaWLK2IF3jc/EoDYbC7AazW6to3TRsfXxscC9UXOB5iDiEQrkyIbWp2SLQda4+QAa7nc3w==", + "dev": true, + "requires": { + "arr-flatten": "^1.1.0", + "array-unique": "^0.3.2", + "extend-shallow": "^2.0.1", + "fill-range": "^4.0.0", + "isobject": "^3.0.1", + "repeat-element": "^1.1.2", + "snapdragon": "^0.8.1", + "snapdragon-node": "^2.0.1", + "split-string": "^3.0.2", + "to-regex": "^3.0.1" + } + }, + "chokidar": { + "version": "2.1.8", + "resolved": "https://registry.npmjs.org/chokidar/-/chokidar-2.1.8.tgz", + "integrity": "sha512-ZmZUazfOzf0Nve7duiCKD23PFSCs4JPoYyccjUFF3aQkQadqBhfzhjkwBH2mNOG9cTBwhamM37EIsIkZw3nRgg==", + "dev": true, + "requires": { + "anymatch": "^2.0.0", + "async-each": "^1.0.1", + "braces": "^2.3.2", + "fsevents": "^1.2.7", + "glob-parent": "^3.1.0", + "inherits": "^2.0.3", + "is-binary-path": "^1.0.0", + "is-glob": "^4.0.0", + "normalize-path": "^3.0.0", + "path-is-absolute": "^1.0.0", + "readdirp": "^2.2.1", + "upath": "^1.1.1" + } + }, + "cliui": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/cliui/-/cliui-5.0.0.tgz", + "integrity": "sha512-PYeGSEmmHM6zvoef2w8TPzlrnNpXIjTipYK780YswmIP9vjxmd6Y2a3CB2Ks6/AU8NHjZugXvo8w3oWM2qnwXA==", + "dev": true, + "requires": { + "string-width": "^3.1.0", + "strip-ansi": "^5.2.0", + "wrap-ansi": "^5.1.0" + }, + "dependencies": { + "ansi-regex": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-4.1.0.tgz", + "integrity": "sha512-1apePfXM1UOSqw0o9IiFAovVz9M5S1Dg+4TrDwfMewQ6p/rmMueb7tWZjQ1rx4Loy1ArBggoqGpfqqdI4rondg==", + "dev": true + }, + "strip-ansi": { + "version": "5.2.0", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-5.2.0.tgz", + "integrity": "sha512-DuRs1gKbBqsMKIZlrffwlug8MHkcnpjs5VPmL1PAh+mA30U0DTotfDZ0d2UUsXpPmPmMMJ6W773MaA3J+lbiWA==", + "dev": true, + "requires": { + "ansi-regex": "^4.1.0" + } + } + } + }, + "del": { + "version": "4.1.1", + "resolved": "https://registry.npmjs.org/del/-/del-4.1.1.tgz", + "integrity": "sha512-QwGuEUouP2kVwQenAsOof5Fv8K9t3D8Ca8NxcXKrIpEHjTXK5J2nXLdP+ALI1cgv8wj7KuwBhTwBkOZSJKM5XQ==", + "dev": true, + "requires": { + "@types/glob": "^7.1.1", + "globby": "^6.1.0", + "is-path-cwd": "^2.0.0", + "is-path-in-cwd": "^2.0.0", + "p-map": "^2.0.0", + "pify": "^4.0.1", + "rimraf": "^2.6.3" + } + }, + "emoji-regex": { + "version": "7.0.3", + "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-7.0.3.tgz", + "integrity": "sha512-CwBLREIQ7LvYFB0WyRvwhq5N5qPhc6PMjD6bYggFlI5YyDgl+0vxq5VHbMOFqLg7hfWzmu8T5Z1QofhmTIhItA==", + "dev": true + }, + "extend-shallow": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/extend-shallow/-/extend-shallow-2.0.1.tgz", + "integrity": "sha1-Ua99YUrZqfYQ6huvu5idaxxWiQ8=", + "dev": true, + "requires": { + "is-extendable": "^0.1.0" + } + }, + "fill-range": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/fill-range/-/fill-range-4.0.0.tgz", + "integrity": "sha1-1USBHUKPmOsGpj3EAtJAPDKMOPc=", + "dev": true, + "requires": { + "extend-shallow": "^2.0.1", + "is-number": "^3.0.0", + "repeat-string": "^1.6.1", + "to-regex-range": "^2.1.0" + } + }, + "find-up": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/find-up/-/find-up-3.0.0.tgz", + "integrity": "sha512-1yD6RmLI1XBfxugvORwlck6f75tYL+iR0jqwsOrOxMZyGYqUuDhJ0l4AXdO1iX/FTs9cBAMEk1gWSEx1kSbylg==", + "dev": true, + "requires": { + "locate-path": "^3.0.0" + } + }, + "fsevents": { + "version": "1.2.13", + "resolved": "https://registry.npmjs.org/fsevents/-/fsevents-1.2.13.tgz", + "integrity": "sha512-oWb1Z6mkHIskLzEJ/XWX0srkpkTQ7vaopMQkyaEIoq0fmtFVxOthb8cCxeT+p3ynTdkk/RZwbgG4brR5BeWECw==", + "dev": true, + "optional": true + }, + "glob-parent": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-3.1.0.tgz", + "integrity": "sha1-nmr2KZ2NO9K9QEMIMr0RPfkGxa4=", + "dev": true, + "requires": { + "is-glob": "^3.1.0", + "path-dirname": "^1.0.0" + }, + "dependencies": { + "is-glob": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-3.1.0.tgz", + "integrity": "sha1-e6WuJCF4BKxwcHuWkiVnSGzD6Eo=", + "dev": true, + "requires": { + "is-extglob": "^2.1.0" + } + } + } + }, + "globby": { + "version": "6.1.0", + "resolved": "https://registry.npmjs.org/globby/-/globby-6.1.0.tgz", + "integrity": "sha1-9abXDoOV4hyFj7BInWTfAkJNUGw=", + "dev": true, + "requires": { + "array-union": "^1.0.1", + "glob": "^7.0.3", + "object-assign": "^4.0.1", + "pify": "^2.0.0", + "pinkie-promise": "^2.0.0" + }, + "dependencies": { + "pify": { + "version": "2.3.0", + "resolved": "https://registry.npmjs.org/pify/-/pify-2.3.0.tgz", + "integrity": "sha1-7RQaasBDqEnqWISY59yosVMw6Qw=", + "dev": true + } + } + }, + "import-local": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/import-local/-/import-local-2.0.0.tgz", + "integrity": "sha512-b6s04m3O+s3CGSbqDIyP4R6aAwAeYlVq9+WUWep6iHa8ETRf9yei1U48C5MmfJmV9AiLYYBKPMq/W+/WRpQmCQ==", + "dev": true, + "requires": { + "pkg-dir": "^3.0.0", + "resolve-cwd": "^2.0.0" + } + }, + "is-absolute-url": { + "version": "3.0.3", + "resolved": "https://registry.npmjs.org/is-absolute-url/-/is-absolute-url-3.0.3.tgz", + "integrity": "sha512-opmNIX7uFnS96NtPmhWQgQx6/NYFgsUXYMllcfzwWKUMwfo8kku1TvE6hkNcH+Q1ts5cMVrsY7j0bxXQDciu9Q==", + "dev": true + }, + "is-binary-path": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/is-binary-path/-/is-binary-path-1.0.1.tgz", + "integrity": "sha1-dfFmQrSA8YenEcgUFh/TpKdlWJg=", + "dev": true, + "requires": { + "binary-extensions": "^1.0.0" + } + }, + "is-fullwidth-code-point": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-2.0.0.tgz", + "integrity": "sha1-o7MKXE8ZkYMWeqq5O+764937ZU8=", + "dev": true + }, + "is-number": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/is-number/-/is-number-3.0.0.tgz", + "integrity": "sha1-JP1iAaR4LPUFYcgQJ2r8fRLXEZU=", + "dev": true, + "requires": { + "kind-of": "^3.0.2" + } + }, + "is-path-cwd": { + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/is-path-cwd/-/is-path-cwd-2.2.0.tgz", + "integrity": "sha512-w942bTcih8fdJPJmQHFzkS76NEP8Kzzvmw92cXsazb8intwLqPibPPdXf4ANdKV3rYMuuQYGIWtvz9JilB3NFQ==", + "dev": true + }, + "is-path-in-cwd": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/is-path-in-cwd/-/is-path-in-cwd-2.1.0.tgz", + "integrity": "sha512-rNocXHgipO+rvnP6dk3zI20RpOtrAM/kzbB258Uw5BWr3TpXi861yzjo16Dn4hUox07iw5AyeMLHWsujkjzvRQ==", + "dev": true, + "requires": { + "is-path-inside": "^2.1.0" + } + }, + "is-path-inside": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/is-path-inside/-/is-path-inside-2.1.0.tgz", + "integrity": "sha512-wiyhTzfDWsvwAW53OBWF5zuvaOGlZ6PwYxAbPVDhpm+gM09xKQGjBq/8uYN12aDvMxnAnq3dxTyoSoRNmg5YFg==", + "dev": true, + "requires": { + "path-is-inside": "^1.0.2" + } + }, + "is-wsl": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/is-wsl/-/is-wsl-1.1.0.tgz", + "integrity": "sha1-HxbkqiKwTRM2tmGIpmrzxgDDpm0=", + "dev": true + }, + "kind-of": { + "version": "3.2.2", + "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-3.2.2.tgz", + "integrity": "sha1-MeohpzS6ubuw8yRm2JOupR5KPGQ=", + "dev": true, + "requires": { + "is-buffer": "^1.1.5" + } + }, + "locate-path": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-3.0.0.tgz", + "integrity": "sha512-7AO748wWnIhNqAuaty2ZWHkQHRSNfPVIsPIfwEOWO22AmaoVrWavlOcMR5nzTLNYvp36X220/maaRsrec1G65A==", + "dev": true, + "requires": { + "p-locate": "^3.0.0", + "path-exists": "^3.0.0" + } + }, + "opn": { + "version": "5.5.0", + "resolved": "https://registry.npmjs.org/opn/-/opn-5.5.0.tgz", + "integrity": "sha512-PqHpggC9bLV0VeWcdKhkpxY+3JTzetLSqTCWL/z/tFIbI6G8JCjondXklT1JinczLz2Xib62sSp0T/gKT4KksA==", + "dev": true, + "requires": { + "is-wsl": "^1.1.0" + } + }, + "p-locate": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-3.0.0.tgz", + "integrity": "sha512-x+12w/To+4GFfgJhBEpiDcLozRJGegY+Ei7/z0tSLkMmxGZNybVMSfWj9aJn8Z5Fc7dBUNJOOVgPv2H7IwulSQ==", + "dev": true, + "requires": { + "p-limit": "^2.0.0" + } + }, + "p-map": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/p-map/-/p-map-2.1.0.tgz", + "integrity": "sha512-y3b8Kpd8OAN444hxfBbFfj1FY/RjtTd8tzYwhUqNYXx0fXx2iX4maP4Qr6qhIKbQXI02wTLAda4fYUbDagTUFw==", + "dev": true + }, + "path-exists": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/path-exists/-/path-exists-3.0.0.tgz", + "integrity": "sha1-zg6+ql94yxiSXqfYENe1mwEP1RU=", + "dev": true + }, + "pify": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/pify/-/pify-4.0.1.tgz", + "integrity": "sha512-uB80kBFb/tfd68bVleG9T5GGsGPjJrLAUpR5PZIrhBnIaRTQRjqdJSsIKkOP6OAIFbj7GOrcudc5pNjZ+geV2g==", + "dev": true + }, + "pkg-dir": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/pkg-dir/-/pkg-dir-3.0.0.tgz", + "integrity": "sha512-/E57AYkoeQ25qkxMj5PBOVgF8Kiu/h7cYS30Z5+R7WaiCCBfLq58ZI/dSeaEKb9WVJV5n/03QwrN3IeWIFllvw==", + "dev": true, + "requires": { + "find-up": "^3.0.0" + } + }, + "punycode": { + "version": "1.3.2", + "resolved": "https://registry.npmjs.org/punycode/-/punycode-1.3.2.tgz", + "integrity": "sha1-llOgNvt8HuQjQvIyXM7v6jkmxI0=", + "dev": true + }, + "readdirp": { + "version": "2.2.1", + "resolved": "https://registry.npmjs.org/readdirp/-/readdirp-2.2.1.tgz", + "integrity": "sha512-1JU/8q+VgFZyxwrJ+SVIOsh+KywWGpds3NTqikiKpDMZWScmAYyKIgqkO+ARvNWJfXeXR1zxz7aHF4u4CyH6vQ==", + "dev": true, + "requires": { + "graceful-fs": "^4.1.11", + "micromatch": "^3.1.10", + "readable-stream": "^2.0.2" + } + }, + "resolve-cwd": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/resolve-cwd/-/resolve-cwd-2.0.0.tgz", + "integrity": "sha1-AKn3OHVW4nA46uIyyqNypqWbZlo=", + "dev": true, + "requires": { + "resolve-from": "^3.0.0" + } + }, + "resolve-from": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/resolve-from/-/resolve-from-3.0.0.tgz", + "integrity": "sha1-six699nWiBvItuZTM17rywoYh0g=", + "dev": true + }, + "rimraf": { + "version": "2.7.1", + "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-2.7.1.tgz", + "integrity": "sha512-uWjbaKIK3T1OSVptzX7Nl6PvQ3qAGtKEtVRjRuazjfL3Bx5eI409VZSqgND+4UNnmzLVdPj9FqFJNPqBZFve4w==", + "dev": true, + "requires": { + "glob": "^7.1.3" + }, + "dependencies": { + "glob": { + "version": "7.1.6", + "resolved": "https://registry.npmjs.org/glob/-/glob-7.1.6.tgz", + "integrity": "sha512-LwaxwyZ72Lk7vZINtNNrywX0ZuLyStrdDtabefZKAY5ZGJhVtgdznluResxNmPitE0SAO+O26sWTHeKSI2wMBA==", + "dev": true, + "requires": { + "fs.realpath": "^1.0.0", + "inflight": "^1.0.4", + "inherits": "2", + "minimatch": "^3.0.4", + "once": "^1.3.0", + "path-is-absolute": "^1.0.0" + } + } + } + }, + "schema-utils": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/schema-utils/-/schema-utils-1.0.0.tgz", + "integrity": "sha512-i27Mic4KovM/lnGsy8whRCHhc7VicJajAjTrYg11K9zfZXnYIt4k5F+kZkwjnrhKzLic/HLU4j11mjsz2G/75g==", + "dev": true, + "requires": { + "ajv": "^6.1.0", + "ajv-errors": "^1.0.0", + "ajv-keywords": "^3.1.0" + } + }, + "semver": { + "version": "6.3.0", + "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.0.tgz", + "integrity": "sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw==", + "dev": true + }, + "string-width": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/string-width/-/string-width-3.1.0.tgz", + "integrity": "sha512-vafcv6KjVZKSgz06oM/H6GDBrAtz8vdhQakGjFIvNrHA6y3HCF1CInLy+QLq8dTJPQ1b+KDUqDFctkdRW44e1w==", + "dev": true, + "requires": { + "emoji-regex": "^7.0.1", + "is-fullwidth-code-point": "^2.0.0", + "strip-ansi": "^5.1.0" + }, + "dependencies": { + "ansi-regex": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-4.1.0.tgz", + "integrity": "sha512-1apePfXM1UOSqw0o9IiFAovVz9M5S1Dg+4TrDwfMewQ6p/rmMueb7tWZjQ1rx4Loy1ArBggoqGpfqqdI4rondg==", + "dev": true + }, + "strip-ansi": { + "version": "5.2.0", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-5.2.0.tgz", + "integrity": "sha512-DuRs1gKbBqsMKIZlrffwlug8MHkcnpjs5VPmL1PAh+mA30U0DTotfDZ0d2UUsXpPmPmMMJ6W773MaA3J+lbiWA==", + "dev": true, + "requires": { + "ansi-regex": "^4.1.0" + } + } + } + }, + "strip-ansi": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-3.0.1.tgz", + "integrity": "sha1-ajhfuIU9lS1f8F0Oiq+UJ43GPc8=", + "dev": true, + "requires": { + "ansi-regex": "^2.0.0" + } + }, + "supports-color": { + "version": "6.1.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-6.1.0.tgz", + "integrity": "sha512-qe1jfm1Mg7Nq/NSh6XE24gPXROEVsWHxC1LIx//XNlD9iw7YZQGjZNjYN7xGaEG6iKdA8EtNFW6R0gjnVXp+wQ==", + "dev": true, + "requires": { + "has-flag": "^3.0.0" + } + }, + "to-regex-range": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/to-regex-range/-/to-regex-range-2.1.1.tgz", + "integrity": "sha1-fIDBe53+vlmeJzZ+DU3VWQFB2zg=", + "dev": true, + "requires": { + "is-number": "^3.0.0", + "repeat-string": "^1.6.1" + } + }, + "url": { + "version": "0.11.0", + "resolved": "https://registry.npmjs.org/url/-/url-0.11.0.tgz", + "integrity": "sha1-ODjpfPxgUh63PFJajlW/3Z4uKPE=", + "dev": true, + "requires": { + "punycode": "1.3.2", + "querystring": "0.2.0" + } + }, + "wrap-ansi": { + "version": "5.1.0", + "resolved": "https://registry.npmjs.org/wrap-ansi/-/wrap-ansi-5.1.0.tgz", + "integrity": "sha512-QC1/iN/2/RPVJ5jYK8BGttj5z83LmSKmvbvrXPNCLZSEb32KKVDJDl/MOt2N01qU2H/FkzEa9PKto1BqDjtd7Q==", + "dev": true, + "requires": { + "ansi-styles": "^3.2.0", + "string-width": "^3.0.0", + "strip-ansi": "^5.0.0" + }, + "dependencies": { + "ansi-regex": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-4.1.0.tgz", + "integrity": "sha512-1apePfXM1UOSqw0o9IiFAovVz9M5S1Dg+4TrDwfMewQ6p/rmMueb7tWZjQ1rx4Loy1ArBggoqGpfqqdI4rondg==", + "dev": true + }, + "strip-ansi": { + "version": "5.2.0", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-5.2.0.tgz", + "integrity": "sha512-DuRs1gKbBqsMKIZlrffwlug8MHkcnpjs5VPmL1PAh+mA30U0DTotfDZ0d2UUsXpPmPmMMJ6W773MaA3J+lbiWA==", + "dev": true, + "requires": { + "ansi-regex": "^4.1.0" + } + } + } + }, + "ws": { + "version": "6.2.1", + "resolved": "https://registry.npmjs.org/ws/-/ws-6.2.1.tgz", + "integrity": "sha512-GIyAXC2cB7LjvpgMt9EKS2ldqr0MTrORaleiOno6TweZ6r3TKtoFQWay/2PceJ3RuBasOHzXNn5Lrw1X0bEjqA==", + "dev": true, + "requires": { + "async-limiter": "~1.0.0" + } + }, + "yargs": { + "version": "13.3.2", + "resolved": "https://registry.npmjs.org/yargs/-/yargs-13.3.2.tgz", + "integrity": "sha512-AX3Zw5iPruN5ie6xGRIDgqkT+ZhnRlZMLMHAs8tg7nRruy2Nb+i5o9bwghAogtM08q1dpr2LVoS8KSTMYpWXUw==", + "dev": true, + "requires": { + "cliui": "^5.0.0", + "find-up": "^3.0.0", + "get-caller-file": "^2.0.1", + "require-directory": "^2.1.1", + "require-main-filename": "^2.0.0", + "set-blocking": "^2.0.0", + "string-width": "^3.0.0", + "which-module": "^2.0.0", + "y18n": "^4.0.0", + "yargs-parser": "^13.1.2" + } + }, + "yargs-parser": { + "version": "13.1.2", + "resolved": "https://registry.npmjs.org/yargs-parser/-/yargs-parser-13.1.2.tgz", + "integrity": "sha512-3lbsNRf/j+A4QuSZfDRA7HRSfWrzO0YjqTJd5kjAq37Zep1CEgaYmrH9Q3GwPiB9cHyd1Y1UwggGhJGoxipbzg==", + "dev": true, + "requires": { + "camelcase": "^5.0.0", + "decamelize": "^1.2.0" + } + } + } + }, + "webpack-log": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/webpack-log/-/webpack-log-2.0.0.tgz", + "integrity": "sha512-cX8G2vR/85UYG59FgkoMamwHUIkSSlV3bBMRsbxVXVUk2j6NleCKjQ/WE9eYg9WY4w25O9w8wKP4rzNZFmUcUg==", + "dev": true, + "requires": { + "ansi-colors": "^3.0.0", + "uuid": "^3.3.2" + }, + "dependencies": { + "ansi-colors": { + "version": "3.2.4", + "resolved": "https://registry.npmjs.org/ansi-colors/-/ansi-colors-3.2.4.tgz", + "integrity": "sha512-hHUXGagefjN2iRrID63xckIvotOXOojhQKWIPUZ4mNUZ9nLZW+7FMNoE1lOkEhNWYsx/7ysGIuJYCiMAA9FnrA==", + "dev": true + }, + "uuid": { + "version": "3.4.0", + "resolved": "https://registry.npmjs.org/uuid/-/uuid-3.4.0.tgz", + "integrity": "sha512-HjSDRw6gZE5JMggctHBcjVak08+KEVhSIiDzFnT9S9aegmp85S/bReBVTb4QTFaRNptJ9kuYaNhnbNEOkbKb/A==", + "dev": true + } + } + }, + "webpack-merge": { + "version": "4.2.2", + "resolved": "https://registry.npmjs.org/webpack-merge/-/webpack-merge-4.2.2.tgz", + "integrity": "sha512-TUE1UGoTX2Cd42j3krGYqObZbOD+xF7u28WB7tfUordytSjbWTIjK/8V0amkBfTYN4/pB/GIDlJZZ657BGG19g==", + "dev": true, + "requires": { + "lodash": "^4.17.15" + } + }, + "webpack-notifier": { + "version": "1.8.0", + "resolved": "https://registry.npmjs.org/webpack-notifier/-/webpack-notifier-1.8.0.tgz", + "integrity": "sha512-I6t76NoPe5DZCCm5geELmDV2wlJ89LbU425uN6T2FG8Ywrrt1ZcUMz6g8yWGNg4pttqTPFQJYUPjWAlzUEQ+cQ==", + "dev": true, + "requires": { + "node-notifier": "^5.1.2", + "object-assign": "^4.1.0", + "strip-ansi": "^3.0.1" + }, + "dependencies": { + "ansi-regex": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-2.1.1.tgz", + "integrity": "sha1-w7M6te42DYbg5ijwRorn7yfWVN8=", + "dev": true + }, + "is-wsl": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/is-wsl/-/is-wsl-1.1.0.tgz", + "integrity": "sha1-HxbkqiKwTRM2tmGIpmrzxgDDpm0=", + "dev": true + }, + "node-notifier": { + "version": "5.4.3", + "resolved": "https://registry.npmjs.org/node-notifier/-/node-notifier-5.4.3.tgz", + "integrity": "sha512-M4UBGcs4jeOK9CjTsYwkvH6/MzuUmGCyTW+kCY7uO+1ZVr0+FHGdPdIf5CCLqAaxnRrWidyoQlNkMIIVwbKB8Q==", + "dev": true, + "requires": { + "growly": "^1.3.0", + "is-wsl": "^1.1.0", + "semver": "^5.5.0", + "shellwords": "^0.1.1", + "which": "^1.3.0" + } + }, + "strip-ansi": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-3.0.1.tgz", + "integrity": "sha1-ajhfuIU9lS1f8F0Oiq+UJ43GPc8=", + "dev": true, + "requires": { + "ansi-regex": "^2.0.0" + } + } + } + }, + "webpack-sources": { + "version": "1.4.3", + "resolved": "https://registry.npmjs.org/webpack-sources/-/webpack-sources-1.4.3.tgz", + "integrity": "sha512-lgTS3Xhv1lCOKo7SA5TjKXMjpSM4sBjNV5+q2bqesbSPs5FjGmU6jjtBSkX9b4qW87vDIsCIlUPOEhbZrMdjeQ==", + "dev": true, + "requires": { + "source-list-map": "^2.0.0", + "source-map": "~0.6.1" + }, + "dependencies": { + "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 + } + } + }, + "websocket-driver": { + "version": "0.6.5", + "resolved": "https://registry.npmjs.org/websocket-driver/-/websocket-driver-0.6.5.tgz", + "integrity": "sha1-XLJVbOuF9Dc8bYI4qmkchFThOjY=", + "dev": true, + "requires": { + "websocket-extensions": ">=0.1.1" + } + }, + "websocket-extensions": { + "version": "0.1.4", + "resolved": "https://registry.npmjs.org/websocket-extensions/-/websocket-extensions-0.1.4.tgz", + "integrity": "sha512-OqedPIGOfsDlo31UNwYbCFMSaO9m9G/0faIHj5/dZFDMFqPTcx6UwqyOy3COEaEOg/9VsGIpdqn62W5KhoKSpg==", + "dev": true + }, + "whatwg-encoding": { + "version": "1.0.5", + "resolved": "https://registry.npmjs.org/whatwg-encoding/-/whatwg-encoding-1.0.5.tgz", + "integrity": "sha512-b5lim54JOPN9HtzvK9HFXvBma/rnfFeqsic0hSpjtDbVxR3dJKLc+KB4V6GgiGOvl7CY/KNh8rxSo9DKQrnUEw==", + "dev": true, + "requires": { + "iconv-lite": "0.4.24" + }, + "dependencies": { + "iconv-lite": { + "version": "0.4.24", + "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.4.24.tgz", + "integrity": "sha512-v3MXnZAcvnywkTUEZomIActle7RXXeedOR31wwl7VlyoXO4Qi9arvSenNQWne1TcRwhCL1HwLI21bEqdpj8/rA==", + "dev": true, + "requires": { + "safer-buffer": ">= 2.1.2 < 3" + } + } + } + }, + "whatwg-mimetype": { + "version": "2.3.0", + "resolved": "https://registry.npmjs.org/whatwg-mimetype/-/whatwg-mimetype-2.3.0.tgz", + "integrity": "sha512-M4yMwr6mAnQz76TbJm914+gPpB/nCwvZbJU28cUD6dR004SAxDLOOSUaB1JDRqLtaOV/vi0IC5lEAGFgrjGv/g==", + "dev": true + }, + "whatwg-url": { + "version": "8.4.0", + "resolved": "https://registry.npmjs.org/whatwg-url/-/whatwg-url-8.4.0.tgz", + "integrity": "sha512-vwTUFf6V4zhcPkWp/4CQPr1TW9Ml6SF4lVyaIMBdJw5i6qUUJ1QWM4Z6YYVkfka0OUIzVo/0aNtGVGk256IKWw==", + "dev": true, + "requires": { + "lodash.sortby": "^4.7.0", + "tr46": "^2.0.2", + "webidl-conversions": "^6.1.0" + } + }, + "which": { + "version": "1.3.1", + "resolved": "https://registry.npmjs.org/which/-/which-1.3.1.tgz", + "integrity": "sha512-HxJdYWq1MTIQbJ3nw0cqssHoTNU267KlrDuGZ1WYlxDStUtKUhOaJmh112/TZmHxxUfuJqPXSOm7tDyas0OSIQ==", + "dev": true, + "requires": { + "isexe": "^2.0.0" + } + }, + "which-module": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/which-module/-/which-module-2.0.0.tgz", + "integrity": "sha1-2e8H3Od7mQK4o6j6SzHD4/fm6Ho=" + }, + "which-pm-runs": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/which-pm-runs/-/which-pm-runs-1.0.0.tgz", + "integrity": "sha1-Zws6+8VS4LVd9rd4DKdGFfI60cs=", + "dev": true + }, + "window-size": { + "version": "0.2.0", + "resolved": "https://registry.npmjs.org/window-size/-/window-size-0.2.0.tgz", + "integrity": "sha1-tDFbtCFKPXBY6+7okuE/ok2YsHU=", + "dev": true + }, + "windows-release": { + "version": "3.3.3", + "resolved": "https://registry.npmjs.org/windows-release/-/windows-release-3.3.3.tgz", + "integrity": "sha512-OSOGH1QYiW5yVor9TtmXKQvt2vjQqbYS+DqmsZw+r7xDwLXEeT3JGW0ZppFmHx4diyXmxt238KFR3N9jzevBRg==", + "dev": true, + "requires": { + "execa": "^1.0.0" + } + }, + "winston": { + "version": "3.2.1", + "resolved": "https://registry.npmjs.org/winston/-/winston-3.2.1.tgz", + "integrity": "sha512-zU6vgnS9dAWCEKg/QYigd6cgMVVNwyTzKs81XZtTFuRwJOcDdBg7AU0mXVyNbs7O5RH2zdv+BdNZUlx7mXPuOw==", + "dev": true, + "requires": { + "async": "^2.6.1", + "diagnostics": "^1.1.1", + "is-stream": "^1.1.0", + "logform": "^2.1.1", + "one-time": "0.0.4", + "readable-stream": "^3.1.1", + "stack-trace": "0.0.x", + "triple-beam": "^1.3.0", + "winston-transport": "^4.3.0" + }, + "dependencies": { + "async": { + "version": "2.6.3", + "resolved": "https://registry.npmjs.org/async/-/async-2.6.3.tgz", + "integrity": "sha512-zflvls11DCy+dQWzTW2dzuilv8Z5X/pjfmZOWba6TNIVDm+2UDaJmXSOXlasHKfNBs8oo3M0aT50fDEWfKZjXg==", + "dev": true, + "requires": { + "lodash": "^4.17.14" + } + }, + "readable-stream": { + "version": "3.6.0", + "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-3.6.0.tgz", + "integrity": "sha512-BViHy7LKeTz4oNnkcLJ+lVSL6vpiFeX6/d3oSH8zCW7UxP2onchk+vTGB143xuFjHS3deTgkKoXXymXqymiIdA==", + "dev": true, + "requires": { + "inherits": "^2.0.3", + "string_decoder": "^1.1.1", + "util-deprecate": "^1.0.1" + } + } + } + }, + "winston-transport": { + "version": "4.4.0", + "resolved": "https://registry.npmjs.org/winston-transport/-/winston-transport-4.4.0.tgz", + "integrity": "sha512-Lc7/p3GtqtqPBYYtS6KCN3c77/2QCev51DvcJKbkFPQNoj1sinkGwLGFDxkXY9J6p9+EPnYs+D90uwbnaiURTw==", + "dev": true, + "requires": { + "readable-stream": "^2.3.7", + "triple-beam": "^1.2.0" + } + }, + "with-open-file": { + "version": "0.1.7", + "resolved": "https://registry.npmjs.org/with-open-file/-/with-open-file-0.1.7.tgz", + "integrity": "sha512-ecJS2/oHtESJ1t3ZfMI3B7KIDKyfN0O16miWxdn30zdh66Yd3LsRFebXZXq6GU4xfxLf6nVxp9kIqElb5fqczA==", + "dev": true, + "requires": { + "p-finally": "^1.0.0", + "p-try": "^2.1.0", + "pify": "^4.0.1" + }, + "dependencies": { + "pify": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/pify/-/pify-4.0.1.tgz", + "integrity": "sha512-uB80kBFb/tfd68bVleG9T5GGsGPjJrLAUpR5PZIrhBnIaRTQRjqdJSsIKkOP6OAIFbj7GOrcudc5pNjZ+geV2g==", + "dev": true + } + } + }, + "word-wrap": { + "version": "1.2.3", + "resolved": "https://registry.npmjs.org/word-wrap/-/word-wrap-1.2.3.tgz", + "integrity": "sha512-Hz/mrNwitNRh/HUAtM/VT/5VH+ygD6DV7mYKZAtHOrbs8U7lvPS6xf7EJKMF0uW1KJCl0H701g3ZGus+muE5vQ==", + "dev": true + }, + "workbox-background-sync": { + "version": "5.1.4", + "resolved": "https://registry.npmjs.org/workbox-background-sync/-/workbox-background-sync-5.1.4.tgz", + "integrity": "sha512-AH6x5pYq4vwQvfRDWH+vfOePfPIYQ00nCEB7dJRU1e0n9+9HMRyvI63FlDvtFT2AvXVRsXvUt7DNMEToyJLpSA==", + "dev": true, + "requires": { + "workbox-core": "^5.1.4" + } + }, + "workbox-broadcast-update": { + "version": "5.1.4", + "resolved": "https://registry.npmjs.org/workbox-broadcast-update/-/workbox-broadcast-update-5.1.4.tgz", + "integrity": "sha512-HTyTWkqXvHRuqY73XrwvXPud/FN6x3ROzkfFPsRjtw/kGZuZkPzfeH531qdUGfhtwjmtO/ZzXcWErqVzJNdXaA==", + "dev": true, + "requires": { + "workbox-core": "^5.1.4" + } + }, + "workbox-build": { + "version": "5.1.4", + "resolved": "https://registry.npmjs.org/workbox-build/-/workbox-build-5.1.4.tgz", + "integrity": "sha512-xUcZn6SYU8usjOlfLb9Y2/f86Gdo+fy1fXgH8tJHjxgpo53VVsqRX0lUDw8/JuyzNmXuo8vXX14pXX2oIm9Bow==", + "dev": true, + "requires": { + "@babel/core": "^7.8.4", + "@babel/preset-env": "^7.8.4", + "@babel/runtime": "^7.8.4", + "@hapi/joi": "^15.1.0", + "@rollup/plugin-node-resolve": "^7.1.1", + "@rollup/plugin-replace": "^2.3.1", + "@surma/rollup-plugin-off-main-thread": "^1.1.1", + "common-tags": "^1.8.0", + "fast-json-stable-stringify": "^2.1.0", + "fs-extra": "^8.1.0", + "glob": "^7.1.6", + "lodash.template": "^4.5.0", + "pretty-bytes": "^5.3.0", + "rollup": "^1.31.1", + "rollup-plugin-babel": "^4.3.3", + "rollup-plugin-terser": "^5.3.1", + "source-map": "^0.7.3", + "source-map-url": "^0.4.0", + "stringify-object": "^3.3.0", + "strip-comments": "^1.0.2", + "tempy": "^0.3.0", + "upath": "^1.2.0", + "workbox-background-sync": "^5.1.4", + "workbox-broadcast-update": "^5.1.4", + "workbox-cacheable-response": "^5.1.4", + "workbox-core": "^5.1.4", + "workbox-expiration": "^5.1.4", + "workbox-google-analytics": "^5.1.4", + "workbox-navigation-preload": "^5.1.4", + "workbox-precaching": "^5.1.4", + "workbox-range-requests": "^5.1.4", + "workbox-routing": "^5.1.4", + "workbox-strategies": "^5.1.4", + "workbox-streams": "^5.1.4", + "workbox-sw": "^5.1.4", + "workbox-window": "^5.1.4" + }, + "dependencies": { + "@babel/core": { + "version": "7.12.8", + "resolved": "https://registry.npmjs.org/@babel/core/-/core-7.12.8.tgz", + "integrity": "sha512-ra28JXL+5z73r1IC/t+FT1ApXU5LsulFDnTDntNfLQaScJUJmcHL5Qxm/IWanCToQk3bPWQo5bflbplU5r15pg==", + "dev": true, + "requires": { + "@babel/code-frame": "^7.10.4", + "@babel/generator": "^7.12.5", + "@babel/helper-module-transforms": "^7.12.1", + "@babel/helpers": "^7.12.5", + "@babel/parser": "^7.12.7", + "@babel/template": "^7.12.7", + "@babel/traverse": "^7.12.8", + "@babel/types": "^7.12.7", + "convert-source-map": "^1.7.0", + "debug": "^4.1.0", + "gensync": "^1.0.0-beta.1", + "json5": "^2.1.2", + "lodash": "^4.17.19", + "resolve": "^1.3.2", + "semver": "^5.4.1", + "source-map": "^0.5.0" + }, + "dependencies": { + "source-map": { + "version": "0.5.7", + "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.5.7.tgz", + "integrity": "sha1-igOdLRAh0i0eoUyA2OpGi6LvP8w=", + "dev": true + } + } + }, + "fs-extra": { + "version": "8.1.0", + "resolved": "https://registry.npmjs.org/fs-extra/-/fs-extra-8.1.0.tgz", + "integrity": "sha512-yhlQgA6mnOJUKOsRUFsgJdQCvkKhcz8tlZG5HBQfReYZy46OwLcY+Zia0mtdHsOo9y/hP+CxMN0TU9QxoOtG4g==", + "dev": true, + "requires": { + "graceful-fs": "^4.2.0", + "jsonfile": "^4.0.0", + "universalify": "^0.1.0" + } + }, + "glob": { + "version": "7.1.6", + "resolved": "https://registry.npmjs.org/glob/-/glob-7.1.6.tgz", + "integrity": "sha512-LwaxwyZ72Lk7vZINtNNrywX0ZuLyStrdDtabefZKAY5ZGJhVtgdznluResxNmPitE0SAO+O26sWTHeKSI2wMBA==", + "dev": true, + "requires": { + "fs.realpath": "^1.0.0", + "inflight": "^1.0.4", + "inherits": "2", + "minimatch": "^3.0.4", + "once": "^1.3.0", + "path-is-absolute": "^1.0.0" + } + }, + "source-map": { + "version": "0.7.3", + "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.7.3.tgz", + "integrity": "sha512-CkCj6giN3S+n9qrYiBTX5gystlENnRW5jZeNLHpe6aue+SrHcG5VYwujhW9s4dY31mEGsxBDrHR6oI69fTXsaQ==", + "dev": true + } + } + }, + "workbox-cacheable-response": { + "version": "5.1.4", + "resolved": "https://registry.npmjs.org/workbox-cacheable-response/-/workbox-cacheable-response-5.1.4.tgz", + "integrity": "sha512-0bfvMZs0Of1S5cdswfQK0BXt6ulU5kVD4lwer2CeI+03czHprXR3V4Y8lPTooamn7eHP8Iywi5QjyAMjw0qauA==", + "dev": true, + "requires": { + "workbox-core": "^5.1.4" + } + }, + "workbox-core": { + "version": "5.1.4", + "resolved": "https://registry.npmjs.org/workbox-core/-/workbox-core-5.1.4.tgz", + "integrity": "sha512-+4iRQan/1D8I81nR2L5vcbaaFskZC2CL17TLbvWVzQ4qiF/ytOGF6XeV54pVxAvKUtkLANhk8TyIUMtiMw2oDg==", + "dev": true + }, + "workbox-expiration": { + "version": "5.1.4", + "resolved": "https://registry.npmjs.org/workbox-expiration/-/workbox-expiration-5.1.4.tgz", + "integrity": "sha512-oDO/5iC65h2Eq7jctAv858W2+CeRW5e0jZBMNRXpzp0ZPvuT6GblUiHnAsC5W5lANs1QS9atVOm4ifrBiYY7AQ==", + "dev": true, + "requires": { + "workbox-core": "^5.1.4" + } + }, + "workbox-google-analytics": { + "version": "5.1.4", + "resolved": "https://registry.npmjs.org/workbox-google-analytics/-/workbox-google-analytics-5.1.4.tgz", + "integrity": "sha512-0IFhKoEVrreHpKgcOoddV+oIaVXBFKXUzJVBI+nb0bxmcwYuZMdteBTp8AEDJacENtc9xbR0wa9RDCnYsCDLjA==", + "dev": true, + "requires": { + "workbox-background-sync": "^5.1.4", + "workbox-core": "^5.1.4", + "workbox-routing": "^5.1.4", + "workbox-strategies": "^5.1.4" + } + }, + "workbox-navigation-preload": { + "version": "5.1.4", + "resolved": "https://registry.npmjs.org/workbox-navigation-preload/-/workbox-navigation-preload-5.1.4.tgz", + "integrity": "sha512-Wf03osvK0wTflAfKXba//QmWC5BIaIZARU03JIhAEO2wSB2BDROWI8Q/zmianf54kdV7e1eLaIEZhth4K4MyfQ==", + "dev": true, + "requires": { + "workbox-core": "^5.1.4" + } + }, + "workbox-precaching": { + "version": "5.1.4", + "resolved": "https://registry.npmjs.org/workbox-precaching/-/workbox-precaching-5.1.4.tgz", + "integrity": "sha512-gCIFrBXmVQLFwvAzuGLCmkUYGVhBb7D1k/IL7pUJUO5xacjLcFUaLnnsoVepBGAiKw34HU1y/YuqvTKim9qAZA==", + "dev": true, + "requires": { + "workbox-core": "^5.1.4" + } + }, + "workbox-range-requests": { + "version": "5.1.4", + "resolved": "https://registry.npmjs.org/workbox-range-requests/-/workbox-range-requests-5.1.4.tgz", + "integrity": "sha512-1HSujLjgTeoxHrMR2muDW2dKdxqCGMc1KbeyGcmjZZAizJTFwu7CWLDmLv6O1ceWYrhfuLFJO+umYMddk2XMhw==", + "dev": true, + "requires": { + "workbox-core": "^5.1.4" + } + }, + "workbox-routing": { + "version": "5.1.4", + "resolved": "https://registry.npmjs.org/workbox-routing/-/workbox-routing-5.1.4.tgz", + "integrity": "sha512-8ljknRfqE1vEQtnMtzfksL+UXO822jJlHTIR7+BtJuxQ17+WPZfsHqvk1ynR/v0EHik4x2+826Hkwpgh4GKDCw==", + "dev": true, + "requires": { + "workbox-core": "^5.1.4" + } + }, + "workbox-strategies": { + "version": "5.1.4", + "resolved": "https://registry.npmjs.org/workbox-strategies/-/workbox-strategies-5.1.4.tgz", + "integrity": "sha512-VVS57LpaJTdjW3RgZvPwX0NlhNmscR7OQ9bP+N/34cYMDzXLyA6kqWffP6QKXSkca1OFo/v6v7hW7zrrguo6EA==", + "dev": true, + "requires": { + "workbox-core": "^5.1.4", + "workbox-routing": "^5.1.4" + } + }, + "workbox-streams": { + "version": "5.1.4", + "resolved": "https://registry.npmjs.org/workbox-streams/-/workbox-streams-5.1.4.tgz", + "integrity": "sha512-xU8yuF1hI/XcVhJUAfbQLa1guQUhdLMPQJkdT0kn6HP5CwiPOGiXnSFq80rAG4b1kJUChQQIGPrq439FQUNVrw==", + "dev": true, + "requires": { + "workbox-core": "^5.1.4", + "workbox-routing": "^5.1.4" + } + }, + "workbox-sw": { + "version": "5.1.4", + "resolved": "https://registry.npmjs.org/workbox-sw/-/workbox-sw-5.1.4.tgz", + "integrity": "sha512-9xKnKw95aXwSNc8kk8gki4HU0g0W6KXu+xks7wFuC7h0sembFnTrKtckqZxbSod41TDaGh+gWUA5IRXrL0ECRA==", + "dev": true + }, + "workbox-webpack-plugin": { + "version": "5.1.3", + "resolved": "https://registry.npmjs.org/workbox-webpack-plugin/-/workbox-webpack-plugin-5.1.3.tgz", + "integrity": "sha512-gxSkZ9GFLrMNC/8DGNRjcMhrt8iu+MMXhH/Fpo3wo9rKaSMsI7esGq0klTH/UloP9pNvBizVydysrB52eRhI7w==", + "dev": true, + "requires": { + "@babel/runtime": "^7.5.5", + "fast-json-stable-stringify": "^2.0.0", + "source-map-url": "^0.4.0", + "upath": "^1.1.2", + "webpack-sources": "^1.3.0", + "workbox-build": "^5.1.3" + } + }, + "workbox-window": { + "version": "5.1.4", + "resolved": "https://registry.npmjs.org/workbox-window/-/workbox-window-5.1.4.tgz", + "integrity": "sha512-vXQtgTeMCUq/4pBWMfQX8Ee7N2wVC4Q7XYFqLnfbXJ2hqew/cU1uMTD2KqGEgEpE4/30luxIxgE+LkIa8glBYw==", + "dev": true, + "requires": { + "workbox-core": "^5.1.4" + } + }, + "worker-farm": { + "version": "1.7.0", + "resolved": "https://registry.npmjs.org/worker-farm/-/worker-farm-1.7.0.tgz", + "integrity": "sha512-rvw3QTZc8lAxyVrqcSGVm5yP/IJ2UcB3U0graE3LCFoZ0Yn2x4EoVSqJKdB/T5M+FLcRPjz4TDacRf3OCfNUzw==", + "dev": true, + "requires": { + "errno": "~0.1.7" + } + }, + "wrap-ansi": { + "version": "6.2.0", + "resolved": "https://registry.npmjs.org/wrap-ansi/-/wrap-ansi-6.2.0.tgz", + "integrity": "sha512-r6lPcBGxZXlIcymEu7InxDMhdW0KDxpLgoFLcguasxCaJ/SOIZwINatK9KY/tf+ZrlywOKU0UDj3ATXUBfxJXA==", + "requires": { + "ansi-styles": "^4.0.0", + "string-width": "^4.1.0", + "strip-ansi": "^6.0.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==", + "requires": { + "color-convert": "^2.0.1" + } + }, + "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==", + "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==" + } + } + }, + "wrappy": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/wrappy/-/wrappy-1.0.2.tgz", + "integrity": "sha1-tSQ9jz7BqjXxNkYFvA0QNuMKtp8=" + }, + "write": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/write/-/write-1.0.3.tgz", + "integrity": "sha512-/lg70HAjtkUgWPVZhZcm+T4hkL8Zbtp1nFNOn3lRrxnlv50SRBv7cR7RqR+GMsd3hUXy9hWBo4CHTbFTcOYwig==", + "dev": true, + "requires": { + "mkdirp": "^0.5.1" + } + }, + "write-file-atomic": { + "version": "3.0.3", + "resolved": "https://registry.npmjs.org/write-file-atomic/-/write-file-atomic-3.0.3.tgz", + "integrity": "sha512-AvHcyZ5JnSfq3ioSyjrBkH9yW4m7Ayk8/9My/DD9onKeu/94fwrMocemO2QAJFAlnnDN+ZDS+ZjAR5ua1/PV/Q==", + "dev": true, + "requires": { + "imurmurhash": "^0.1.4", + "is-typedarray": "^1.0.0", + "signal-exit": "^3.0.2", + "typedarray-to-buffer": "^3.1.5" + } + }, + "write-file-webpack-plugin": { + "version": "4.5.1", + "resolved": "https://registry.npmjs.org/write-file-webpack-plugin/-/write-file-webpack-plugin-4.5.1.tgz", + "integrity": "sha512-AZ7qJUvhTCBiOtG21aFJUcNuLVo2FFM6JMGKvaUGAH+QDqQAp2iG0nq3GcuXmJOFQR2JjpjhyYkyPrbFKhdjNQ==", + "dev": true, + "requires": { + "chalk": "^2.4.0", + "debug": "^3.1.0", + "filesize": "^3.6.1", + "lodash": "^4.17.13", + "mkdirp": "^0.5.1", + "moment": "^2.22.1", + "write-file-atomic": "^2.3.0" + }, + "dependencies": { + "debug": { + "version": "3.2.7", + "resolved": "https://registry.npmjs.org/debug/-/debug-3.2.7.tgz", + "integrity": "sha512-CFjzYYAi4ThfiQvizrFQevTTXHtnCqWfe7x1AhgEscTz6ZbLbfoLRLPugTQyBth6f8ZERVUSyWHFD/7Wu4t1XQ==", + "dev": true, + "requires": { + "ms": "^2.1.1" + } + }, + "write-file-atomic": { + "version": "2.4.3", + "resolved": "https://registry.npmjs.org/write-file-atomic/-/write-file-atomic-2.4.3.tgz", + "integrity": "sha512-GaETH5wwsX+GcnzhPgKcKjJ6M2Cq3/iZp1WyY/X1CSqrW+jVNM9Y7D8EC2sM4ZG/V8wZlSniJnCKWPmBYAucRQ==", + "dev": true, + "requires": { + "graceful-fs": "^4.1.11", + "imurmurhash": "^0.1.4", + "signal-exit": "^3.0.2" + } + } + } + }, + "ws": { + "version": "6.1.4", + "resolved": "https://registry.npmjs.org/ws/-/ws-6.1.4.tgz", + "integrity": "sha512-eqZfL+NE/YQc1/ZynhojeV8q+H050oR8AZ2uIev7RU10svA9ZnJUddHcOUZTJLinZ9yEfdA2kSATS2qZK5fhJA==", + "dev": true, + "requires": { + "async-limiter": "~1.0.0" + } + }, + "xml": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/xml/-/xml-1.0.1.tgz", + "integrity": "sha1-eLpyAgApxbyHuKgaPPzXS0ovweU=", + "dev": true + }, + "xml-name-validator": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/xml-name-validator/-/xml-name-validator-3.0.0.tgz", + "integrity": "sha512-A5CUptxDsvxKJEU3yO6DuWBSJz/qizqzJKOMIfUJHETbBw/sFaDxgd6fxm1ewUaM0jZ444Fc5vC5ROYurg/4Pw==", + "dev": true + }, + "xml2js": { + "version": "0.4.19", + "resolved": "https://registry.npmjs.org/xml2js/-/xml2js-0.4.19.tgz", + "integrity": "sha512-esZnJZJOiJR9wWKMyuvSE1y6Dq5LCuJanqhxslH2bxM6duahNZ+HMpCLhBQGZkbX6xRf8x1Y2eJlgt2q3qo49Q==", + "dev": true, + "requires": { + "sax": ">=0.6.0", + "xmlbuilder": "~9.0.1" + } + }, + "xmlbuilder": { + "version": "9.0.7", + "resolved": "https://registry.npmjs.org/xmlbuilder/-/xmlbuilder-9.0.7.tgz", + "integrity": "sha1-Ey7mPS7FVlxVfiD0wi35rKaGsQ0=", + "dev": true + }, + "xmlchars": { + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/xmlchars/-/xmlchars-2.2.0.tgz", + "integrity": "sha512-JZnDKK8B0RCDw84FNdDAIpZK+JuJw+s7Lz8nksI7SIuU3UXJJslUthsi+uWBUYOwPFwW7W7PRLRfUKpxjtjFCw==", + "dev": true + }, + "xmlhttprequest-ssl": { + "version": "1.5.5", + "resolved": "https://registry.npmjs.org/xmlhttprequest-ssl/-/xmlhttprequest-ssl-1.5.5.tgz", + "integrity": "sha1-wodrBhaKrcQOV9l+gRkayPQ5iz4=", + "dev": true + }, + "xtend": { + "version": "4.0.2", + "resolved": "https://registry.npmjs.org/xtend/-/xtend-4.0.2.tgz", + "integrity": "sha512-LKYU1iAXJXUgAXn9URjiu+MWhyUXHsvfp7mcuYm9dSUKK0/CjtrUwFAxD82/mCWbtLsGjFIad0wIsod4zrTAEQ==", + "dev": true + }, + "y18n": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/y18n/-/y18n-4.0.0.tgz", + "integrity": "sha512-r9S/ZyXu/Xu9q1tYlpsLIsa3EeLXXk0VwlxqTcFRfg9EhMW+17kbt9G0NrgCmhGb5vT2hyhJZLfDGx+7+5Uj/w==" + }, + "yallist": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/yallist/-/yallist-4.0.0.tgz", + "integrity": "sha512-3wdGidZyq5PB084XLES5TpOSRA3wjXAlIWMhum2kRcv/41Sn2emQ0dycQW4uZXLejwKvg6EsvbdlVL+FYEct7A==", + "dev": true + }, + "yaml": { + "version": "1.10.0", + "resolved": "https://registry.npmjs.org/yaml/-/yaml-1.10.0.tgz", + "integrity": "sha512-yr2icI4glYaNG+KWONODapy2/jDdMSDnrONSjblABjD9B4Z5LgiircSt8m8sRZFNi08kG9Sm0uSHtEmP3zaEGg==", + "dev": true + }, + "yargs": { + "version": "15.3.0", + "resolved": "https://registry.npmjs.org/yargs/-/yargs-15.3.0.tgz", + "integrity": "sha512-g/QCnmjgOl1YJjGsnUg2SatC7NUYEiLXJqxNOQU9qSpjzGtGXda9b+OKccr1kLTy8BN9yqEyqfq5lxlwdc13TA==", + "requires": { + "cliui": "^6.0.0", + "decamelize": "^1.2.0", + "find-up": "^4.1.0", + "get-caller-file": "^2.0.1", + "require-directory": "^2.1.1", + "require-main-filename": "^2.0.0", + "set-blocking": "^2.0.0", + "string-width": "^4.2.0", + "which-module": "^2.0.0", + "y18n": "^4.0.0", + "yargs-parser": "^18.1.0" + } + }, + "yargs-parser": { + "version": "18.1.3", + "resolved": "https://registry.npmjs.org/yargs-parser/-/yargs-parser-18.1.3.tgz", + "integrity": "sha512-o50j0JeToy/4K6OZcaQmW6lyXXKhq7csREXcDwk2omFPJEwUNOVtJKvmDr9EI1fAJZUyZcRF7kxGBWmRXudrCQ==", + "requires": { + "camelcase": "^5.0.0", + "decamelize": "^1.2.0" + } + }, + "yeast": { + "version": "0.1.2", + "resolved": "https://registry.npmjs.org/yeast/-/yeast-0.1.2.tgz", + "integrity": "sha1-AI4G2AlDIMNy28L47XagymyKxBk=", + "dev": true + }, + "yeoman-environment": { + "version": "2.10.3", + "resolved": "https://registry.npmjs.org/yeoman-environment/-/yeoman-environment-2.10.3.tgz", + "integrity": "sha512-pLIhhU9z/G+kjOXmJ2bPFm3nejfbH+f1fjYRSOteEXDBrv1EoJE/e+kuHixSXfCYfTkxjYsvRaDX+1QykLCnpQ==", + "dev": true, + "requires": { + "chalk": "^2.4.1", + "debug": "^3.1.0", + "diff": "^3.5.0", + "escape-string-regexp": "^1.0.2", + "execa": "^4.0.0", + "globby": "^8.0.1", + "grouped-queue": "^1.1.0", + "inquirer": "^7.1.0", + "is-scoped": "^1.0.0", + "lodash": "^4.17.10", + "log-symbols": "^2.2.0", + "mem-fs": "^1.1.0", + "mem-fs-editor": "^6.0.0", + "npm-api": "^1.0.0", + "semver": "^7.1.3", + "strip-ansi": "^4.0.0", + "text-table": "^0.2.0", + "untildify": "^3.0.3", + "yeoman-generator": "^4.8.2" + }, + "dependencies": { + "@nodelib/fs.stat": { + "version": "1.1.3", + "resolved": "https://registry.npmjs.org/@nodelib/fs.stat/-/fs.stat-1.1.3.tgz", + "integrity": "sha512-shAmDyaQC4H92APFoIaVDHCx5bStIocgvbwQyxPRrbUY20V1EYTbSDchWbuwlMG3V17cprZhA6+78JfB+3DTPw==", + "dev": true + }, + "ansi-regex": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-3.0.0.tgz", + "integrity": "sha1-7QMXwyIGT3lGbAKWa922Bas32Zg=", + "dev": true + }, + "array-union": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/array-union/-/array-union-1.0.2.tgz", + "integrity": "sha1-mjRBDk9OPaI96jdb5b5w8kd47Dk=", + "dev": true, + "requires": { + "array-uniq": "^1.0.1" + } + }, + "arrify": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/arrify/-/arrify-1.0.1.tgz", + "integrity": "sha1-iYUI2iIm84DfkEcoRWhJwVAaSw0=", + "dev": 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" + } + }, + "debug": { + "version": "3.2.7", + "resolved": "https://registry.npmjs.org/debug/-/debug-3.2.7.tgz", + "integrity": "sha512-CFjzYYAi4ThfiQvizrFQevTTXHtnCqWfe7x1AhgEscTz6ZbLbfoLRLPugTQyBth6f8ZERVUSyWHFD/7Wu4t1XQ==", + "dev": true, + "requires": { + "ms": "^2.1.1" + } + }, + "dir-glob": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/dir-glob/-/dir-glob-2.0.0.tgz", + "integrity": "sha512-37qirFDz8cA5fimp9feo43fSuRo2gHwaIn6dXL8Ber1dGwUosDrGZeCCXq57WnIqE4aQ+u3eQZzsk1yOzhdwag==", + "dev": true, + "requires": { + "arrify": "^1.0.1", + "path-type": "^3.0.0" + } + }, + "execa": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/execa/-/execa-4.1.0.tgz", + "integrity": "sha512-j5W0//W7f8UxAn8hXVnwG8tLwdiUy4FJLcSupCg6maBYZDpyBvTApK7KyuI4bKj8KOh1r2YH+6ucuYtJv1bTZA==", + "dev": true, + "requires": { + "cross-spawn": "^7.0.0", + "get-stream": "^5.0.0", + "human-signals": "^1.1.1", + "is-stream": "^2.0.0", + "merge-stream": "^2.0.0", + "npm-run-path": "^4.0.0", + "onetime": "^5.1.0", + "signal-exit": "^3.0.2", + "strip-final-newline": "^2.0.0" + } + }, + "fast-glob": { + "version": "2.2.7", + "resolved": "https://registry.npmjs.org/fast-glob/-/fast-glob-2.2.7.tgz", + "integrity": "sha512-g1KuQwHOZAmOZMuBtHdxDtju+T2RT8jgCC9aANsbpdiDDTSnjgfuVsIBNKbUeJI3oKMRExcfNDtJl4OhbffMsw==", + "dev": true, + "requires": { + "@mrmlnc/readdir-enhanced": "^2.2.1", + "@nodelib/fs.stat": "^1.1.2", + "glob-parent": "^3.1.0", + "is-glob": "^4.0.0", + "merge2": "^1.2.3", + "micromatch": "^3.1.10" + } + }, + "get-stream": { + "version": "5.2.0", + "resolved": "https://registry.npmjs.org/get-stream/-/get-stream-5.2.0.tgz", + "integrity": "sha512-nBF+F1rAZVCu/p7rjzgA+Yb4lfYXrpl7a6VmJrU8wF9I1CKvP/QwPNZHnOlwbTkY6dvtFIzFMSyQXbLoTQPRpA==", + "dev": true, + "requires": { + "pump": "^3.0.0" + } + }, + "glob-parent": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-3.1.0.tgz", + "integrity": "sha1-nmr2KZ2NO9K9QEMIMr0RPfkGxa4=", + "dev": true, + "requires": { + "is-glob": "^3.1.0", + "path-dirname": "^1.0.0" + }, + "dependencies": { + "is-glob": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-3.1.0.tgz", + "integrity": "sha1-e6WuJCF4BKxwcHuWkiVnSGzD6Eo=", + "dev": true, + "requires": { + "is-extglob": "^2.1.0" + } + } + } + }, + "globby": { + "version": "8.0.2", + "resolved": "https://registry.npmjs.org/globby/-/globby-8.0.2.tgz", + "integrity": "sha512-yTzMmKygLp8RUpG1Ymu2VXPSJQZjNAZPD4ywgYEaG7e4tBJeUQBO8OpXrf1RCNcEs5alsoJYPAMiIHP0cmeC7w==", + "dev": true, + "requires": { + "array-union": "^1.0.1", + "dir-glob": "2.0.0", + "fast-glob": "^2.0.2", + "glob": "^7.1.2", + "ignore": "^3.3.5", + "pify": "^3.0.0", + "slash": "^1.0.0" + } + }, + "ignore": { + "version": "3.3.10", + "resolved": "https://registry.npmjs.org/ignore/-/ignore-3.3.10.tgz", + "integrity": "sha512-Pgs951kaMm5GXP7MOvxERINe3gsaVjUWFm+UZPSq9xYriQAksyhg0csnS0KXSNRD5NmNdapXEpjxG49+AKh/ug==", + "dev": true + }, + "is-stream": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/is-stream/-/is-stream-2.0.0.tgz", + "integrity": "sha512-XCoy+WlUr7d1+Z8GgSuXmpuUFC9fOhRXglJMx+dwLKTkL44Cjd4W1Z5P+BQZpr+cR93aGP4S/s7Ftw6Nd/kiEw==", + "dev": true + }, + "log-symbols": { + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/log-symbols/-/log-symbols-2.2.0.tgz", + "integrity": "sha512-VeIAFslyIerEJLXHziedo2basKbMKtTw3vfn5IzG0XTjhAVEJyNHnL2p7vc+wBDSdQuUpNw3M2u6xb9QsAY5Eg==", + "dev": true, + "requires": { + "chalk": "^2.0.1" + } + }, + "npm-run-path": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/npm-run-path/-/npm-run-path-4.0.1.tgz", + "integrity": "sha512-S48WzZW777zhNIrn7gxOlISNAqi9ZC/uQFnRdbeIHhZhCA6UqpkOT8T1G7BvfdgP4Er8gF4sUbaS0i7QvIfCWw==", + "dev": true, + "requires": { + "path-key": "^3.0.0" + } + }, + "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-type": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/path-type/-/path-type-3.0.0.tgz", + "integrity": "sha512-T2ZUsdZFHgA3u4e5PfPbjd7HDDpxPnQb5jN0SrDsjNSuVXHJqtwTnWqG0B1jZrgmJ/7lj1EmVIByWt1gxGkWvg==", + "dev": true, + "requires": { + "pify": "^3.0.0" + } + }, + "pify": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/pify/-/pify-3.0.0.tgz", + "integrity": "sha1-5aSs0sEB/fPZpNB/DbxNtJ3SgXY=", + "dev": true + }, + "semver": { + "version": "7.3.2", + "resolved": "https://registry.npmjs.org/semver/-/semver-7.3.2.tgz", + "integrity": "sha512-OrOb32TeeambH6UrhtShmF7CRDqhL6/5XpPNp2DuRH6+9QLw/orhp72j87v8Qa1ScDkvrrBNpZcDejAirJmfXQ==", + "dev": true + }, + "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 + }, + "slash": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/slash/-/slash-1.0.0.tgz", + "integrity": "sha1-xB8vbDn8FtHNF61LXYlhFK5HDVU=", + "dev": true + }, + "strip-ansi": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-4.0.0.tgz", + "integrity": "sha1-qEeQIusaw2iocTibY1JixQXuNo8=", + "dev": true, + "requires": { + "ansi-regex": "^3.0.0" + } + }, + "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" + } + } + } + }, + "yeoman-generator": { + "version": "4.11.0", + "resolved": "https://registry.npmjs.org/yeoman-generator/-/yeoman-generator-4.11.0.tgz", + "integrity": "sha512-++t6t2Z6HjL5F1/UM7+uNvGknKmQdF8tstJx8WKzsUSEpB+19kLVtapSfQIh9uWqm0L59fLWDzUui//WXoynPw==", + "dev": true, + "requires": { + "async": "^2.6.2", + "chalk": "^2.4.2", + "cli-table": "^0.3.1", + "cross-spawn": "^6.0.5", + "dargs": "^6.1.0", + "dateformat": "^3.0.3", + "debug": "^4.1.1", + "diff": "^4.0.1", + "error": "^7.0.2", + "find-up": "^3.0.0", + "github-username": "^3.0.0", + "grouped-queue": "^1.1.0", + "istextorbinary": "^2.5.1", + "lodash": "^4.17.11", + "make-dir": "^3.0.0", + "mem-fs-editor": "^7.0.1", + "minimist": "^1.2.5", + "pretty-bytes": "^5.2.0", + "read-chunk": "^3.2.0", + "read-pkg-up": "^5.0.0", + "rimraf": "^2.6.3", + "run-async": "^2.0.0", + "semver": "^7.2.1", + "shelljs": "^0.8.3", + "text-table": "^0.2.0", + "through2": "^3.0.1", + "yeoman-environment": "^2.9.5" + }, + "dependencies": { + "@nodelib/fs.stat": { + "version": "1.1.3", + "resolved": "https://registry.npmjs.org/@nodelib/fs.stat/-/fs.stat-1.1.3.tgz", + "integrity": "sha512-shAmDyaQC4H92APFoIaVDHCx5bStIocgvbwQyxPRrbUY20V1EYTbSDchWbuwlMG3V17cprZhA6+78JfB+3DTPw==", + "dev": true + }, + "array-union": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/array-union/-/array-union-1.0.2.tgz", + "integrity": "sha1-mjRBDk9OPaI96jdb5b5w8kd47Dk=", + "dev": true, + "requires": { + "array-uniq": "^1.0.1" + } + }, + "async": { + "version": "2.6.3", + "resolved": "https://registry.npmjs.org/async/-/async-2.6.3.tgz", + "integrity": "sha512-zflvls11DCy+dQWzTW2dzuilv8Z5X/pjfmZOWba6TNIVDm+2UDaJmXSOXlasHKfNBs8oo3M0aT50fDEWfKZjXg==", + "dev": true, + "requires": { + "lodash": "^4.17.14" + } + }, + "diff": { + "version": "4.0.2", + "resolved": "https://registry.npmjs.org/diff/-/diff-4.0.2.tgz", + "integrity": "sha512-58lmxKSA4BNyLz+HHMUzlOEpg09FV+ev6ZMe3vJihgdxzgcwZ8VoEEPmALCZG9LmqfVoNMMKpttIYTVG6uDY7A==", + "dev": true + }, + "dir-glob": { + "version": "2.2.2", + "resolved": "https://registry.npmjs.org/dir-glob/-/dir-glob-2.2.2.tgz", + "integrity": "sha512-f9LBi5QWzIW3I6e//uxZoLBlUt9kcp66qo0sSCxL6YZKc75R1c4MFCoe/LaZiBGmgujvQdxc5Bn3QhfyvK5Hsw==", + "dev": true, + "requires": { + "path-type": "^3.0.0" + } + }, + "ejs": { + "version": "3.1.5", + "resolved": "https://registry.npmjs.org/ejs/-/ejs-3.1.5.tgz", + "integrity": "sha512-dldq3ZfFtgVTJMLjOe+/3sROTzALlL9E34V4/sDtUd/KlBSS0s6U1/+WPE1B4sj9CXHJpL1M6rhNJnc9Wbal9w==", + "dev": true, + "requires": { + "jake": "^10.6.1" + } + }, + "fast-glob": { + "version": "2.2.7", + "resolved": "https://registry.npmjs.org/fast-glob/-/fast-glob-2.2.7.tgz", + "integrity": "sha512-g1KuQwHOZAmOZMuBtHdxDtju+T2RT8jgCC9aANsbpdiDDTSnjgfuVsIBNKbUeJI3oKMRExcfNDtJl4OhbffMsw==", + "dev": true, + "requires": { + "@mrmlnc/readdir-enhanced": "^2.2.1", + "@nodelib/fs.stat": "^1.1.2", + "glob-parent": "^3.1.0", + "is-glob": "^4.0.0", + "merge2": "^1.2.3", + "micromatch": "^3.1.10" + } + }, + "find-up": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/find-up/-/find-up-3.0.0.tgz", + "integrity": "sha512-1yD6RmLI1XBfxugvORwlck6f75tYL+iR0jqwsOrOxMZyGYqUuDhJ0l4AXdO1iX/FTs9cBAMEk1gWSEx1kSbylg==", + "dev": true, + "requires": { + "locate-path": "^3.0.0" + } + }, + "glob": { + "version": "7.1.6", + "resolved": "https://registry.npmjs.org/glob/-/glob-7.1.6.tgz", + "integrity": "sha512-LwaxwyZ72Lk7vZINtNNrywX0ZuLyStrdDtabefZKAY5ZGJhVtgdznluResxNmPitE0SAO+O26sWTHeKSI2wMBA==", + "dev": true, + "requires": { + "fs.realpath": "^1.0.0", + "inflight": "^1.0.4", + "inherits": "2", + "minimatch": "^3.0.4", + "once": "^1.3.0", + "path-is-absolute": "^1.0.0" + } + }, + "glob-parent": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-3.1.0.tgz", + "integrity": "sha1-nmr2KZ2NO9K9QEMIMr0RPfkGxa4=", + "dev": true, + "requires": { + "is-glob": "^3.1.0", + "path-dirname": "^1.0.0" + }, + "dependencies": { + "is-glob": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-3.1.0.tgz", + "integrity": "sha1-e6WuJCF4BKxwcHuWkiVnSGzD6Eo=", + "dev": true, + "requires": { + "is-extglob": "^2.1.0" + } + } + } + }, + "globby": { + "version": "9.2.0", + "resolved": "https://registry.npmjs.org/globby/-/globby-9.2.0.tgz", + "integrity": "sha512-ollPHROa5mcxDEkwg6bPt3QbEf4pDQSNtd6JPL1YvOvAo/7/0VAm9TccUeoTmarjPw4pfUthSCqcyfNB1I3ZSg==", + "dev": true, + "requires": { + "@types/glob": "^7.1.1", + "array-union": "^1.0.2", + "dir-glob": "^2.2.2", + "fast-glob": "^2.2.6", + "glob": "^7.1.3", + "ignore": "^4.0.3", + "pify": "^4.0.1", + "slash": "^2.0.0" + } + }, + "ignore": { + "version": "4.0.6", + "resolved": "https://registry.npmjs.org/ignore/-/ignore-4.0.6.tgz", + "integrity": "sha512-cyFDKrqc/YdcWFniJhzI42+AzS+gNwmUzOSFcRCQYwySuBBBy/KjuxWLZ/FHEH6Moq1NizMOBWyTcv8O4OZIMg==", + "dev": true + }, + "locate-path": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-3.0.0.tgz", + "integrity": "sha512-7AO748wWnIhNqAuaty2ZWHkQHRSNfPVIsPIfwEOWO22AmaoVrWavlOcMR5nzTLNYvp36X220/maaRsrec1G65A==", + "dev": true, + "requires": { + "p-locate": "^3.0.0", + "path-exists": "^3.0.0" + } + }, + "mem-fs-editor": { + "version": "7.1.0", + "resolved": "https://registry.npmjs.org/mem-fs-editor/-/mem-fs-editor-7.1.0.tgz", + "integrity": "sha512-BH6QEqCXSqGeX48V7zu+e3cMwHU7x640NB8Zk8VNvVZniz+p4FK60pMx/3yfkzo6miI6G3a8pH6z7FeuIzqrzA==", + "dev": true, + "requires": { + "commondir": "^1.0.1", + "deep-extend": "^0.6.0", + "ejs": "^3.1.5", + "glob": "^7.1.4", + "globby": "^9.2.0", + "isbinaryfile": "^4.0.0", + "mkdirp": "^1.0.0", + "multimatch": "^4.0.0", + "rimraf": "^3.0.0", + "through2": "^3.0.2", + "vinyl": "^2.2.1" + }, + "dependencies": { + "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" + } + } + } + }, + "mkdirp": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/mkdirp/-/mkdirp-1.0.4.tgz", + "integrity": "sha512-vVqVZQyf3WLx2Shd0qJ9xuvqgAyKPLAiqITEtqW0oIUjzo3PePDd6fW9iFz30ef7Ysp/oiWqbhszeGWW2T6Gzw==", + "dev": true + }, + "p-locate": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-3.0.0.tgz", + "integrity": "sha512-x+12w/To+4GFfgJhBEpiDcLozRJGegY+Ei7/z0tSLkMmxGZNybVMSfWj9aJn8Z5Fc7dBUNJOOVgPv2H7IwulSQ==", + "dev": true, + "requires": { + "p-limit": "^2.0.0" + } + }, + "parse-json": { + "version": "5.1.0", + "resolved": "https://registry.npmjs.org/parse-json/-/parse-json-5.1.0.tgz", + "integrity": "sha512-+mi/lmVVNKFNVyLXV31ERiy2CY5E1/F6QtJFEzoChPRwwngMNXRDQ9GJ5WdE2Z2P4AujsOi0/+2qHID68KwfIQ==", + "dev": true, + "requires": { + "@babel/code-frame": "^7.0.0", + "error-ex": "^1.3.1", + "json-parse-even-better-errors": "^2.3.0", + "lines-and-columns": "^1.1.6" + } + }, + "path-exists": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/path-exists/-/path-exists-3.0.0.tgz", + "integrity": "sha1-zg6+ql94yxiSXqfYENe1mwEP1RU=", + "dev": true + }, + "path-type": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/path-type/-/path-type-3.0.0.tgz", + "integrity": "sha512-T2ZUsdZFHgA3u4e5PfPbjd7HDDpxPnQb5jN0SrDsjNSuVXHJqtwTnWqG0B1jZrgmJ/7lj1EmVIByWt1gxGkWvg==", + "dev": true, + "requires": { + "pify": "^3.0.0" + }, + "dependencies": { + "pify": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/pify/-/pify-3.0.0.tgz", + "integrity": "sha1-5aSs0sEB/fPZpNB/DbxNtJ3SgXY=", + "dev": true + } + } + }, + "pify": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/pify/-/pify-4.0.1.tgz", + "integrity": "sha512-uB80kBFb/tfd68bVleG9T5GGsGPjJrLAUpR5PZIrhBnIaRTQRjqdJSsIKkOP6OAIFbj7GOrcudc5pNjZ+geV2g==", + "dev": true + }, + "read-pkg": { + "version": "5.2.0", + "resolved": "https://registry.npmjs.org/read-pkg/-/read-pkg-5.2.0.tgz", + "integrity": "sha512-Ug69mNOpfvKDAc2Q8DRpMjjzdtrnv9HcSMX+4VsZxD1aZ6ZzrIE7rlzXBtWTyhULSMKg076AW6WR5iZpD0JiOg==", + "dev": true, + "requires": { + "@types/normalize-package-data": "^2.4.0", + "normalize-package-data": "^2.5.0", + "parse-json": "^5.0.0", + "type-fest": "^0.6.0" + } + }, + "read-pkg-up": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/read-pkg-up/-/read-pkg-up-5.0.0.tgz", + "integrity": "sha512-XBQjqOBtTzyol2CpsQOw8LHV0XbDZVG7xMMjmXAJomlVY03WOBRmYgDJETlvcg0H63AJvPRwT7GFi5rvOzUOKg==", + "dev": true, + "requires": { + "find-up": "^3.0.0", + "read-pkg": "^5.0.0" + } + }, + "rimraf": { + "version": "2.7.1", + "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-2.7.1.tgz", + "integrity": "sha512-uWjbaKIK3T1OSVptzX7Nl6PvQ3qAGtKEtVRjRuazjfL3Bx5eI409VZSqgND+4UNnmzLVdPj9FqFJNPqBZFve4w==", + "dev": true, + "requires": { + "glob": "^7.1.3" + } + }, + "semver": { + "version": "7.3.2", + "resolved": "https://registry.npmjs.org/semver/-/semver-7.3.2.tgz", + "integrity": "sha512-OrOb32TeeambH6UrhtShmF7CRDqhL6/5XpPNp2DuRH6+9QLw/orhp72j87v8Qa1ScDkvrrBNpZcDejAirJmfXQ==", + "dev": true + }, + "slash": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/slash/-/slash-2.0.0.tgz", + "integrity": "sha512-ZYKh3Wh2z1PpEXWr0MpSBZ0V6mZHAQfYevttO11c51CaWjGTaadiKZ+wVt1PbMlDV5qhMFslpZCemhwOK7C89A==", + "dev": true + }, + "through2": { + "version": "3.0.2", + "resolved": "https://registry.npmjs.org/through2/-/through2-3.0.2.tgz", + "integrity": "sha512-enaDQ4MUyP2W6ZyT6EsMzqBPZaM/avg8iuo+l2d3QCs0J+6RaqkHV/2/lOwDTueBHeJ/2LG9lrLW3d5rWPucuQ==", + "dev": true, + "requires": { + "inherits": "^2.0.4", + "readable-stream": "2 || 3" + } + }, + "type-fest": { + "version": "0.6.0", + "resolved": "https://registry.npmjs.org/type-fest/-/type-fest-0.6.0.tgz", + "integrity": "sha512-q+MB8nYR1KDLrgr4G5yemftpMC7/QLqVndBmEEdqzmNj5dcFOO4Oo8qlwZE3ULT3+Zim1F8Kq4cBnikNhlCMlg==", + "dev": true + } + } + }, + "yup": { + "version": "0.27.0", + "resolved": "https://registry.npmjs.org/yup/-/yup-0.27.0.tgz", + "integrity": "sha512-v1yFnE4+u9za42gG/b/081E7uNW9mUj3qtkmelLbW5YPROZzSH/KUUyJu9Wt8vxFJcT9otL/eZopS0YK1L5yPQ==", + "dev": true, + "requires": { + "@babel/runtime": "^7.0.0", + "fn-name": "~2.0.1", + "lodash": "^4.17.11", + "property-expr": "^1.5.0", + "synchronous-promise": "^2.0.6", + "toposort": "^2.0.2" + } + }, + "zone.js": { + "version": "0.10.3", + "resolved": "https://registry.npmjs.org/zone.js/-/zone.js-0.10.3.tgz", + "integrity": "sha512-LXVLVEq0NNOqK/fLJo3d0kfzd4sxwn2/h67/02pjCjfKDxgx1i9QqpvtHD8CrBnSSwMw5+dy11O7FRX5mkO7Cg==" + } + } +} diff --git a/package.json b/package.json new file mode 100644 index 0000000..4524345 --- /dev/null +++ b/package.json @@ -0,0 +1,123 @@ +{ + "name": "epasmed", + "version": "0.0.1-SNAPSHOT", + "description": "Description for epasmed", + "private": true, + "license": "UNLICENSED", + "cacheDirectories": [ + "node_modules" + ], + "dependencies": { + "@angular/common": "10.0.0", + "@angular/compiler": "10.0.0", + "@angular/core": "10.0.0", + "@angular/forms": "10.0.0", + "@angular/localize": "10.0.0", + "@angular/platform-browser": "10.0.0", + "@angular/platform-browser-dynamic": "10.0.0", + "@angular/router": "10.0.0", + "@fortawesome/angular-fontawesome": "0.6.1", + "@fortawesome/fontawesome-svg-core": "1.2.29", + "@fortawesome/free-solid-svg-icons": "5.13.1", + "@ng-bootstrap/ng-bootstrap": "6.1.0", + "@ngx-translate/core": "12.1.2", + "@ngx-translate/http-loader": "5.0.0", + "bootstrap": "4.5.0", + "moment": "2.27.0", + "ng-jhipster": "0.15.0", + "ngx-cookie-service": "3.0.4", + "ngx-infinite-scroll": "9.0.0", + "ngx-webstorage": "5.0.0", + "rxjs": "6.5.5", + "swagger-ui-dist": "3.25.1", + "tslib": "2.0.0", + "zone.js": "0.10.3" + }, + "devDependencies": { + "@angular/cli": "10.0.0", + "@angular/compiler-cli": "10.0.0", + "@ngtools/webpack": "10.0.0", + "@openapitools/openapi-generator-cli": "1.0.13-4.3.1", + "@types/jest": "26.0.3", + "@types/node": "13.13.4", + "@typescript-eslint/eslint-plugin": "2.30.0", + "@typescript-eslint/eslint-plugin-tslint": "2.30.0", + "@typescript-eslint/parser": "2.30.0", + "autoprefixer": "9.8.4", + "browser-sync": "2.26.7", + "browser-sync-webpack-plugin": "2.2.2", + "codelyzer": "5.2.2", + "copy-webpack-plugin": "6.0.2", + "css-loader": "3.6.0", + "eslint": "6.8.0", + "eslint-config-jhipster": "0.0.1", + "eslint-config-prettier": "6.11.0", + "eslint-loader": "4.0.2", + "file-loader": "6.0.0", + "friendly-errors-webpack-plugin": "1.7.0", + "generator-jhipster": "6.10.4", + "html-loader": "1.1.0", + "html-webpack-plugin": "4.3.0", + "husky": "4.2.5", + "jest": "26.1.0", + "jest-date-mock": "1.0.8", + "jest-junit": "11.0.1", + "jest-preset-angular": "8.2.1", + "jest-sonar-reporter": "2.0.0", + "lint-staged": "8.2.1", + "mini-css-extract-plugin": "0.9.0", + "moment-locales-webpack-plugin": "1.2.0", + "optimize-css-assets-webpack-plugin": "5.0.3", + "postcss-loader": "3.0.0", + "prettier": "2.1.2", + "rimraf": "3.0.2", + "sass": "1.26.9", + "sass-loader": "8.0.2", + "simple-progress-webpack-plugin": "1.1.2", + "style-loader": "1.2.1", + "terser-webpack-plugin": "3.0.6", + "thread-loader": "2.1.3", + "to-string-loader": "1.1.6", + "ts-loader": "7.0.5", + "tslint": "6.1.2", + "typescript": "3.9.5", + "webpack": "4.43.0", + "webpack-bundle-analyzer": "3.8.0", + "webpack-cli": "3.3.12", + "webpack-dev-server": "3.11.0", + "webpack-merge": "4.2.2", + "webpack-notifier": "1.8.0", + "workbox-webpack-plugin": "5.1.3", + "write-file-webpack-plugin": "4.5.1" + }, + "engines": { + "node": ">=12.16.1" + }, + "scripts": { + "prettier:format": "prettier --write \"{,src/**/}*.{md,json,ts,css,scss,yml}\"", + "lint": "eslint . --ext .js,.ts", + "lint:fix": "npm run lint -- --fix", + "ngc": "ngc -p tsconfig.app.json", + "cleanup": "rimraf target/classes/static/", + "clean-www": "rimraf target/classes/static/app/{src,target/}", + "start": "npm run webpack:dev", + "start-tls": "npm run webpack:dev -- --env.tls", + "serve": "npm run start", + "build": "npm run webpack:prod", + "test": "npm run lint && jest --coverage --logHeapUsage -w=2 --config src/test/javascript/jest.conf.js", + "test:watch": "npm run test -- --watch", + "webpack:dev": "npm run webpack-dev-server -- --config webpack/webpack.dev.js --inline --hot --port=9060 --watch-content-base --env.stats=minimal", + "webpack:dev-verbose": "npm run webpack-dev-server -- --config webpack/webpack.dev.js --inline --hot --port=9060 --watch-content-base --profile --progress --env.stats=normal", + "webpack:build:main": "npm run webpack -- --config webpack/webpack.dev.js --env.stats=minimal", + "webpack:build": "npm run cleanup && npm run webpack:build:main", + "webpack:prod:main": "npm run webpack -- --config webpack/webpack.prod.js --profile", + "webpack:prod": "npm run cleanup && npm run webpack:prod:main && npm run clean-www", + "webpack:test": "npm run test", + "webpack-dev-server": "node --max_old_space_size=4096 node_modules/webpack-dev-server/bin/webpack-dev-server.js", + "webpack": "node --max_old_space_size=4096 node_modules/webpack/bin/webpack.js" + }, + "jestSonar": { + "reportPath": "target/test-results/jest", + "reportFile": "TESTS-results-sonar.xml" + } +} diff --git a/pom.xml b/pom.xml new file mode 100644 index 0000000..cf4a152 --- /dev/null +++ b/pom.xml @@ -0,0 +1,1202 @@ + + + 4.0.0 + + it.cnr.isti.epasmed + epasmed + 0.0.1-SNAPSHOT + jar + epasmed + + + + + + + + + + + + + + 3.3.9 + 1.8 + v12.16.1 + 6.14.5 + UTF-8 + UTF-8 + ${project.build.directory}/test-results + yyyyMMddHHmmss + ${java.version} + ${java.version} + -Djava.security.egd=file:/dev/./urandom -Xmx256m + jdt_apt + false + + + + + + + 3.9.0 + + 2.2.7.RELEASE + + 5.4.15.Final + + 3.24.0-GA + + 3.9.0 + 3.8 + 1.4.200 + 2.0.1.Final + 2.3.3 + 0.14.1 + 1.3.1.Final + 0.2.1 + + 3.1.0 + 3.8.1 + 3.2.0 + 2.10 + 3.0.0-M3 + 3.0.0-M4 + 2.2.1 + 3.1.0 + 3.0.0-M4 + 3.2.3 + 3.1.1 + 8.32 + 0.0.4.RELEASE + 1.10.0 + 4.0.0 + 0.8.5 + 2.4.0 + 1.0.0 + 4.3.1 + 1.0.0 + 3.7.0.1746 + ${project.build.directory}/jacoco/test + ${jacoco.utReportFolder}/test.exec + ${project.build.directory}/jacoco/integrationTest + ${jacoco.itReportFolder}/integrationTest.exec + ${project.testresult.directory}/test + ${project.testresult.directory}/integrationTest + + 1.18.12 + true + + + + + + + io.github.jhipster + jhipster-dependencies + ${jhipster-dependencies.version} + pom + import + + + + + + + + io.github.jhipster + jhipster-framework + + + javax.annotation + javax.annotation-api + + + + com.fasterxml.jackson.module + jackson-module-jaxb-annotations + + + com.fasterxml.jackson.datatype + jackson-datatype-hibernate5 + + + com.fasterxml.jackson.datatype + jackson-datatype-hppc + + + com.fasterxml.jackson.datatype + jackson-datatype-jsr310 + + + com.h2database + h2 + test + + + io.springfox + springfox-swagger2 + + + io.springfox + springfox-bean-validators + + + com.zaxxer + HikariCP + + + org.openapitools + jackson-databind-nullable + ${jackson-databind-nullable.version} + + + org.testcontainers + postgresql + test + + + org.hibernate + hibernate-jpamodelgen + provided + + + org.hibernate + hibernate-core + + + org.hibernate.validator + hibernate-validator + + + org.liquibase + liquibase-core + + ${liquibase.version} + + + net.logstash.logback + logstash-logback-encoder + + + org.postgresql + postgresql + + + org.mapstruct + mapstruct + + + org.mapstruct + mapstruct-processor + provided + + + org.springframework.boot + spring-boot-configuration-processor + provided + + + org.springframework.boot + spring-boot-loader-tools + + + org.springframework.boot + spring-boot-starter-actuator + + + org.springframework.boot + spring-boot-starter-aop + + + org.springframework.boot + spring-boot-starter-data-jpa + + + org.springframework.data + spring-data-jdbc + + + org.springframework.boot + spring-boot-starter-logging + + + org.springframework.boot + spring-boot-starter-mail + + + org.springframework.boot + spring-boot-starter-security + + + org.springframework.boot + spring-boot-starter-thymeleaf + + + org.springframework.boot + spring-boot-starter-web + + + org.springframework.boot + spring-boot-starter-test + test + + + org.junit.vintage + junit-vintage-engine + + + + + org.springframework.boot + spring-boot-test + test + + + org.springframework.security + spring-security-test + test + + + com.tngtech.archunit + archunit-junit5-api + ${archunit-junit5.version} + test + + + com.tngtech.archunit + archunit-junit5-engine + ${archunit-junit5.version} + test + + + org.zalando + problem-spring-web + + + + org.springframework.boot + spring-boot-starter-cloud-connectors + + + org.springframework.security + spring-security-data + + + io.micrometer + micrometer-registry-prometheus + + + io.dropwizard.metrics + metrics-core + + + + + org.projectlombok + lombok + provided + + + + + spring-boot:run + + + org.apache.maven.plugins + maven-compiler-plugin + + + org.apache.maven.plugins + maven-checkstyle-plugin + + + org.apache.maven.plugins + maven-javadoc-plugin + + + org.apache.maven.plugins + maven-eclipse-plugin + + + org.apache.maven.plugins + maven-enforcer-plugin + + + org.apache.maven.plugins + maven-failsafe-plugin + + + org.apache.maven.plugins + maven-idea-plugin + + + org.apache.maven.plugins + maven-resources-plugin + + + org.apache.maven.plugins + maven-surefire-plugin + + + org.jacoco + jacoco-maven-plugin + + + org.sonarsource.scanner.maven + sonar-maven-plugin + + + org.liquibase + liquibase-maven-plugin + + + org.springframework.boot + spring-boot-maven-plugin + + + com.google.cloud.tools + jib-maven-plugin + + + org.codehaus.mojo + properties-maven-plugin + + + + org.openapitools + openapi-generator-maven-plugin + + + + + + + + + org.apache.maven.plugins + maven-checkstyle-plugin + ${maven-checkstyle.version} + + + com.puppycrawl.tools + checkstyle + ${checkstyle.version} + + + io.spring.nohttp + nohttp-checkstyle + ${spring-nohttp-checkstyle.version} + + + + checkstyle.xml + pom.xml,README.md + .git/**/*,target/**/*,node_modules/**/*,node/**/* + ./ + + + + + check + + + + + + org.apache.maven.plugins + maven-compiler-plugin + ${maven-compiler-plugin.version} + + ${java.version} + ${java.version} + + + org.projectlombok + lombok + ${lombok.version} + + + org.springframework.boot + spring-boot-configuration-processor + ${spring-boot.version} + + + org.mapstruct + mapstruct-processor + ${mapstruct.version} + + + + org.hibernate + hibernate-jpamodelgen + ${hibernate.version} + + + org.glassfish.jaxb + jaxb-runtime + ${jaxb-runtime.version} + + + + + + + org.apache.maven.plugins + maven-javadoc-plugin + ${maven-javadoc-plugin.version} + + ${maven.compiler.source} + + + + org.apache.maven.plugins + maven-war-plugin + ${maven-war-plugin.version} + + + default-war + + war + + package + + + + WEB-INF/**,META-INF/** + false + target/classes/static/ + + + src/main/webapp + + WEB-INF/** + + + + + + + com.github.eirslett + frontend-maven-plugin + ${frontend-maven-plugin.version} + + + org.codehaus.mojo + properties-maven-plugin + ${properties-maven-plugin.version} + + + initialize + + read-project-properties + + + + sonar-project.properties + + + + + + + + pl.project13.maven + git-commit-id-plugin + ${git-commit-id-plugin.version} + + + + revision + + + + + false + false + true + + ^git.commit.id.abbrev$ + ^git.commit.id.describe$ + ^git.branch$ + + + + + org.jacoco + jacoco-maven-plugin + ${jacoco-maven-plugin.version} + + + pre-unit-tests + + prepare-agent + + + + ${jacoco.utReportFile} + + + + + post-unit-test + test + + report + + + ${jacoco.utReportFile} + ${jacoco.utReportFolder} + + + + pre-integration-tests + + prepare-agent-integration + + + + ${jacoco.itReportFile} + + + + + post-integration-tests + post-integration-test + + report-integration + + + ${jacoco.itReportFile} + ${jacoco.itReportFolder} + + + + + + com.google.cloud.tools + jib-maven-plugin + ${jib-maven-plugin.version} + + + + openjdk:8-jre + + + epasmed:latest + + + + bash + + /entrypoint.sh + + + 8080 + + + ALWAYS + 0 + + USE_CURRENT_TIMESTAMP + + + + + src/main/jib + + + src/main/javasecurity + /usr/local/openjdk-8/lib/security/ + + + + + /entrypoint.sh + 755 + + + /usr/local/openjdk-8/lib/security/java.security + 644 + + + + + + + org.liquibase + liquibase-maven-plugin + ${liquibase.version} + + ${project.basedir}/src/main/resources/config/liquibase/master.xml + ${project.basedir}/src/main/resources/config/liquibase/changelog/${maven.build.timestamp}_changelog.xml + org.h2.Driver + jdbc:h2:file:${project.build.directory}/h2db/db/epasmed + + epasmed + + hibernate:spring:it.cnr.isti.epasmed.domain?dialect=org.hibernate.dialect.H2Dialect&hibernate.physical_naming_strategy=org.springframework.boot.orm.jpa.hibernate.SpringPhysicalNamingStrategy&hibernate.implicit_naming_strategy=org.springframework.boot.orm.jpa.hibernate.SpringImplicitNamingStrategy + true + debug + !test + + + + org.liquibase + liquibase-core + ${liquibase.version} + + + org.liquibase.ext + liquibase-hibernate5 + ${liquibase-hibernate5.version} + + + org.springframework.boot + spring-boot-starter-data-jpa + ${spring-boot.version} + + + javax.validation + validation-api + ${validation-api.version} + + + org.javassist + javassist + ${javassist.version} + + + com.h2database + h2 + ${h2.version} + + + + + maven-clean-plugin + ${maven-clean-plugin.version} + + + org.apache.maven.plugins + maven-eclipse-plugin + ${maven-eclipse-plugin.version} + + true + true + + + + org.apache.maven.plugins + maven-enforcer-plugin + ${maven-enforcer-plugin.version} + + + enforce-versions + + enforce + + + + enforce-dependencyConvergence + + + + + false + + + enforce + + + + + + + You are running an older version of Maven. JHipster + requires at least Maven ${maven.version} + [${maven.version},) + + + You are running an incompatible version of Java. + JHipster supports JDK 8 to 14. + [1.8,15) + + + + + + org.apache.maven.plugins + maven-idea-plugin + ${maven-idea-plugin.version} + + node_modules + + + + org.apache.maven.plugins + maven-resources-plugin + ${maven-resources-plugin.version} + + + default-resources + validate + + copy-resources + + + ${project.build.directory}/classes + false + + # + + + + src/main/resources/ + true + + config/*.yml + + + + src/main/resources/ + false + + config/*.yml + + + + + + + + + org.apache.maven.plugins + maven-surefire-plugin + ${maven-surefire-plugin.version} + + + alphabetical + ${junit.utReportFolder} + + **/*IT* + **/*IntTest* + + + + + org.apache.maven.plugins + maven-failsafe-plugin + ${maven-failsafe-plugin.version} + + + ${project.build.outputDirectory} + + alphabetical + ${junit.itReportFolder} + + **/*IT* + **/*IntTest* + + + + + integration-test + + integration-test + + + + verify + + verify + + + + + + + org.openapitools + openapi-generator-maven-plugin + ${openapi-generator-maven-plugin.version} + + + + generate + + + ${project.basedir}/src/main/resources/swagger/api.yml + spring + it.cnr.isti.epasmed.web.api + it.cnr.isti.epasmed.web.api.model + ApiUtil.java + Problem=org.zalando.problem.Problem + false + + true + epasmed + + + + + + + org.sonarsource.scanner.maven + sonar-maven-plugin + ${sonar-maven-plugin.version} + + + org.springframework.boot + spring-boot-maven-plugin + ${spring-boot.version} + + + + repackage + + + + + ${start-class} + true + + + + + + + + + + no-liquibase + + ,no-liquibase + + + + swagger + + ,swagger + + + + tls + + ,tls + + + + webpack + + true + + + + com.h2database + h2 + + + + + + com.github.eirslett + frontend-maven-plugin + + + install node and npm + + install-node-and-npm + + + ${node.version} + ${npm.version} + + + + npm install + + npm + + + + webpack build dev + + npm + + generate-resources + + run webpack:build + + ${project.version} + + false + + + + + + + + + dev${profile.no-liquibase} + + + + dev + + true + + + + org.springframework.boot + spring-boot-starter-undertow + + + org.springframework.boot + spring-boot-devtools + true + + + com.h2database + h2 + + + + + dev${profile.tls}${profile.no-liquibase} + + + + prod + + + org.springframework.boot + spring-boot-starter-undertow + + + + + + maven-clean-plugin + + + + target/classes/static/ + + + + + + org.springframework.boot + spring-boot-maven-plugin + + + pre-integration-test + + start + + + ${skip.test} + + + + post-integration-test + + stop + + + ${skip.test} + + + + + build-info + + + + + + com.github.eirslett + frontend-maven-plugin + + + install node and npm + + install-node-and-npm + + + ${node.version} + ${npm.version} + + + + npm install + + npm + + + install + + + + webpack build test + + npm + + test + + run webpack:test + false + + + + webpack build prod + + npm + + generate-resources + + run webpack:prod + + ${project.version} + + false + + + + + + pl.project13.maven + git-commit-id-plugin + + + com.google.cloud.tools + jib-maven-plugin + + + + + + prod${profile.swagger}${profile.tls}${profile.no-liquibase} + + + + war + + + + org.apache.maven.plugins + maven-war-plugin + + + + + + + IDE + + + org.mapstruct + mapstruct-processor + + + org.hibernate + hibernate-jpamodelgen + + + + + + eclipse + + + m2e.version + + + + + + org.springframework.boot + spring-boot-starter-undertow + + + + + + + + org.eclipse.m2e + lifecycle-mapping + ${lifecycle-mapping.version} + + + + + + org.jacoco + + jacoco-maven-plugin + + + ${jacoco-maven-plugin.version} + + + prepare-agent + + + + + + + + + com.github.eirslett + frontend-maven-plugin + ${frontend-maven-plugin.version} + + install-node-and-npm + npm + + + + + + + + + org.openapitools + openapi-generator-maven-plugin + ${openapi-generator-maven-plugin.version} + + generate + + + + + + + + + + + + + + + + + diff --git a/postcss.config.js b/postcss.config.js new file mode 100644 index 0000000..a26de7e --- /dev/null +++ b/postcss.config.js @@ -0,0 +1,5 @@ +module.exports = { + plugins: [ + require('autoprefixer') + ] +} diff --git a/proxy.conf.json b/proxy.conf.json new file mode 100644 index 0000000..3bfe56a --- /dev/null +++ b/proxy.conf.json @@ -0,0 +1,7 @@ +{ + "*": { + "target": "http://localhost:8080", + "secure": false, + "loglevel": "debug" + } +} diff --git a/sonar-project.properties b/sonar-project.properties new file mode 100644 index 0000000..a37672a --- /dev/null +++ b/sonar-project.properties @@ -0,0 +1,30 @@ +sonar.projectKey=epasmed +sonar.projectName=epasmed generated by jhipster +sonar.projectVersion=1.0 + +sonar.sources=src/main/ +sonar.host.url=http://localhost:9001 + +sonar.tests=src/test/ +sonar.coverage.jacoco.xmlReportPaths=target/jacoco/test/jacoco.xml,target/jacoco/integrationTest/jacoco.xml +sonar.java.codeCoveragePlugin=jacoco +sonar.junit.reportPaths=target/test-results/test,target/test-results/integrationTest +sonar.testExecutionReportPaths=target/test-results/jest/TESTS-results-sonar.xml +sonar.typescript.lcov.reportPaths=target/test-results/lcov.info + +sonar.sourceEncoding=UTF-8 +sonar.exclusions=src/main/webapp/content/**/*.*, src/main/webapp/i18n/*.js, target/classes/static/**/*.* + +sonar.issue.ignore.multicriteria=S3437,S4684,UndocumentedApi,BoldAndItalicTagsCheck +# Rule https://sonarcloud.io/coding_rules?open=squid%3AS3437&rule_key=squid%3AS3437 is ignored, as a JPA-managed field cannot be transient +sonar.issue.ignore.multicriteria.S3437.resourceKey=src/main/java/**/* +sonar.issue.ignore.multicriteria.S3437.ruleKey=squid:S3437 +# Rule https://sonarcloud.io/coding_rules?open=squid%3AUndocumentedApi&rule_key=squid%3AUndocumentedApi is ignored, as we want to follow "clean code" guidelines and classes, methods and arguments names should be self-explanatory +sonar.issue.ignore.multicriteria.UndocumentedApi.resourceKey=src/main/java/**/* +sonar.issue.ignore.multicriteria.UndocumentedApi.ruleKey=squid:UndocumentedApi +# Rule https://sonarcloud.io/coding_rules?open=java%3AS4684&rule_key=java%3AS4684 +sonar.issue.ignore.multicriteria.S4684.resourceKey=src/main/java/**/* +sonar.issue.ignore.multicriteria.S4684.ruleKey=java:S4684 +# Rule https://sonarcloud.io/coding_rules?open=Web%3ABoldAndItalicTagsCheck&rule_key=Web%3ABoldAndItalicTagsCheck is ignored. Even if we agree that using the "i" tag is an awful practice, this is what is recommended by http://fontawesome.io/examples/ +sonar.issue.ignore.multicriteria.BoldAndItalicTagsCheck.resourceKey=src/main/webapp/app/**/*.* +sonar.issue.ignore.multicriteria.BoldAndItalicTagsCheck.ruleKey=Web:BoldAndItalicTagsCheck diff --git a/src/main/docker/app.yml b/src/main/docker/app.yml new file mode 100644 index 0000000..90d9e44 --- /dev/null +++ b/src/main/docker/app.yml @@ -0,0 +1,17 @@ +version: '2' +services: + epasmed-app: + image: epasmed + environment: + - _JAVA_OPTIONS=-Xmx512m -Xms256m + - SPRING_PROFILES_ACTIVE=prod,swagger + - MANAGEMENT_METRICS_EXPORT_PROMETHEUS_ENABLED=true + - APPLICATION_DATASOURCEEPASMED_DATASOURCE_URL=jdbc:postgresql://epasmed-postgresql:5432/epasmed + #- SPRING_DATASOURCE_URL=jdbc:postgresql://epasmed-postgresql:5432/epasmed + - JHIPSTER_SLEEP=30 # gives time for other services to boot before the application + ports: + - 8080:8080 + epasmed-postgresql: + extends: + file: postgresql.yml + service: epasmed-postgresql diff --git a/src/main/docker/grafana/provisioning/dashboards/JVM.json b/src/main/docker/grafana/provisioning/dashboards/JVM.json new file mode 100644 index 0000000..5104abc --- /dev/null +++ b/src/main/docker/grafana/provisioning/dashboards/JVM.json @@ -0,0 +1,3778 @@ +{ + "annotations": { + "list": [ + { + "builtIn": 1, + "datasource": "-- Grafana --", + "enable": true, + "hide": true, + "iconColor": "rgba(0, 211, 255, 1)", + "limit": 100, + "name": "Annotations & Alerts", + "showIn": 0, + "type": "dashboard" + }, + { + "datasource": "Prometheus", + "enable": true, + "expr": "resets(process_uptime_seconds{application=\"$application\", instance=\"$instance\"}[1m]) > 0", + "iconColor": "rgba(255, 96, 96, 1)", + "name": "Restart Detection", + "showIn": 0, + "step": "1m", + "tagKeys": "restart-tag", + "textFormat": "uptime reset", + "titleFormat": "Restart" + } + ] + }, + "description": "Dashboard for Micrometer instrumented applications (Java, Spring Boot, Micronaut)", + "editable": true, + "gnetId": 4701, + "graphTooltip": 1, + "iteration": 1553765841423, + "links": [], + "panels": [ + { + "content": "\n# Acknowledgments\n\nThank you to [Michael Weirauch](https://twitter.com/emwexx) for creating this dashboard: see original JVM (Micrometer) dashboard at [https://grafana.com/dashboards/4701](https://grafana.com/dashboards/4701)\n\n\n\n", + "gridPos": { + "h": 3, + "w": 24, + "x": 0, + "y": 0 + }, + "id": 141, + "links": [], + "mode": "markdown", + "timeFrom": null, + "timeShift": null, + "title": "Acknowledgments", + "type": "text" + }, + { + "collapsed": false, + "gridPos": { + "h": 1, + "w": 24, + "x": 0, + "y": 3 + }, + "id": 125, + "panels": [], + "repeat": null, + "title": "Quick Facts", + "type": "row" + }, + { + "cacheTimeout": null, + "colorBackground": false, + "colorValue": true, + "colors": ["rgba(245, 54, 54, 0.9)", "rgba(237, 129, 40, 0.89)", "rgba(50, 172, 45, 0.97)"], + "datasource": "Prometheus", + "decimals": 1, + "editable": true, + "error": false, + "format": "s", + "gauge": { + "maxValue": 100, + "minValue": 0, + "show": false, + "thresholdLabels": false, + "thresholdMarkers": true + }, + "gridPos": { + "h": 3, + "w": 6, + "x": 0, + "y": 4 + }, + "height": "", + "id": 63, + "interval": null, + "links": [], + "mappingType": 1, + "mappingTypes": [ + { + "name": "value to text", + "value": 1 + }, + { + "name": "range to text", + "value": 2 + } + ], + "maxDataPoints": 100, + "nullPointMode": "connected", + "nullText": null, + "postfix": "", + "postfixFontSize": "50%", + "prefix": "", + "prefixFontSize": "70%", + "rangeMaps": [ + { + "from": "null", + "text": "N/A", + "to": "null" + } + ], + "sparkline": { + "fillColor": "rgba(31, 118, 189, 0.18)", + "full": false, + "lineColor": "rgb(31, 120, 193)", + "show": false + }, + "tableColumn": "", + "targets": [ + { + "expr": "process_uptime_seconds{application=\"$application\", instance=\"$instance\"}", + "format": "time_series", + "intervalFactor": 2, + "legendFormat": "", + "metric": "", + "refId": "A", + "step": 14400 + } + ], + "thresholds": "", + "title": "Uptime", + "type": "singlestat", + "valueFontSize": "80%", + "valueMaps": [ + { + "op": "=", + "text": "N/A", + "value": "null" + } + ], + "valueName": "current" + }, + { + "cacheTimeout": null, + "colorBackground": false, + "colorValue": true, + "colors": ["rgba(245, 54, 54, 0.9)", "rgba(237, 129, 40, 0.89)", "rgba(50, 172, 45, 0.97)"], + "datasource": "Prometheus", + "decimals": null, + "editable": true, + "error": false, + "format": "dateTimeAsIso", + "gauge": { + "maxValue": 100, + "minValue": 0, + "show": false, + "thresholdLabels": false, + "thresholdMarkers": true + }, + "gridPos": { + "h": 3, + "w": 6, + "x": 6, + "y": 4 + }, + "height": "", + "id": 92, + "interval": null, + "links": [], + "mappingType": 1, + "mappingTypes": [ + { + "name": "value to text", + "value": 1 + }, + { + "name": "range to text", + "value": 2 + } + ], + "maxDataPoints": 100, + "nullPointMode": "connected", + "nullText": null, + "postfix": "", + "postfixFontSize": "50%", + "prefix": "", + "prefixFontSize": "70%", + "rangeMaps": [ + { + "from": "null", + "text": "N/A", + "to": "null" + } + ], + "sparkline": { + "fillColor": "rgba(31, 118, 189, 0.18)", + "full": false, + "lineColor": "rgb(31, 120, 193)", + "show": false + }, + "tableColumn": "", + "targets": [ + { + "expr": "process_start_time_seconds{application=\"$application\", instance=\"$instance\"}*1000", + "format": "time_series", + "intervalFactor": 2, + "legendFormat": "", + "metric": "", + "refId": "A", + "step": 14400 + } + ], + "thresholds": "", + "title": "Start time", + "type": "singlestat", + "valueFontSize": "70%", + "valueMaps": [ + { + "op": "=", + "text": "N/A", + "value": "null" + } + ], + "valueName": "current" + }, + { + "cacheTimeout": null, + "colorBackground": false, + "colorValue": true, + "colors": ["rgba(50, 172, 45, 0.97)", "rgba(237, 129, 40, 0.89)", "rgba(245, 54, 54, 0.9)"], + "datasource": "Prometheus", + "decimals": 2, + "editable": true, + "error": false, + "format": "percent", + "gauge": { + "maxValue": 100, + "minValue": 0, + "show": false, + "thresholdLabels": false, + "thresholdMarkers": true + }, + "gridPos": { + "h": 3, + "w": 6, + "x": 12, + "y": 4 + }, + "id": 65, + "interval": null, + "links": [], + "mappingType": 1, + "mappingTypes": [ + { + "name": "value to text", + "value": 1 + }, + { + "name": "range to text", + "value": 2 + } + ], + "maxDataPoints": 100, + "nullPointMode": "connected", + "nullText": null, + "postfix": "", + "postfixFontSize": "50%", + "prefix": "", + "prefixFontSize": "70%", + "rangeMaps": [ + { + "from": "null", + "text": "N/A", + "to": "null" + } + ], + "sparkline": { + "fillColor": "rgba(31, 118, 189, 0.18)", + "full": false, + "lineColor": "rgb(31, 120, 193)", + "show": false + }, + "tableColumn": "", + "targets": [ + { + "expr": "sum(jvm_memory_used_bytes{application=\"$application\", instance=\"$instance\", area=\"heap\"})*100/sum(jvm_memory_max_bytes{application=\"$application\",instance=\"$instance\", area=\"heap\"})", + "format": "time_series", + "intervalFactor": 2, + "legendFormat": "", + "refId": "A", + "step": 14400 + } + ], + "thresholds": "70,90", + "title": "Heap used", + "type": "singlestat", + "valueFontSize": "80%", + "valueMaps": [ + { + "op": "=", + "text": "N/A", + "value": "null" + } + ], + "valueName": "current" + }, + { + "cacheTimeout": null, + "colorBackground": false, + "colorValue": true, + "colors": ["rgba(50, 172, 45, 0.97)", "rgba(237, 129, 40, 0.89)", "rgba(245, 54, 54, 0.9)"], + "datasource": "Prometheus", + "decimals": 2, + "editable": true, + "error": false, + "format": "percent", + "gauge": { + "maxValue": 100, + "minValue": 0, + "show": false, + "thresholdLabels": false, + "thresholdMarkers": true + }, + "gridPos": { + "h": 3, + "w": 6, + "x": 18, + "y": 4 + }, + "id": 75, + "interval": null, + "links": [], + "mappingType": 2, + "mappingTypes": [ + { + "name": "value to text", + "value": 1 + }, + { + "name": "range to text", + "value": 2 + } + ], + "maxDataPoints": 100, + "nullPointMode": "connected", + "nullText": null, + "postfix": "", + "postfixFontSize": "50%", + "prefix": "", + "prefixFontSize": "70%", + "rangeMaps": [ + { + "from": "null", + "text": "N/A", + "to": "null" + }, + { + "from": "-99999999999999999999999999999999", + "text": "N/A", + "to": "0" + } + ], + "sparkline": { + "fillColor": "rgba(31, 118, 189, 0.18)", + "full": false, + "lineColor": "rgb(31, 120, 193)", + "show": false + }, + "tableColumn": "", + "targets": [ + { + "expr": "sum(jvm_memory_used_bytes{application=\"$application\", instance=\"$instance\", area=\"nonheap\"})*100/sum(jvm_memory_max_bytes{application=\"$application\",instance=\"$instance\", area=\"nonheap\"})", + "format": "time_series", + "intervalFactor": 2, + "legendFormat": "", + "refId": "A", + "step": 14400 + } + ], + "thresholds": "70,90", + "title": "Non-Heap used", + "type": "singlestat", + "valueFontSize": "80%", + "valueMaps": [ + { + "op": "=", + "text": "N/A", + "value": "null" + }, + { + "op": "=", + "text": "x", + "value": "" + } + ], + "valueName": "current" + }, + { + "collapsed": false, + "gridPos": { + "h": 1, + "w": 24, + "x": 0, + "y": 7 + }, + "id": 126, + "panels": [], + "repeat": null, + "title": "I/O Overview", + "type": "row" + }, + { + "aliasColors": {}, + "bars": false, + "dashLength": 10, + "dashes": false, + "datasource": "Prometheus", + "fill": 1, + "gridPos": { + "h": 7, + "w": 8, + "x": 0, + "y": 8 + }, + "id": 111, + "legend": { + "avg": false, + "current": true, + "max": false, + "min": false, + "show": true, + "total": false, + "values": true + }, + "lines": true, + "linewidth": 1, + "links": [], + "nullPointMode": "null", + "paceLength": 10, + "percentage": false, + "pointradius": 5, + "points": false, + "renderer": "flot", + "seriesOverrides": [], + "spaceLength": 10, + "stack": false, + "steppedLine": false, + "targets": [ + { + "expr": "sum(rate(http_server_requests_seconds_count{application=\"$application\", instance=\"$instance\"}[1m]))", + "format": "time_series", + "intervalFactor": 1, + "legendFormat": "HTTP", + "refId": "A" + } + ], + "thresholds": [], + "timeFrom": null, + "timeRegions": [], + "timeShift": null, + "title": "Rate", + "tooltip": { + "shared": true, + "sort": 0, + "value_type": "individual" + }, + "type": "graph", + "xaxis": { + "buckets": null, + "mode": "time", + "name": null, + "show": true, + "values": [] + }, + "yaxes": [ + { + "decimals": null, + "format": "ops", + "label": null, + "logBase": 1, + "max": null, + "min": "0", + "show": true + }, + { + "format": "short", + "label": null, + "logBase": 1, + "max": null, + "min": null, + "show": true + } + ], + "yaxis": { + "align": false, + "alignLevel": null + } + }, + { + "aliasColors": { + "HTTP": "#890f02", + "HTTP - 5xx": "#bf1b00" + }, + "bars": false, + "dashLength": 10, + "dashes": false, + "datasource": "Prometheus", + "fill": 1, + "gridPos": { + "h": 7, + "w": 8, + "x": 8, + "y": 8 + }, + "id": 112, + "legend": { + "avg": false, + "current": true, + "max": false, + "min": false, + "show": true, + "total": false, + "values": true + }, + "lines": true, + "linewidth": 1, + "links": [], + "nullPointMode": "null", + "paceLength": 10, + "percentage": false, + "pointradius": 5, + "points": false, + "renderer": "flot", + "seriesOverrides": [], + "spaceLength": 10, + "stack": false, + "steppedLine": false, + "targets": [ + { + "expr": "sum(rate(http_server_requests_seconds_count{application=\"$application\", instance=\"$instance\", status=~\"5..\"}[1m]))", + "format": "time_series", + "intervalFactor": 1, + "legendFormat": "HTTP - 5xx", + "refId": "A" + } + ], + "thresholds": [], + "timeFrom": null, + "timeRegions": [], + "timeShift": null, + "title": "Errors", + "tooltip": { + "shared": true, + "sort": 0, + "value_type": "individual" + }, + "type": "graph", + "xaxis": { + "buckets": null, + "mode": "time", + "name": null, + "show": true, + "values": [] + }, + "yaxes": [ + { + "decimals": null, + "format": "ops", + "label": null, + "logBase": 1, + "max": null, + "min": "0", + "show": true + }, + { + "format": "short", + "label": null, + "logBase": 1, + "max": null, + "min": null, + "show": true + } + ], + "yaxis": { + "align": false, + "alignLevel": null + } + }, + { + "aliasColors": {}, + "bars": false, + "dashLength": 10, + "dashes": false, + "datasource": "Prometheus", + "fill": 1, + "gridPos": { + "h": 7, + "w": 8, + "x": 16, + "y": 8 + }, + "id": 113, + "legend": { + "avg": false, + "current": true, + "max": false, + "min": false, + "show": true, + "total": false, + "values": true + }, + "lines": true, + "linewidth": 1, + "links": [], + "nullPointMode": "null", + "paceLength": 10, + "percentage": false, + "pointradius": 5, + "points": false, + "renderer": "flot", + "seriesOverrides": [], + "spaceLength": 10, + "stack": false, + "steppedLine": false, + "targets": [ + { + "expr": "sum(rate(http_server_requests_seconds_sum{application=\"$application\", instance=\"$instance\", status!~\"5..\"}[1m]))/sum(rate(http_server_requests_seconds_count{application=\"$application\", instance=\"$instance\", status!~\"5..\"}[1m]))", + "format": "time_series", + "hide": false, + "intervalFactor": 1, + "legendFormat": "HTTP - AVG", + "refId": "A" + }, + { + "expr": "max(http_server_requests_seconds_max{application=\"$application\", instance=\"$instance\", status!~\"5..\"})", + "format": "time_series", + "hide": false, + "intervalFactor": 1, + "legendFormat": "HTTP - MAX", + "refId": "B" + } + ], + "thresholds": [], + "timeFrom": null, + "timeRegions": [], + "timeShift": null, + "title": "Duration", + "tooltip": { + "shared": true, + "sort": 0, + "value_type": "individual" + }, + "type": "graph", + "xaxis": { + "buckets": null, + "mode": "time", + "name": null, + "show": true, + "values": [] + }, + "yaxes": [ + { + "format": "s", + "label": null, + "logBase": 1, + "max": null, + "min": "0", + "show": true + }, + { + "format": "short", + "label": null, + "logBase": 1, + "max": null, + "min": null, + "show": true + } + ], + "yaxis": { + "align": false, + "alignLevel": null + } + }, + { + "collapsed": false, + "gridPos": { + "h": 1, + "w": 24, + "x": 0, + "y": 15 + }, + "id": 127, + "panels": [], + "repeat": null, + "title": "JVM Memory", + "type": "row" + }, + { + "aliasColors": {}, + "bars": false, + "dashLength": 10, + "dashes": false, + "datasource": "Prometheus", + "editable": true, + "error": false, + "fill": 1, + "grid": { + "leftLogBase": 1, + "leftMax": null, + "leftMin": null, + "rightLogBase": 1, + "rightMax": null, + "rightMin": null + }, + "gridPos": { + "h": 7, + "w": 8, + "x": 0, + "y": 16 + }, + "id": 24, + "legend": { + "avg": false, + "current": true, + "max": true, + "min": false, + "show": true, + "total": false, + "values": true + }, + "lines": true, + "linewidth": 1, + "links": [], + "nullPointMode": "null", + "paceLength": 10, + "percentage": false, + "pointradius": 5, + "points": false, + "renderer": "flot", + "seriesOverrides": [], + "spaceLength": 10, + "stack": false, + "steppedLine": false, + "targets": [ + { + "expr": "sum(jvm_memory_used_bytes{application=\"$application\", instance=\"$instance\", area=\"heap\"})", + "format": "time_series", + "intervalFactor": 2, + "legendFormat": "used", + "metric": "", + "refId": "A", + "step": 2400 + }, + { + "expr": "sum(jvm_memory_committed_bytes{application=\"$application\", instance=\"$instance\", area=\"heap\"})", + "format": "time_series", + "intervalFactor": 2, + "legendFormat": "committed", + "refId": "B", + "step": 2400 + }, + { + "expr": "sum(jvm_memory_max_bytes{application=\"$application\", instance=\"$instance\", area=\"heap\"})", + "format": "time_series", + "intervalFactor": 2, + "legendFormat": "max", + "refId": "C", + "step": 2400 + } + ], + "thresholds": [], + "timeFrom": null, + "timeRegions": [], + "timeShift": null, + "title": "JVM Heap", + "tooltip": { + "msResolution": false, + "shared": true, + "sort": 0, + "value_type": "cumulative" + }, + "type": "graph", + "x-axis": true, + "xaxis": { + "buckets": null, + "mode": "time", + "name": null, + "show": true, + "values": [] + }, + "y-axis": true, + "y_formats": ["mbytes", "short"], + "yaxes": [ + { + "format": "bytes", + "label": null, + "logBase": 1, + "max": null, + "min": 0, + "show": true + }, + { + "format": "short", + "label": null, + "logBase": 1, + "max": null, + "min": null, + "show": true + } + ], + "yaxis": { + "align": false, + "alignLevel": null + } + }, + { + "aliasColors": {}, + "bars": false, + "dashLength": 10, + "dashes": false, + "datasource": "Prometheus", + "editable": true, + "error": false, + "fill": 1, + "grid": { + "leftLogBase": 1, + "leftMax": null, + "leftMin": null, + "rightLogBase": 1, + "rightMax": null, + "rightMin": null + }, + "gridPos": { + "h": 7, + "w": 8, + "x": 8, + "y": 16 + }, + "id": 25, + "legend": { + "avg": false, + "current": true, + "max": true, + "min": false, + "show": true, + "total": false, + "values": true + }, + "lines": true, + "linewidth": 1, + "links": [], + "nullPointMode": "null", + "paceLength": 10, + "percentage": false, + "pointradius": 5, + "points": false, + "renderer": "flot", + "seriesOverrides": [], + "spaceLength": 10, + "stack": false, + "steppedLine": false, + "targets": [ + { + "expr": "sum(jvm_memory_used_bytes{application=\"$application\", instance=\"$instance\", area=\"nonheap\"})", + "format": "time_series", + "interval": "", + "intervalFactor": 2, + "legendFormat": "used", + "metric": "", + "refId": "A", + "step": 2400 + }, + { + "expr": "sum(jvm_memory_committed_bytes{application=\"$application\", instance=\"$instance\", area=\"nonheap\"})", + "format": "time_series", + "intervalFactor": 2, + "legendFormat": "committed", + "refId": "B", + "step": 2400 + }, + { + "expr": "sum(jvm_memory_max_bytes{application=\"$application\", instance=\"$instance\", area=\"nonheap\"})", + "format": "time_series", + "intervalFactor": 2, + "legendFormat": "max", + "refId": "C", + "step": 2400 + } + ], + "thresholds": [], + "timeFrom": null, + "timeRegions": [], + "timeShift": null, + "title": "JVM Non-Heap", + "tooltip": { + "msResolution": false, + "shared": true, + "sort": 0, + "value_type": "cumulative" + }, + "type": "graph", + "x-axis": true, + "xaxis": { + "buckets": null, + "mode": "time", + "name": null, + "show": true, + "values": [] + }, + "y-axis": true, + "y_formats": ["mbytes", "short"], + "yaxes": [ + { + "format": "bytes", + "label": null, + "logBase": 1, + "max": null, + "min": 0, + "show": true + }, + { + "format": "short", + "label": null, + "logBase": 1, + "max": null, + "min": null, + "show": true + } + ], + "yaxis": { + "align": false, + "alignLevel": null + } + }, + { + "aliasColors": {}, + "bars": false, + "dashLength": 10, + "dashes": false, + "datasource": "Prometheus", + "editable": true, + "error": false, + "fill": 1, + "grid": { + "leftLogBase": 1, + "leftMax": null, + "leftMin": null, + "rightLogBase": 1, + "rightMax": null, + "rightMin": null + }, + "gridPos": { + "h": 7, + "w": 8, + "x": 16, + "y": 16 + }, + "id": 26, + "legend": { + "alignAsTable": false, + "avg": false, + "current": true, + "max": true, + "min": false, + "show": true, + "total": false, + "values": true + }, + "lines": true, + "linewidth": 1, + "links": [], + "nullPointMode": "null", + "paceLength": 10, + "percentage": false, + "pointradius": 5, + "points": false, + "renderer": "flot", + "seriesOverrides": [], + "spaceLength": 10, + "stack": false, + "steppedLine": false, + "targets": [ + { + "expr": "sum(jvm_memory_used_bytes{application=\"$application\", instance=\"$instance\"})", + "format": "time_series", + "intervalFactor": 2, + "legendFormat": "used", + "metric": "", + "refId": "A", + "step": 2400 + }, + { + "expr": "sum(jvm_memory_committed_bytes{application=\"$application\", instance=\"$instance\"})", + "format": "time_series", + "intervalFactor": 2, + "legendFormat": "committed", + "refId": "B", + "step": 2400 + }, + { + "expr": "sum(jvm_memory_max_bytes{application=\"$application\", instance=\"$instance\"})", + "format": "time_series", + "intervalFactor": 2, + "legendFormat": "max", + "refId": "C", + "step": 2400 + }, + { + "expr": "process_memory_vss_bytes{application=\"$application\", instance=\"$instance\"}", + "format": "time_series", + "hide": true, + "intervalFactor": 2, + "legendFormat": "vss", + "metric": "", + "refId": "D", + "step": 2400 + }, + { + "expr": "process_memory_rss_bytes{application=\"$application\", instance=\"$instance\"}", + "format": "time_series", + "intervalFactor": 2, + "legendFormat": "rss", + "refId": "E", + "step": 2400 + }, + { + "expr": "process_memory_pss_bytes{application=\"$application\", instance=\"$instance\"}", + "format": "time_series", + "intervalFactor": 2, + "legendFormat": "pss", + "refId": "F", + "step": 2400 + }, + { + "expr": "process_memory_swap_bytes{application=\"$application\", instance=\"$instance\"}", + "format": "time_series", + "intervalFactor": 2, + "legendFormat": "swap", + "refId": "G", + "step": 2400 + }, + { + "expr": "process_memory_swappss_bytes{application=\"$application\", instance=\"$instance\"}", + "format": "time_series", + "intervalFactor": 2, + "legendFormat": "swappss", + "refId": "H", + "step": 2400 + }, + { + "expr": "process_memory_pss_bytes{application=\"$application\", instance=\"$instance\"} + process_memory_swap_bytes{application=\"$application\", instance=\"$instance\"}", + "format": "time_series", + "intervalFactor": 2, + "legendFormat": "phys (pss+swap)", + "refId": "I", + "step": 2400 + } + ], + "thresholds": [], + "timeFrom": null, + "timeRegions": [], + "timeShift": null, + "title": "JVM Total", + "tooltip": { + "msResolution": false, + "shared": true, + "sort": 0, + "value_type": "cumulative" + }, + "type": "graph", + "x-axis": true, + "xaxis": { + "buckets": null, + "mode": "time", + "name": null, + "show": true, + "values": [] + }, + "y-axis": true, + "y_formats": ["mbytes", "short"], + "yaxes": [ + { + "format": "bytes", + "label": "", + "logBase": 1, + "max": null, + "min": 0, + "show": true + }, + { + "format": "short", + "label": null, + "logBase": 1, + "max": null, + "min": null, + "show": true + } + ], + "yaxis": { + "align": false, + "alignLevel": null + } + }, + { + "collapsed": false, + "gridPos": { + "h": 1, + "w": 24, + "x": 0, + "y": 23 + }, + "id": 128, + "panels": [], + "repeat": null, + "title": "JVM Misc", + "type": "row" + }, + { + "aliasColors": {}, + "bars": false, + "dashLength": 10, + "dashes": false, + "datasource": "Prometheus", + "editable": true, + "error": false, + "fill": 1, + "grid": { + "leftLogBase": 1, + "leftMax": null, + "leftMin": null, + "rightLogBase": 1, + "rightMax": null, + "rightMin": null + }, + "gridPos": { + "h": 7, + "w": 6, + "x": 0, + "y": 24 + }, + "id": 106, + "legend": { + "avg": false, + "current": true, + "max": true, + "min": false, + "show": true, + "total": false, + "values": true + }, + "lines": true, + "linewidth": 1, + "links": [], + "nullPointMode": "null", + "paceLength": 10, + "percentage": false, + "pointradius": 5, + "points": false, + "renderer": "flot", + "seriesOverrides": [], + "spaceLength": 10, + "stack": false, + "steppedLine": false, + "targets": [ + { + "expr": "system_cpu_usage{application=\"$application\", instance=\"$instance\"}", + "format": "time_series", + "hide": false, + "intervalFactor": 1, + "legendFormat": "system", + "metric": "", + "refId": "A", + "step": 2400 + }, + { + "expr": "process_cpu_usage{application=\"$application\", instance=\"$instance\"}", + "format": "time_series", + "hide": false, + "intervalFactor": 1, + "legendFormat": "process", + "refId": "B" + }, + { + "expr": "avg_over_time(process_cpu_usage{application=\"$application\", instance=\"$instance\"}[1h])", + "format": "time_series", + "hide": false, + "intervalFactor": 1, + "legendFormat": "process-1h", + "refId": "C" + } + ], + "thresholds": [], + "timeFrom": null, + "timeRegions": [], + "timeShift": null, + "title": "CPU", + "tooltip": { + "msResolution": false, + "shared": true, + "sort": 0, + "value_type": "cumulative" + }, + "type": "graph", + "x-axis": true, + "xaxis": { + "buckets": null, + "mode": "time", + "name": null, + "show": true, + "values": [] + }, + "y-axis": true, + "y_formats": ["short", "short"], + "yaxes": [ + { + "decimals": 1, + "format": "percentunit", + "label": "", + "logBase": 1, + "max": "1", + "min": 0, + "show": true + }, + { + "format": "short", + "label": null, + "logBase": 1, + "max": null, + "min": null, + "show": true + } + ], + "yaxis": { + "align": false, + "alignLevel": null + } + }, + { + "aliasColors": {}, + "bars": false, + "dashLength": 10, + "dashes": false, + "datasource": "Prometheus", + "editable": true, + "error": false, + "fill": 1, + "grid": { + "leftLogBase": 1, + "leftMax": null, + "leftMin": null, + "rightLogBase": 1, + "rightMax": null, + "rightMin": null + }, + "gridPos": { + "h": 7, + "w": 6, + "x": 6, + "y": 24 + }, + "id": 93, + "legend": { + "avg": false, + "current": true, + "max": true, + "min": false, + "show": true, + "total": false, + "values": true + }, + "lines": true, + "linewidth": 1, + "links": [], + "nullPointMode": "null", + "paceLength": 10, + "percentage": false, + "pointradius": 5, + "points": false, + "renderer": "flot", + "seriesOverrides": [], + "spaceLength": 10, + "stack": false, + "steppedLine": false, + "targets": [ + { + "expr": "system_load_average_1m{application=\"$application\", instance=\"$instance\"}", + "format": "time_series", + "intervalFactor": 2, + "legendFormat": "system-1m", + "metric": "", + "refId": "A", + "step": 2400 + }, + { + "expr": "", + "format": "time_series", + "intervalFactor": 2, + "refId": "B" + }, + { + "expr": "system_cpu_count{application=\"$application\", instance=\"$instance\"}", + "format": "time_series", + "intervalFactor": 2, + "legendFormat": "cpu", + "refId": "C" + } + ], + "thresholds": [], + "timeFrom": null, + "timeRegions": [], + "timeShift": null, + "title": "Load", + "tooltip": { + "msResolution": false, + "shared": true, + "sort": 0, + "value_type": "cumulative" + }, + "type": "graph", + "x-axis": true, + "xaxis": { + "buckets": null, + "mode": "time", + "name": null, + "show": true, + "values": [] + }, + "y-axis": true, + "y_formats": ["short", "short"], + "yaxes": [ + { + "decimals": 1, + "format": "short", + "label": "", + "logBase": 1, + "max": null, + "min": 0, + "show": true + }, + { + "format": "short", + "label": null, + "logBase": 1, + "max": null, + "min": null, + "show": true + } + ], + "yaxis": { + "align": false, + "alignLevel": null + } + }, + { + "aliasColors": {}, + "bars": false, + "dashLength": 10, + "dashes": false, + "datasource": "Prometheus", + "editable": true, + "error": false, + "fill": 1, + "grid": { + "leftLogBase": 1, + "leftMax": null, + "leftMin": null, + "rightLogBase": 1, + "rightMax": null, + "rightMin": null + }, + "gridPos": { + "h": 7, + "w": 6, + "x": 12, + "y": 24 + }, + "id": 32, + "legend": { + "avg": false, + "current": true, + "max": true, + "min": false, + "show": true, + "total": false, + "values": true + }, + "lines": true, + "linewidth": 1, + "links": [], + "nullPointMode": "null", + "paceLength": 10, + "percentage": false, + "pointradius": 5, + "points": false, + "renderer": "flot", + "seriesOverrides": [], + "spaceLength": 10, + "stack": false, + "steppedLine": false, + "targets": [ + { + "expr": "jvm_threads_live{application=\"$application\", instance=\"$instance\"} or jvm_threads_live_threads{application=\"$application\", instance=\"$instance\"}", + "format": "time_series", + "intervalFactor": 2, + "legendFormat": "live", + "metric": "", + "refId": "A", + "step": 2400 + }, + { + "expr": "jvm_threads_daemon{application=\"$application\", instance=\"$instance\"} or jvm_threads_daemon_threads{application=\"$application\", instance=\"$instance\"}", + "format": "time_series", + "intervalFactor": 2, + "legendFormat": "daemon", + "metric": "", + "refId": "B", + "step": 2400 + }, + { + "expr": "jvm_threads_peak{application=\"$application\", instance=\"$instance\"} or jvm_threads_peak_threads{application=\"$application\", instance=\"$instance\"}", + "format": "time_series", + "intervalFactor": 2, + "legendFormat": "peak", + "refId": "C", + "step": 2400 + }, + { + "expr": "process_threads{application=\"$application\", instance=\"$instance\"}", + "format": "time_series", + "interval": "", + "intervalFactor": 2, + "legendFormat": "process", + "refId": "D", + "step": 2400 + } + ], + "thresholds": [], + "timeFrom": null, + "timeRegions": [], + "timeShift": null, + "title": "Threads", + "tooltip": { + "msResolution": false, + "shared": true, + "sort": 0, + "value_type": "cumulative" + }, + "type": "graph", + "x-axis": true, + "xaxis": { + "buckets": null, + "mode": "time", + "name": null, + "show": true, + "values": [] + }, + "y-axis": true, + "y_formats": ["short", "short"], + "yaxes": [ + { + "decimals": 0, + "format": "short", + "label": null, + "logBase": 1, + "max": null, + "min": 0, + "show": true + }, + { + "format": "short", + "label": null, + "logBase": 1, + "max": null, + "min": null, + "show": true + } + ], + "yaxis": { + "align": false, + "alignLevel": null + } + }, + { + "aliasColors": { + "blocked": "#bf1b00", + "new": "#fce2de", + "runnable": "#7eb26d", + "terminated": "#511749", + "timed-waiting": "#c15c17", + "waiting": "#eab839" + }, + "bars": false, + "dashLength": 10, + "dashes": false, + "datasource": "Prometheus", + "fill": 1, + "gridPos": { + "h": 7, + "w": 6, + "x": 18, + "y": 24 + }, + "id": 124, + "legend": { + "alignAsTable": false, + "avg": false, + "current": true, + "max": true, + "min": false, + "rightSide": false, + "show": true, + "total": false, + "values": true + }, + "lines": true, + "linewidth": 1, + "links": [], + "nullPointMode": "null", + "paceLength": 10, + "percentage": false, + "pointradius": 5, + "points": false, + "renderer": "flot", + "seriesOverrides": [], + "spaceLength": 10, + "stack": false, + "steppedLine": false, + "targets": [ + { + "expr": "jvm_threads_states_threads{application=\"$application\", instance=\"$instance\"}", + "format": "time_series", + "intervalFactor": 2, + "legendFormat": "{{state}}", + "refId": "A" + } + ], + "thresholds": [], + "timeFrom": null, + "timeRegions": [], + "timeShift": null, + "title": "Thread States", + "tooltip": { + "shared": true, + "sort": 0, + "value_type": "individual" + }, + "type": "graph", + "xaxis": { + "buckets": null, + "mode": "time", + "name": null, + "show": true, + "values": [] + }, + "yaxes": [ + { + "format": "short", + "label": null, + "logBase": 1, + "max": null, + "min": null, + "show": true + }, + { + "format": "short", + "label": null, + "logBase": 1, + "max": null, + "min": null, + "show": true + } + ], + "yaxis": { + "align": false, + "alignLevel": null + } + }, + { + "aliasColors": { + "debug": "#1F78C1", + "error": "#BF1B00", + "info": "#508642", + "trace": "#6ED0E0", + "warn": "#EAB839" + }, + "bars": false, + "dashLength": 10, + "dashes": false, + "datasource": "Prometheus", + "editable": true, + "error": false, + "fill": 1, + "grid": { + "leftLogBase": 1, + "leftMax": null, + "leftMin": null, + "rightLogBase": 1, + "rightMax": null, + "rightMin": null + }, + "gridPos": { + "h": 7, + "w": 18, + "x": 0, + "y": 31 + }, + "height": "", + "id": 91, + "legend": { + "alignAsTable": false, + "avg": false, + "current": true, + "hideEmpty": false, + "hideZero": false, + "max": true, + "min": false, + "rightSide": false, + "show": true, + "total": false, + "values": true + }, + "lines": true, + "linewidth": 1, + "links": [], + "nullPointMode": "null", + "paceLength": 10, + "percentage": true, + "pointradius": 5, + "points": false, + "renderer": "flot", + "seriesOverrides": [ + { + "alias": "error", + "yaxis": 1 + }, + { + "alias": "warn", + "yaxis": 1 + } + ], + "spaceLength": 10, + "stack": false, + "steppedLine": false, + "targets": [ + { + "expr": "increase(logback_events_total{application=\"$application\", instance=\"$instance\"}[1m])", + "format": "time_series", + "interval": "", + "intervalFactor": 2, + "legendFormat": "{{level}}", + "metric": "", + "refId": "A", + "step": 1200 + } + ], + "thresholds": [], + "timeFrom": null, + "timeRegions": [], + "timeShift": null, + "title": "Log Events (1m)", + "tooltip": { + "msResolution": false, + "shared": true, + "sort": 0, + "value_type": "individual" + }, + "type": "graph", + "x-axis": true, + "xaxis": { + "buckets": null, + "mode": "time", + "name": null, + "show": true, + "values": [] + }, + "y-axis": true, + "y_formats": ["short", "short"], + "yaxes": [ + { + "decimals": 0, + "format": "short", + "label": null, + "logBase": 1, + "max": null, + "min": "0", + "show": true + }, + { + "format": "short", + "label": null, + "logBase": 1, + "max": null, + "min": null, + "show": true + } + ], + "yaxis": { + "align": false, + "alignLevel": null + } + }, + { + "aliasColors": {}, + "bars": false, + "dashLength": 10, + "dashes": false, + "datasource": "Prometheus", + "editable": true, + "error": false, + "fill": 1, + "grid": { + "leftLogBase": 1, + "leftMax": null, + "leftMin": null, + "rightLogBase": 1, + "rightMax": null, + "rightMin": null + }, + "gridPos": { + "h": 7, + "w": 6, + "x": 18, + "y": 31 + }, + "id": 61, + "legend": { + "avg": false, + "current": true, + "max": true, + "min": false, + "show": true, + "total": false, + "values": true + }, + "lines": true, + "linewidth": 1, + "links": [], + "nullPointMode": "null", + "paceLength": 10, + "percentage": false, + "pointradius": 5, + "points": false, + "renderer": "flot", + "seriesOverrides": [], + "spaceLength": 10, + "stack": false, + "steppedLine": false, + "targets": [ + { + "expr": "process_open_fds{application=\"$application\", instance=\"$instance\"}", + "format": "time_series", + "hide": false, + "intervalFactor": 2, + "legendFormat": "open", + "metric": "", + "refId": "A", + "step": 2400 + }, + { + "expr": "process_max_fds{application=\"$application\", instance=\"$instance\"}", + "format": "time_series", + "hide": false, + "intervalFactor": 2, + "legendFormat": "max", + "metric": "", + "refId": "B", + "step": 2400 + }, + { + "expr": "process_files_open{application=\"$application\", instance=\"$instance\"} or process_files_open_files{application=\"$application\", instance=\"$instance\"}", + "format": "time_series", + "intervalFactor": 2, + "legendFormat": "open", + "refId": "C" + }, + { + "expr": "process_files_max{application=\"$application\", instance=\"$instance\"} or process_files_max_files{application=\"$application\", instance=\"$instance\"}", + "format": "time_series", + "intervalFactor": 2, + "legendFormat": "max", + "refId": "D" + } + ], + "thresholds": [], + "timeFrom": null, + "timeRegions": [], + "timeShift": null, + "title": "File Descriptors", + "tooltip": { + "msResolution": false, + "shared": true, + "sort": 0, + "value_type": "cumulative" + }, + "type": "graph", + "x-axis": true, + "xaxis": { + "buckets": null, + "mode": "time", + "name": null, + "show": true, + "values": [] + }, + "y-axis": true, + "y_formats": ["short", "short"], + "yaxes": [ + { + "decimals": 0, + "format": "short", + "label": null, + "logBase": 10, + "max": null, + "min": 0, + "show": true + }, + { + "format": "short", + "label": null, + "logBase": 1, + "max": null, + "min": null, + "show": true + } + ], + "yaxis": { + "align": false, + "alignLevel": null + } + }, + { + "collapsed": false, + "gridPos": { + "h": 1, + "w": 24, + "x": 0, + "y": 38 + }, + "id": 129, + "panels": [], + "repeat": "persistence_counts", + "title": "JVM Memory Pools (Heap)", + "type": "row" + }, + { + "aliasColors": {}, + "bars": false, + "dashLength": 10, + "dashes": false, + "datasource": "Prometheus", + "editable": true, + "error": false, + "fill": 1, + "grid": { + "leftLogBase": 1, + "leftMax": null, + "leftMin": null, + "rightLogBase": 1, + "rightMax": null, + "rightMin": null + }, + "gridPos": { + "h": 7, + "w": 8, + "x": 0, + "y": 39 + }, + "id": 3, + "legend": { + "alignAsTable": false, + "avg": false, + "current": true, + "max": true, + "min": false, + "rightSide": false, + "show": true, + "total": false, + "values": true + }, + "lines": true, + "linewidth": 1, + "links": [], + "maxPerRow": 3, + "nullPointMode": "null", + "paceLength": 10, + "percentage": false, + "pointradius": 5, + "points": false, + "renderer": "flot", + "repeat": "jvm_memory_pool_heap", + "scopedVars": { + "jvm_memory_pool_heap": { + "selected": false, + "text": "PS Eden Space", + "value": "PS Eden Space" + } + }, + "seriesOverrides": [], + "spaceLength": 10, + "stack": false, + "steppedLine": false, + "targets": [ + { + "expr": "jvm_memory_used_bytes{application=\"$application\", instance=\"$instance\", id=\"$jvm_memory_pool_heap\"}", + "format": "time_series", + "hide": false, + "interval": "", + "intervalFactor": 2, + "legendFormat": "used", + "metric": "", + "refId": "A", + "step": 1800 + }, + { + "expr": "jvm_memory_committed_bytes{application=\"$application\", instance=\"$instance\", id=\"$jvm_memory_pool_heap\"}", + "format": "time_series", + "hide": false, + "interval": "", + "intervalFactor": 2, + "legendFormat": "commited", + "metric": "", + "refId": "B", + "step": 1800 + }, + { + "expr": "jvm_memory_max_bytes{application=\"$application\", instance=\"$instance\", id=\"$jvm_memory_pool_heap\"}", + "format": "time_series", + "hide": false, + "interval": "", + "intervalFactor": 2, + "legendFormat": "max", + "metric": "", + "refId": "C", + "step": 1800 + } + ], + "thresholds": [], + "timeFrom": null, + "timeRegions": [], + "timeShift": null, + "title": "$jvm_memory_pool_heap", + "tooltip": { + "msResolution": false, + "shared": true, + "sort": 0, + "value_type": "cumulative" + }, + "type": "graph", + "x-axis": true, + "xaxis": { + "buckets": null, + "mode": "time", + "name": null, + "show": true, + "values": [] + }, + "y-axis": true, + "y_formats": ["mbytes", "short"], + "yaxes": [ + { + "format": "bytes", + "label": null, + "logBase": 1, + "max": null, + "min": 0, + "show": true + }, + { + "format": "short", + "label": null, + "logBase": 1, + "max": null, + "min": null, + "show": true + } + ], + "yaxis": { + "align": false, + "alignLevel": null + } + }, + { + "aliasColors": {}, + "bars": false, + "dashLength": 10, + "dashes": false, + "datasource": "Prometheus", + "editable": true, + "error": false, + "fill": 1, + "grid": { + "leftLogBase": 1, + "leftMax": null, + "leftMin": null, + "rightLogBase": 1, + "rightMax": null, + "rightMin": null + }, + "gridPos": { + "h": 7, + "w": 8, + "x": 8, + "y": 39 + }, + "id": 134, + "legend": { + "alignAsTable": false, + "avg": false, + "current": true, + "max": true, + "min": false, + "rightSide": false, + "show": true, + "total": false, + "values": true + }, + "lines": true, + "linewidth": 1, + "links": [], + "maxPerRow": 3, + "nullPointMode": "null", + "paceLength": 10, + "percentage": false, + "pointradius": 5, + "points": false, + "renderer": "flot", + "repeat": null, + "repeatIteration": 1553765841423, + "repeatPanelId": 3, + "scopedVars": { + "jvm_memory_pool_heap": { + "selected": false, + "text": "PS Old Gen", + "value": "PS Old Gen" + } + }, + "seriesOverrides": [], + "spaceLength": 10, + "stack": false, + "steppedLine": false, + "targets": [ + { + "expr": "jvm_memory_used_bytes{application=\"$application\", instance=\"$instance\", id=\"$jvm_memory_pool_heap\"}", + "format": "time_series", + "hide": false, + "interval": "", + "intervalFactor": 2, + "legendFormat": "used", + "metric": "", + "refId": "A", + "step": 1800 + }, + { + "expr": "jvm_memory_committed_bytes{application=\"$application\", instance=\"$instance\", id=\"$jvm_memory_pool_heap\"}", + "format": "time_series", + "hide": false, + "interval": "", + "intervalFactor": 2, + "legendFormat": "commited", + "metric": "", + "refId": "B", + "step": 1800 + }, + { + "expr": "jvm_memory_max_bytes{application=\"$application\", instance=\"$instance\", id=\"$jvm_memory_pool_heap\"}", + "format": "time_series", + "hide": false, + "interval": "", + "intervalFactor": 2, + "legendFormat": "max", + "metric": "", + "refId": "C", + "step": 1800 + } + ], + "thresholds": [], + "timeFrom": null, + "timeRegions": [], + "timeShift": null, + "title": "$jvm_memory_pool_heap", + "tooltip": { + "msResolution": false, + "shared": true, + "sort": 0, + "value_type": "cumulative" + }, + "type": "graph", + "x-axis": true, + "xaxis": { + "buckets": null, + "mode": "time", + "name": null, + "show": true, + "values": [] + }, + "y-axis": true, + "y_formats": ["mbytes", "short"], + "yaxes": [ + { + "format": "bytes", + "label": null, + "logBase": 1, + "max": null, + "min": 0, + "show": true + }, + { + "format": "short", + "label": null, + "logBase": 1, + "max": null, + "min": null, + "show": true + } + ], + "yaxis": { + "align": false, + "alignLevel": null + } + }, + { + "aliasColors": {}, + "bars": false, + "dashLength": 10, + "dashes": false, + "datasource": "Prometheus", + "editable": true, + "error": false, + "fill": 1, + "grid": { + "leftLogBase": 1, + "leftMax": null, + "leftMin": null, + "rightLogBase": 1, + "rightMax": null, + "rightMin": null + }, + "gridPos": { + "h": 7, + "w": 8, + "x": 16, + "y": 39 + }, + "id": 135, + "legend": { + "alignAsTable": false, + "avg": false, + "current": true, + "max": true, + "min": false, + "rightSide": false, + "show": true, + "total": false, + "values": true + }, + "lines": true, + "linewidth": 1, + "links": [], + "maxPerRow": 3, + "nullPointMode": "null", + "paceLength": 10, + "percentage": false, + "pointradius": 5, + "points": false, + "renderer": "flot", + "repeat": null, + "repeatIteration": 1553765841423, + "repeatPanelId": 3, + "scopedVars": { + "jvm_memory_pool_heap": { + "selected": false, + "text": "PS Survivor Space", + "value": "PS Survivor Space" + } + }, + "seriesOverrides": [], + "spaceLength": 10, + "stack": false, + "steppedLine": false, + "targets": [ + { + "expr": "jvm_memory_used_bytes{application=\"$application\", instance=\"$instance\", id=\"$jvm_memory_pool_heap\"}", + "format": "time_series", + "hide": false, + "interval": "", + "intervalFactor": 2, + "legendFormat": "used", + "metric": "", + "refId": "A", + "step": 1800 + }, + { + "expr": "jvm_memory_committed_bytes{application=\"$application\", instance=\"$instance\", id=\"$jvm_memory_pool_heap\"}", + "format": "time_series", + "hide": false, + "interval": "", + "intervalFactor": 2, + "legendFormat": "commited", + "metric": "", + "refId": "B", + "step": 1800 + }, + { + "expr": "jvm_memory_max_bytes{application=\"$application\", instance=\"$instance\", id=\"$jvm_memory_pool_heap\"}", + "format": "time_series", + "hide": false, + "interval": "", + "intervalFactor": 2, + "legendFormat": "max", + "metric": "", + "refId": "C", + "step": 1800 + } + ], + "thresholds": [], + "timeFrom": null, + "timeRegions": [], + "timeShift": null, + "title": "$jvm_memory_pool_heap", + "tooltip": { + "msResolution": false, + "shared": true, + "sort": 0, + "value_type": "cumulative" + }, + "type": "graph", + "x-axis": true, + "xaxis": { + "buckets": null, + "mode": "time", + "name": null, + "show": true, + "values": [] + }, + "y-axis": true, + "y_formats": ["mbytes", "short"], + "yaxes": [ + { + "format": "bytes", + "label": null, + "logBase": 1, + "max": null, + "min": 0, + "show": true + }, + { + "format": "short", + "label": null, + "logBase": 1, + "max": null, + "min": null, + "show": true + } + ], + "yaxis": { + "align": false, + "alignLevel": null + } + }, + { + "collapsed": false, + "gridPos": { + "h": 1, + "w": 24, + "x": 0, + "y": 46 + }, + "id": 130, + "panels": [], + "repeat": null, + "title": "JVM Memory Pools (Non-Heap)", + "type": "row" + }, + { + "aliasColors": {}, + "bars": false, + "dashLength": 10, + "dashes": false, + "datasource": "Prometheus", + "editable": true, + "error": false, + "fill": 1, + "grid": { + "leftLogBase": 1, + "leftMax": null, + "leftMin": null, + "rightLogBase": 1, + "rightMax": null, + "rightMin": null + }, + "gridPos": { + "h": 7, + "w": 8, + "x": 0, + "y": 47 + }, + "id": 78, + "legend": { + "alignAsTable": false, + "avg": false, + "current": true, + "max": true, + "min": false, + "rightSide": false, + "show": true, + "total": false, + "values": true + }, + "lines": true, + "linewidth": 1, + "links": [], + "maxPerRow": 3, + "nullPointMode": "null", + "paceLength": 10, + "percentage": false, + "pointradius": 5, + "points": false, + "renderer": "flot", + "repeat": "jvm_memory_pool_nonheap", + "scopedVars": { + "jvm_memory_pool_nonheap": { + "selected": false, + "text": "Metaspace", + "value": "Metaspace" + } + }, + "seriesOverrides": [], + "spaceLength": 10, + "stack": false, + "steppedLine": false, + "targets": [ + { + "expr": "jvm_memory_used_bytes{application=\"$application\", instance=\"$instance\", id=\"$jvm_memory_pool_nonheap\"}", + "format": "time_series", + "hide": false, + "interval": "", + "intervalFactor": 2, + "legendFormat": "used", + "metric": "", + "refId": "A", + "step": 1800 + }, + { + "expr": "jvm_memory_committed_bytes{application=\"$application\", instance=\"$instance\", id=\"$jvm_memory_pool_nonheap\"}", + "format": "time_series", + "hide": false, + "interval": "", + "intervalFactor": 2, + "legendFormat": "commited", + "metric": "", + "refId": "B", + "step": 1800 + }, + { + "expr": "jvm_memory_max_bytes{application=\"$application\", instance=\"$instance\", id=\"$jvm_memory_pool_nonheap\"}", + "format": "time_series", + "hide": false, + "interval": "", + "intervalFactor": 2, + "legendFormat": "max", + "metric": "", + "refId": "C", + "step": 1800 + } + ], + "thresholds": [], + "timeFrom": null, + "timeRegions": [], + "timeShift": null, + "title": "$jvm_memory_pool_nonheap", + "tooltip": { + "msResolution": false, + "shared": true, + "sort": 0, + "value_type": "cumulative" + }, + "type": "graph", + "x-axis": true, + "xaxis": { + "buckets": null, + "mode": "time", + "name": null, + "show": true, + "values": [] + }, + "y-axis": true, + "y_formats": ["mbytes", "short"], + "yaxes": [ + { + "format": "bytes", + "label": null, + "logBase": 1, + "max": null, + "min": 0, + "show": true + }, + { + "format": "short", + "label": null, + "logBase": 1, + "max": null, + "min": null, + "show": true + } + ], + "yaxis": { + "align": false, + "alignLevel": null + } + }, + { + "aliasColors": {}, + "bars": false, + "dashLength": 10, + "dashes": false, + "datasource": "Prometheus", + "editable": true, + "error": false, + "fill": 1, + "grid": { + "leftLogBase": 1, + "leftMax": null, + "leftMin": null, + "rightLogBase": 1, + "rightMax": null, + "rightMin": null + }, + "gridPos": { + "h": 7, + "w": 8, + "x": 8, + "y": 47 + }, + "id": 136, + "legend": { + "alignAsTable": false, + "avg": false, + "current": true, + "max": true, + "min": false, + "rightSide": false, + "show": true, + "total": false, + "values": true + }, + "lines": true, + "linewidth": 1, + "links": [], + "maxPerRow": 3, + "nullPointMode": "null", + "paceLength": 10, + "percentage": false, + "pointradius": 5, + "points": false, + "renderer": "flot", + "repeat": null, + "repeatIteration": 1553765841423, + "repeatPanelId": 78, + "scopedVars": { + "jvm_memory_pool_nonheap": { + "selected": false, + "text": "Compressed Class Space", + "value": "Compressed Class Space" + } + }, + "seriesOverrides": [], + "spaceLength": 10, + "stack": false, + "steppedLine": false, + "targets": [ + { + "expr": "jvm_memory_used_bytes{application=\"$application\", instance=\"$instance\", id=\"$jvm_memory_pool_nonheap\"}", + "format": "time_series", + "hide": false, + "interval": "", + "intervalFactor": 2, + "legendFormat": "used", + "metric": "", + "refId": "A", + "step": 1800 + }, + { + "expr": "jvm_memory_committed_bytes{application=\"$application\", instance=\"$instance\", id=\"$jvm_memory_pool_nonheap\"}", + "format": "time_series", + "hide": false, + "interval": "", + "intervalFactor": 2, + "legendFormat": "commited", + "metric": "", + "refId": "B", + "step": 1800 + }, + { + "expr": "jvm_memory_max_bytes{application=\"$application\", instance=\"$instance\", id=\"$jvm_memory_pool_nonheap\"}", + "format": "time_series", + "hide": false, + "interval": "", + "intervalFactor": 2, + "legendFormat": "max", + "metric": "", + "refId": "C", + "step": 1800 + } + ], + "thresholds": [], + "timeFrom": null, + "timeRegions": [], + "timeShift": null, + "title": "$jvm_memory_pool_nonheap", + "tooltip": { + "msResolution": false, + "shared": true, + "sort": 0, + "value_type": "cumulative" + }, + "type": "graph", + "x-axis": true, + "xaxis": { + "buckets": null, + "mode": "time", + "name": null, + "show": true, + "values": [] + }, + "y-axis": true, + "y_formats": ["mbytes", "short"], + "yaxes": [ + { + "format": "bytes", + "label": null, + "logBase": 1, + "max": null, + "min": 0, + "show": true + }, + { + "format": "short", + "label": null, + "logBase": 1, + "max": null, + "min": null, + "show": true + } + ], + "yaxis": { + "align": false, + "alignLevel": null + } + }, + { + "aliasColors": {}, + "bars": false, + "dashLength": 10, + "dashes": false, + "datasource": "Prometheus", + "editable": true, + "error": false, + "fill": 1, + "grid": { + "leftLogBase": 1, + "leftMax": null, + "leftMin": null, + "rightLogBase": 1, + "rightMax": null, + "rightMin": null + }, + "gridPos": { + "h": 7, + "w": 8, + "x": 16, + "y": 47 + }, + "id": 137, + "legend": { + "alignAsTable": false, + "avg": false, + "current": true, + "max": true, + "min": false, + "rightSide": false, + "show": true, + "total": false, + "values": true + }, + "lines": true, + "linewidth": 1, + "links": [], + "maxPerRow": 3, + "nullPointMode": "null", + "paceLength": 10, + "percentage": false, + "pointradius": 5, + "points": false, + "renderer": "flot", + "repeat": null, + "repeatIteration": 1553765841423, + "repeatPanelId": 78, + "scopedVars": { + "jvm_memory_pool_nonheap": { + "selected": false, + "text": "Code Cache", + "value": "Code Cache" + } + }, + "seriesOverrides": [], + "spaceLength": 10, + "stack": false, + "steppedLine": false, + "targets": [ + { + "expr": "jvm_memory_used_bytes{application=\"$application\", instance=\"$instance\", id=\"$jvm_memory_pool_nonheap\"}", + "format": "time_series", + "hide": false, + "interval": "", + "intervalFactor": 2, + "legendFormat": "used", + "metric": "", + "refId": "A", + "step": 1800 + }, + { + "expr": "jvm_memory_committed_bytes{application=\"$application\", instance=\"$instance\", id=\"$jvm_memory_pool_nonheap\"}", + "format": "time_series", + "hide": false, + "interval": "", + "intervalFactor": 2, + "legendFormat": "commited", + "metric": "", + "refId": "B", + "step": 1800 + }, + { + "expr": "jvm_memory_max_bytes{application=\"$application\", instance=\"$instance\", id=\"$jvm_memory_pool_nonheap\"}", + "format": "time_series", + "hide": false, + "interval": "", + "intervalFactor": 2, + "legendFormat": "max", + "metric": "", + "refId": "C", + "step": 1800 + } + ], + "thresholds": [], + "timeFrom": null, + "timeRegions": [], + "timeShift": null, + "title": "$jvm_memory_pool_nonheap", + "tooltip": { + "msResolution": false, + "shared": true, + "sort": 0, + "value_type": "cumulative" + }, + "type": "graph", + "x-axis": true, + "xaxis": { + "buckets": null, + "mode": "time", + "name": null, + "show": true, + "values": [] + }, + "y-axis": true, + "y_formats": ["mbytes", "short"], + "yaxes": [ + { + "format": "bytes", + "label": null, + "logBase": 1, + "max": null, + "min": 0, + "show": true + }, + { + "format": "short", + "label": null, + "logBase": 1, + "max": null, + "min": null, + "show": true + } + ], + "yaxis": { + "align": false, + "alignLevel": null + } + }, + { + "collapsed": false, + "gridPos": { + "h": 1, + "w": 24, + "x": 0, + "y": 54 + }, + "id": 131, + "panels": [], + "repeat": null, + "title": "Garbage Collection", + "type": "row" + }, + { + "aliasColors": {}, + "bars": false, + "dashLength": 10, + "dashes": false, + "datasource": "Prometheus", + "fill": 1, + "gridPos": { + "h": 7, + "w": 8, + "x": 0, + "y": 55 + }, + "id": 98, + "legend": { + "avg": false, + "current": false, + "max": false, + "min": false, + "show": true, + "total": false, + "values": false + }, + "lines": true, + "linewidth": 1, + "links": [], + "nullPointMode": "null", + "paceLength": 10, + "percentage": false, + "pointradius": 5, + "points": false, + "renderer": "flot", + "seriesOverrides": [], + "spaceLength": 10, + "stack": false, + "steppedLine": false, + "targets": [ + { + "expr": "rate(jvm_gc_pause_seconds_count{application=\"$application\", instance=\"$instance\"}[1m])", + "format": "time_series", + "hide": false, + "intervalFactor": 2, + "legendFormat": "{{action}} ({{cause}})", + "refId": "A" + } + ], + "thresholds": [], + "timeFrom": null, + "timeRegions": [], + "timeShift": null, + "title": "Collections", + "tooltip": { + "shared": true, + "sort": 0, + "value_type": "individual" + }, + "type": "graph", + "xaxis": { + "buckets": null, + "mode": "time", + "name": null, + "show": true, + "values": [] + }, + "yaxes": [ + { + "format": "ops", + "label": null, + "logBase": 1, + "max": null, + "min": "0", + "show": true + }, + { + "format": "short", + "label": "", + "logBase": 1, + "max": null, + "min": null, + "show": true + } + ], + "yaxis": { + "align": false, + "alignLevel": null + } + }, + { + "aliasColors": {}, + "bars": false, + "dashLength": 10, + "dashes": false, + "datasource": "Prometheus", + "fill": 1, + "gridPos": { + "h": 7, + "w": 8, + "x": 8, + "y": 55 + }, + "id": 101, + "legend": { + "avg": false, + "current": false, + "max": false, + "min": false, + "show": true, + "total": false, + "values": false + }, + "lines": true, + "linewidth": 1, + "links": [], + "nullPointMode": "null", + "paceLength": 10, + "percentage": false, + "pointradius": 5, + "points": false, + "renderer": "flot", + "seriesOverrides": [], + "spaceLength": 10, + "stack": false, + "steppedLine": false, + "targets": [ + { + "expr": "rate(jvm_gc_pause_seconds_sum{application=\"$application\", instance=\"$instance\"}[1m])/rate(jvm_gc_pause_seconds_count{application=\"$application\", instance=\"$instance\"}[1m])", + "format": "time_series", + "hide": false, + "instant": false, + "intervalFactor": 1, + "legendFormat": "avg {{action}} ({{cause}})", + "refId": "A" + }, + { + "expr": "jvm_gc_pause_seconds_max{application=\"$application\", instance=\"$instance\"}", + "format": "time_series", + "hide": false, + "instant": false, + "intervalFactor": 1, + "legendFormat": "max {{action}} ({{cause}})", + "refId": "B" + } + ], + "thresholds": [], + "timeFrom": null, + "timeRegions": [], + "timeShift": null, + "title": "Pause Durations", + "tooltip": { + "shared": true, + "sort": 0, + "value_type": "individual" + }, + "type": "graph", + "xaxis": { + "buckets": null, + "mode": "time", + "name": null, + "show": true, + "values": [] + }, + "yaxes": [ + { + "format": "s", + "label": null, + "logBase": 1, + "max": null, + "min": "0", + "show": true + }, + { + "format": "short", + "label": "", + "logBase": 1, + "max": null, + "min": null, + "show": true + } + ], + "yaxis": { + "align": false, + "alignLevel": null + } + }, + { + "aliasColors": {}, + "bars": false, + "dashLength": 10, + "dashes": false, + "datasource": "Prometheus", + "fill": 1, + "gridPos": { + "h": 7, + "w": 8, + "x": 16, + "y": 55 + }, + "id": 99, + "legend": { + "avg": false, + "current": false, + "max": false, + "min": false, + "show": true, + "total": false, + "values": false + }, + "lines": true, + "linewidth": 1, + "links": [], + "nullPointMode": "null", + "paceLength": 10, + "percentage": false, + "pointradius": 5, + "points": false, + "renderer": "flot", + "seriesOverrides": [], + "spaceLength": 10, + "stack": false, + "steppedLine": false, + "targets": [ + { + "expr": "rate(jvm_gc_memory_allocated_bytes_total{application=\"$application\", instance=\"$instance\"}[1m])", + "format": "time_series", + "interval": "", + "intervalFactor": 1, + "legendFormat": "allocated", + "refId": "A" + }, + { + "expr": "rate(jvm_gc_memory_promoted_bytes_total{application=\"$application\", instance=\"$instance\"}[1m])", + "format": "time_series", + "interval": "", + "intervalFactor": 1, + "legendFormat": "promoted", + "refId": "B" + } + ], + "thresholds": [], + "timeFrom": null, + "timeRegions": [], + "timeShift": null, + "title": "Allocated/Promoted", + "tooltip": { + "shared": true, + "sort": 0, + "value_type": "individual" + }, + "type": "graph", + "xaxis": { + "buckets": null, + "mode": "time", + "name": null, + "show": true, + "values": [] + }, + "yaxes": [ + { + "format": "bytes", + "label": null, + "logBase": 1, + "max": null, + "min": "0", + "show": true + }, + { + "format": "short", + "label": null, + "logBase": 1, + "max": null, + "min": null, + "show": true + } + ], + "yaxis": { + "align": false, + "alignLevel": null + } + }, + { + "collapsed": false, + "gridPos": { + "h": 1, + "w": 24, + "x": 0, + "y": 62 + }, + "id": 132, + "panels": [], + "repeat": null, + "title": "Classloading", + "type": "row" + }, + { + "aliasColors": {}, + "bars": false, + "dashLength": 10, + "dashes": false, + "datasource": "Prometheus", + "editable": true, + "error": false, + "fill": 1, + "grid": { + "leftLogBase": 1, + "leftMax": null, + "leftMin": null, + "rightLogBase": 1, + "rightMax": null, + "rightMin": null + }, + "gridPos": { + "h": 7, + "w": 12, + "x": 0, + "y": 63 + }, + "id": 37, + "legend": { + "avg": false, + "current": false, + "max": false, + "min": false, + "show": true, + "total": false, + "values": false + }, + "lines": true, + "linewidth": 1, + "links": [], + "nullPointMode": "null", + "paceLength": 10, + "percentage": false, + "pointradius": 5, + "points": false, + "renderer": "flot", + "seriesOverrides": [], + "spaceLength": 10, + "stack": false, + "steppedLine": false, + "targets": [ + { + "expr": "jvm_classes_loaded{application=\"$application\", instance=\"$instance\"} or jvm_classes_loaded_classes{application=\"$application\", instance=\"$instance\"}", + "format": "time_series", + "intervalFactor": 2, + "legendFormat": "loaded", + "metric": "", + "refId": "A", + "step": 1200 + } + ], + "thresholds": [], + "timeFrom": null, + "timeRegions": [], + "timeShift": null, + "title": "Classes loaded", + "tooltip": { + "msResolution": false, + "shared": true, + "sort": 0, + "value_type": "cumulative" + }, + "type": "graph", + "x-axis": true, + "xaxis": { + "buckets": null, + "mode": "time", + "name": null, + "show": true, + "values": [] + }, + "y-axis": true, + "y_formats": ["short", "short"], + "yaxes": [ + { + "format": "short", + "label": null, + "logBase": 1, + "max": null, + "min": 0, + "show": true + }, + { + "format": "short", + "label": null, + "logBase": 1, + "max": null, + "min": null, + "show": true + } + ], + "yaxis": { + "align": false, + "alignLevel": null + } + }, + { + "aliasColors": {}, + "bars": false, + "dashLength": 10, + "dashes": false, + "datasource": "Prometheus", + "editable": true, + "error": false, + "fill": 1, + "grid": { + "leftLogBase": 1, + "leftMax": null, + "leftMin": null, + "rightLogBase": 1, + "rightMax": null, + "rightMin": null + }, + "gridPos": { + "h": 7, + "w": 12, + "x": 12, + "y": 63 + }, + "id": 38, + "legend": { + "avg": false, + "current": false, + "max": false, + "min": false, + "show": true, + "total": false, + "values": false + }, + "lines": true, + "linewidth": 1, + "links": [], + "nullPointMode": "null", + "paceLength": 10, + "percentage": false, + "pointradius": 5, + "points": false, + "renderer": "flot", + "seriesOverrides": [], + "spaceLength": 10, + "stack": false, + "steppedLine": false, + "targets": [ + { + "expr": "delta(jvm_classes_loaded{application=\"$application\",instance=\"$instance\"}[5m]) or delta(jvm_classes_loaded_classes{application=\"$application\",instance=\"$instance\"}[5m])", + "format": "time_series", + "hide": false, + "interval": "", + "intervalFactor": 2, + "legendFormat": "delta", + "metric": "", + "refId": "A", + "step": 1200 + } + ], + "thresholds": [], + "timeFrom": null, + "timeRegions": [], + "timeShift": null, + "title": "Class delta (5m)", + "tooltip": { + "msResolution": false, + "shared": true, + "sort": 0, + "value_type": "cumulative" + }, + "type": "graph", + "x-axis": true, + "xaxis": { + "buckets": null, + "mode": "time", + "name": null, + "show": true, + "values": [] + }, + "y-axis": true, + "y_formats": ["ops", "short"], + "yaxes": [ + { + "decimals": null, + "format": "short", + "label": "", + "logBase": 1, + "max": null, + "min": null, + "show": true + }, + { + "format": "short", + "label": null, + "logBase": 1, + "max": null, + "min": null, + "show": true + } + ], + "yaxis": { + "align": false, + "alignLevel": null + } + }, + { + "collapsed": false, + "gridPos": { + "h": 1, + "w": 24, + "x": 0, + "y": 70 + }, + "id": 133, + "panels": [], + "repeat": null, + "title": "Buffer Pools", + "type": "row" + }, + { + "aliasColors": {}, + "bars": false, + "dashLength": 10, + "dashes": false, + "datasource": "Prometheus", + "editable": true, + "error": false, + "fill": 1, + "grid": { + "leftLogBase": 1, + "leftMax": null, + "leftMin": null, + "rightLogBase": 1, + "rightMax": null, + "rightMin": null + }, + "gridPos": { + "h": 7, + "w": 6, + "x": 0, + "y": 71 + }, + "id": 33, + "legend": { + "avg": false, + "current": false, + "max": false, + "min": false, + "show": true, + "total": false, + "values": false + }, + "lines": true, + "linewidth": 1, + "links": [], + "nullPointMode": "null", + "paceLength": 10, + "percentage": false, + "pointradius": 5, + "points": false, + "renderer": "flot", + "seriesOverrides": [], + "spaceLength": 10, + "stack": false, + "steppedLine": false, + "targets": [ + { + "expr": "jvm_buffer_memory_used_bytes{application=\"$application\", instance=\"$instance\", id=\"direct\"}", + "format": "time_series", + "intervalFactor": 2, + "legendFormat": "used", + "metric": "", + "refId": "A", + "step": 2400 + }, + { + "expr": "jvm_buffer_total_capacity_bytes{application=\"$application\", instance=\"$instance\", id=\"direct\"}", + "format": "time_series", + "intervalFactor": 2, + "legendFormat": "capacity", + "metric": "", + "refId": "B", + "step": 2400 + } + ], + "thresholds": [], + "timeFrom": null, + "timeRegions": [], + "timeShift": null, + "title": "Direct Buffers", + "tooltip": { + "msResolution": false, + "shared": true, + "sort": 0, + "value_type": "cumulative" + }, + "type": "graph", + "x-axis": true, + "xaxis": { + "buckets": null, + "mode": "time", + "name": null, + "show": true, + "values": [] + }, + "y-axis": true, + "y_formats": ["short", "short"], + "yaxes": [ + { + "format": "bytes", + "label": null, + "logBase": 1, + "max": null, + "min": 0, + "show": true + }, + { + "format": "short", + "label": null, + "logBase": 1, + "max": null, + "min": null, + "show": true + } + ], + "yaxis": { + "align": false, + "alignLevel": null + } + }, + { + "aliasColors": {}, + "bars": false, + "dashLength": 10, + "dashes": false, + "datasource": "Prometheus", + "editable": true, + "error": false, + "fill": 1, + "grid": { + "leftLogBase": 1, + "leftMax": null, + "leftMin": null, + "rightLogBase": 1, + "rightMax": null, + "rightMin": null + }, + "gridPos": { + "h": 7, + "w": 6, + "x": 6, + "y": 71 + }, + "id": 83, + "legend": { + "avg": false, + "current": false, + "max": false, + "min": false, + "show": true, + "total": false, + "values": false + }, + "lines": true, + "linewidth": 1, + "links": [], + "nullPointMode": "null", + "paceLength": 10, + "percentage": false, + "pointradius": 5, + "points": false, + "renderer": "flot", + "seriesOverrides": [], + "spaceLength": 10, + "stack": false, + "steppedLine": false, + "targets": [ + { + "expr": "jvm_buffer_count{application=\"$application\", instance=\"$instance\", id=\"direct\"} or jvm_buffer_count_buffers{application=\"$application\", instance=\"$instance\", id=\"direct\"}", + "format": "time_series", + "intervalFactor": 2, + "legendFormat": "count", + "metric": "", + "refId": "A", + "step": 2400 + } + ], + "thresholds": [], + "timeFrom": null, + "timeRegions": [], + "timeShift": null, + "title": "Direct Buffers", + "tooltip": { + "msResolution": false, + "shared": true, + "sort": 0, + "value_type": "cumulative" + }, + "type": "graph", + "x-axis": true, + "xaxis": { + "buckets": null, + "mode": "time", + "name": null, + "show": true, + "values": [] + }, + "y-axis": true, + "y_formats": ["short", "short"], + "yaxes": [ + { + "decimals": 0, + "format": "short", + "label": null, + "logBase": 1, + "max": null, + "min": 0, + "show": true + }, + { + "format": "short", + "label": null, + "logBase": 1, + "max": null, + "min": null, + "show": true + } + ], + "yaxis": { + "align": false, + "alignLevel": null + } + }, + { + "aliasColors": {}, + "bars": false, + "dashLength": 10, + "dashes": false, + "datasource": "Prometheus", + "editable": true, + "error": false, + "fill": 1, + "grid": { + "leftLogBase": 1, + "leftMax": null, + "leftMin": null, + "rightLogBase": 1, + "rightMax": null, + "rightMin": null + }, + "gridPos": { + "h": 7, + "w": 6, + "x": 12, + "y": 71 + }, + "id": 85, + "legend": { + "avg": false, + "current": false, + "max": false, + "min": false, + "show": true, + "total": false, + "values": false + }, + "lines": true, + "linewidth": 1, + "links": [], + "nullPointMode": "null", + "paceLength": 10, + "percentage": false, + "pointradius": 5, + "points": false, + "renderer": "flot", + "seriesOverrides": [], + "spaceLength": 10, + "stack": false, + "steppedLine": false, + "targets": [ + { + "expr": "jvm_buffer_memory_used_bytes{application=\"$application\", instance=\"$instance\", id=\"mapped\"}", + "format": "time_series", + "intervalFactor": 2, + "legendFormat": "used", + "metric": "", + "refId": "A", + "step": 2400 + }, + { + "expr": "jvm_buffer_total_capacity_bytes{application=\"$application\", instance=\"$instance\", id=\"mapped\"}", + "format": "time_series", + "intervalFactor": 2, + "legendFormat": "capacity", + "metric": "", + "refId": "B", + "step": 2400 + } + ], + "thresholds": [], + "timeFrom": null, + "timeRegions": [], + "timeShift": null, + "title": "Mapped Buffers", + "tooltip": { + "msResolution": false, + "shared": true, + "sort": 0, + "value_type": "cumulative" + }, + "type": "graph", + "x-axis": true, + "xaxis": { + "buckets": null, + "mode": "time", + "name": null, + "show": true, + "values": [] + }, + "y-axis": true, + "y_formats": ["short", "short"], + "yaxes": [ + { + "format": "bytes", + "label": null, + "logBase": 1, + "max": null, + "min": 0, + "show": true + }, + { + "format": "short", + "label": null, + "logBase": 1, + "max": null, + "min": null, + "show": true + } + ], + "yaxis": { + "align": false, + "alignLevel": null + } + }, + { + "aliasColors": {}, + "bars": false, + "dashLength": 10, + "dashes": false, + "datasource": "Prometheus", + "editable": true, + "error": false, + "fill": 1, + "grid": { + "leftLogBase": 1, + "leftMax": null, + "leftMin": null, + "rightLogBase": 1, + "rightMax": null, + "rightMin": null + }, + "gridPos": { + "h": 7, + "w": 6, + "x": 18, + "y": 71 + }, + "id": 84, + "legend": { + "avg": false, + "current": false, + "max": false, + "min": false, + "show": true, + "total": false, + "values": false + }, + "lines": true, + "linewidth": 1, + "links": [], + "nullPointMode": "null", + "paceLength": 10, + "percentage": false, + "pointradius": 5, + "points": false, + "renderer": "flot", + "seriesOverrides": [], + "spaceLength": 10, + "stack": false, + "steppedLine": false, + "targets": [ + { + "expr": "jvm_buffer_count{application=\"$application\", instance=\"$instance\", id=\"mapped\"} or jvm_buffer_count_buffers{application=\"$application\", instance=\"$instance\", id=\"mapped\"}", + "format": "time_series", + "intervalFactor": 2, + "legendFormat": "count", + "metric": "", + "refId": "A", + "step": 2400 + } + ], + "thresholds": [], + "timeFrom": null, + "timeRegions": [], + "timeShift": null, + "title": "Mapped Buffers", + "tooltip": { + "msResolution": false, + "shared": true, + "sort": 0, + "value_type": "cumulative" + }, + "type": "graph", + "x-axis": true, + "xaxis": { + "buckets": null, + "mode": "time", + "name": null, + "show": true, + "values": [] + }, + "y-axis": true, + "y_formats": ["short", "short"], + "yaxes": [ + { + "decimals": 0, + "format": "short", + "label": null, + "logBase": 1, + "max": null, + "min": 0, + "show": true + }, + { + "format": "short", + "label": null, + "logBase": 1, + "max": null, + "min": null, + "show": true + } + ], + "yaxis": { + "align": false, + "alignLevel": null + } + } + ], + "refresh": "10s", + "schemaVersion": 18, + "style": "dark", + "tags": [], + "templating": { + "list": [ + { + "allValue": null, + "current": { + "text": "test", + "value": "test" + }, + "datasource": "Prometheus", + "definition": "", + "hide": 0, + "includeAll": false, + "label": "Application", + "multi": false, + "name": "application", + "options": [], + "query": "label_values(application)", + "refresh": 2, + "regex": "", + "skipUrlSync": false, + "sort": 0, + "tagValuesQuery": "", + "tags": [], + "tagsQuery": "", + "type": "query", + "useTags": false + }, + { + "allFormat": "glob", + "allValue": null, + "current": { + "text": "localhost:8080", + "value": "localhost:8080" + }, + "datasource": "Prometheus", + "definition": "", + "hide": 0, + "includeAll": false, + "label": "Instance", + "multi": false, + "multiFormat": "glob", + "name": "instance", + "options": [], + "query": "label_values(jvm_memory_used_bytes{application=\"$application\"}, instance)", + "refresh": 2, + "regex": "", + "skipUrlSync": false, + "sort": 0, + "tagValuesQuery": "", + "tags": [], + "tagsQuery": "", + "type": "query", + "useTags": false + }, + { + "allFormat": "glob", + "allValue": null, + "current": { + "text": "All", + "value": "$__all" + }, + "datasource": "Prometheus", + "definition": "", + "hide": 0, + "includeAll": true, + "label": "JVM Memory Pools Heap", + "multi": false, + "multiFormat": "glob", + "name": "jvm_memory_pool_heap", + "options": [], + "query": "label_values(jvm_memory_used_bytes{application=\"$application\", instance=\"$instance\", area=\"heap\"},id)", + "refresh": 1, + "regex": "", + "skipUrlSync": false, + "sort": 1, + "tagValuesQuery": "", + "tags": [], + "tagsQuery": "", + "type": "query", + "useTags": false + }, + { + "allFormat": "glob", + "allValue": null, + "current": { + "text": "All", + "value": "$__all" + }, + "datasource": "Prometheus", + "definition": "", + "hide": 0, + "includeAll": true, + "label": "JVM Memory Pools Non-Heap", + "multi": false, + "multiFormat": "glob", + "name": "jvm_memory_pool_nonheap", + "options": [], + "query": "label_values(jvm_memory_used_bytes{application=\"$application\", instance=\"$instance\", area=\"nonheap\"},id)", + "refresh": 1, + "regex": "", + "skipUrlSync": false, + "sort": 2, + "tagValuesQuery": "", + "tags": [], + "tagsQuery": "", + "type": "query", + "useTags": false + } + ] + }, + "time": { + "from": "now-30m", + "to": "now" + }, + "timepicker": { + "now": true, + "refresh_intervals": ["5s", "10s", "30s", "1m", "5m", "15m", "30m", "1h", "2h", "1d"], + "time_options": ["5m", "15m", "1h", "6h", "12h", "24h", "2d", "7d", "30d"] + }, + "timezone": "browser", + "title": "JVM (Micrometer)", + "uid": "Ud1CFe3iz", + "version": 1 +} diff --git a/src/main/docker/grafana/provisioning/dashboards/dashboard.yml b/src/main/docker/grafana/provisioning/dashboards/dashboard.yml new file mode 100644 index 0000000..4817a83 --- /dev/null +++ b/src/main/docker/grafana/provisioning/dashboards/dashboard.yml @@ -0,0 +1,11 @@ +apiVersion: 1 + +providers: + - name: 'Prometheus' + orgId: 1 + folder: '' + type: file + disableDeletion: false + editable: true + options: + path: /etc/grafana/provisioning/dashboards diff --git a/src/main/docker/grafana/provisioning/datasources/datasource.yml b/src/main/docker/grafana/provisioning/datasources/datasource.yml new file mode 100644 index 0000000..57b2bb3 --- /dev/null +++ b/src/main/docker/grafana/provisioning/datasources/datasource.yml @@ -0,0 +1,50 @@ +apiVersion: 1 + +# list of datasources that should be deleted from the database +deleteDatasources: + - name: Prometheus + orgId: 1 + +# list of datasources to insert/update depending +# whats available in the database +datasources: + # name of the datasource. Required + - name: Prometheus + # datasource type. Required + type: prometheus + # access mode. direct or proxy. Required + access: proxy + # org id. will default to orgId 1 if not specified + orgId: 1 + # url + # On MacOS, replace localhost by host.docker.internal + url: http://localhost:9090 + # database password, if used + password: + # database user, if used + user: + # database name, if used + database: + # enable/disable basic auth + basicAuth: false + # basic auth username + basicAuthUser: admin + # basic auth password + basicAuthPassword: admin + # enable/disable with credentials headers + withCredentials: + # mark as default datasource. Max one per org + isDefault: true + # fields that will be converted to json and stored in json_data + jsonData: + graphiteVersion: '1.1' + tlsAuth: false + tlsAuthWithCACert: false + # json object of data that will be encrypted. + secureJsonData: + tlsCACert: '...' + tlsClientCert: '...' + tlsClientKey: '...' + version: 1 + # allow users to edit datasources from the UI. + editable: true diff --git a/src/main/docker/monitoring.yml b/src/main/docker/monitoring.yml new file mode 100644 index 0000000..b179489 --- /dev/null +++ b/src/main/docker/monitoring.yml @@ -0,0 +1,26 @@ +version: '2' +services: + epasmed-prometheus: + image: prom/prometheus:v2.18.1 + volumes: + - ./prometheus/:/etc/prometheus/ + command: + - '--config.file=/etc/prometheus/prometheus.yml' + ports: + - 9090:9090 + # On MacOS, remove next line and replace localhost by host.docker.internal in prometheus/prometheus.yml and + # grafana/provisioning/datasources/datasource.yml + network_mode: 'host' # to test locally running service + epasmed-grafana: + image: grafana/grafana:7.0.1 + volumes: + - ./grafana/provisioning/:/etc/grafana/provisioning/ + environment: + - GF_SECURITY_ADMIN_PASSWORD=admin + - GF_USERS_ALLOW_SIGN_UP=false + - GF_INSTALL_PLUGINS=grafana-piechart-panel + ports: + - 3000:3000 + # On MacOS, remove next line and replace localhost by host.docker.internal in prometheus/prometheus.yml and + # grafana/provisioning/datasources/datasource.yml + network_mode: 'host' # to test locally running service diff --git a/src/main/docker/postgresql.yml b/src/main/docker/postgresql.yml new file mode 100644 index 0000000..a224102 --- /dev/null +++ b/src/main/docker/postgresql.yml @@ -0,0 +1,12 @@ +version: '2' +services: + epasmed-postgresql: + image: postgres:12.3 + # volumes: + # - ~/volumes/jhipster/epasmed/postgresql/:/var/lib/postgresql/data/ + environment: + - POSTGRES_USER=epasmed + - POSTGRES_PASSWORD= + - POSTGRES_HOST_AUTH_METHOD=trust + ports: + - 5432:5432 diff --git a/src/main/docker/prometheus/prometheus.yml b/src/main/docker/prometheus/prometheus.yml new file mode 100644 index 0000000..2da050b --- /dev/null +++ b/src/main/docker/prometheus/prometheus.yml @@ -0,0 +1,31 @@ +# Sample global config for monitoring JHipster applications +global: + scrape_interval: 15s # By default, scrape targets every 15 seconds. + evaluation_interval: 15s # By default, scrape targets every 15 seconds. + # scrape_timeout is set to the global default (10s). + + # Attach these labels to any time series or alerts when communicating with + # external systems (federation, remote storage, Alertmanager). + external_labels: + monitor: 'epasmed' + +# A scrape configuration containing exactly one endpoint to scrape: +# Here it's Prometheus itself. +scrape_configs: + # The job name is added as a label `job=` to any timeseries scraped from this config. + - job_name: 'prometheus' + + # Override the global default and scrape targets from this job every 5 seconds. + scrape_interval: 5s + + # scheme defaults to 'http' enable https in case your application is server via https + #scheme: https + # basic auth is not needed by default. See https://www.jhipster.tech/monitoring/#configuring-metrics-forwarding for details + #basic_auth: + # username: admin + # password: admin + metrics_path: /management/prometheus + static_configs: + - targets: + # On MacOS, replace localhost by host.docker.internal + - localhost:8080 diff --git a/src/main/docker/sonar.yml b/src/main/docker/sonar.yml new file mode 100644 index 0000000..0d1a751 --- /dev/null +++ b/src/main/docker/sonar.yml @@ -0,0 +1,7 @@ +version: '2' +services: + epasmed-sonar: + image: sonarqube:8.3.1-community + ports: + - 9001:9000 + - 9092:9092 diff --git a/src/main/docker/swagger-editor.yml b/src/main/docker/swagger-editor.yml new file mode 100644 index 0000000..1762f6b --- /dev/null +++ b/src/main/docker/swagger-editor.yml @@ -0,0 +1,6 @@ +version: '2' +services: + swagger-editor: + image: swaggerapi/swagger-editor:latest + ports: + - 7742:8080 diff --git a/src/main/java/it/cnr/isti/epasmed/ApplicationWebXml.java b/src/main/java/it/cnr/isti/epasmed/ApplicationWebXml.java new file mode 100644 index 0000000..487359b --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/ApplicationWebXml.java @@ -0,0 +1,20 @@ +package it.cnr.isti.epasmed; + +import io.github.jhipster.config.DefaultProfileUtil; + +import org.springframework.boot.builder.SpringApplicationBuilder; +import org.springframework.boot.web.servlet.support.SpringBootServletInitializer; + +/** + * This is a helper Java class that provides an alternative to creating a {@code web.xml}. + * This will be invoked only when the application is deployed to a Servlet container like Tomcat, JBoss etc. + */ +public class ApplicationWebXml extends SpringBootServletInitializer { + + @Override + protected SpringApplicationBuilder configure(SpringApplicationBuilder application) { + // set a default to use when no profile is configured. + DefaultProfileUtil.addDefaultProfile(application.application()); + return application.sources(EpasmedApp.class); + } +} diff --git a/src/main/java/it/cnr/isti/epasmed/EpasmedApp.java b/src/main/java/it/cnr/isti/epasmed/EpasmedApp.java new file mode 100644 index 0000000..94a940f --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/EpasmedApp.java @@ -0,0 +1,108 @@ +package it.cnr.isti.epasmed; + +import java.net.InetAddress; +import java.net.UnknownHostException; +import java.util.Arrays; +import java.util.Collection; + +import javax.annotation.PostConstruct; + +import org.apache.commons.lang3.StringUtils; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; +import org.springframework.boot.SpringApplication; +import org.springframework.boot.autoconfigure.EnableAutoConfiguration; +import org.springframework.boot.autoconfigure.SpringBootApplication; +import org.springframework.boot.autoconfigure.jdbc.DataSourceAutoConfiguration; +import org.springframework.boot.autoconfigure.jdbc.DataSourceTransactionManagerAutoConfiguration; +import org.springframework.boot.autoconfigure.liquibase.LiquibaseProperties; +import org.springframework.boot.context.properties.EnableConfigurationProperties; +import org.springframework.core.env.Environment; + +import io.github.jhipster.config.DefaultProfileUtil; +import io.github.jhipster.config.JHipsterConstants; +import it.cnr.isti.epasmed.config.ApplicationProperties; + +@SpringBootApplication +//@ComponentScan +//@Configuration +@EnableConfigurationProperties({LiquibaseProperties.class, ApplicationProperties.class}) +//@EnableAutoConfiguration(exclude = {DataSourceAutoConfiguration.class, +// DataSourceTransactionManagerAutoConfiguration.class, +// HibernateJpaAutoConfiguration.class}) +@EnableAutoConfiguration(exclude={DataSourceAutoConfiguration.class, + DataSourceTransactionManagerAutoConfiguration.class}) +public class EpasmedApp { + + private static final Logger log = LoggerFactory.getLogger(EpasmedApp.class); + + private final Environment env; + + public EpasmedApp(Environment env) { + this.env = env; + } + + /** + * Initializes epasmed. + *

+ * Spring profiles can be configured with a program argument --spring.profiles.active=your-active-profile + *

+ * You can find more information on how profiles work with JHipster on https://www.jhipster.tech/profiles/. + */ + @PostConstruct + public void initApplication() { + Collection activeProfiles = Arrays.asList(env.getActiveProfiles()); + if (activeProfiles.contains(JHipsterConstants.SPRING_PROFILE_DEVELOPMENT) && activeProfiles.contains(JHipsterConstants.SPRING_PROFILE_PRODUCTION)) { + log.error("You have misconfigured your application! It should not run " + + "with both the 'dev' and 'prod' profiles at the same time."); + } + if (activeProfiles.contains(JHipsterConstants.SPRING_PROFILE_DEVELOPMENT) && activeProfiles.contains(JHipsterConstants.SPRING_PROFILE_CLOUD)) { + log.error("You have misconfigured your application! It should not " + + "run with both the 'dev' and 'cloud' profiles at the same time."); + } + } + + /** + * Main method, used to run the application. + * + * @param args the command line arguments. + */ + public static void main(String[] args) { + SpringApplication app = new SpringApplication(EpasmedApp.class); + DefaultProfileUtil.addDefaultProfile(app); + Environment env = app.run(args).getEnvironment(); + logApplicationStartup(env); + } + + private static void logApplicationStartup(Environment env) { + String protocol = "http"; + if (env.getProperty("server.ssl.key-store") != null) { + protocol = "https"; + } + String serverPort = env.getProperty("server.port"); + String contextPath = env.getProperty("server.servlet.context-path"); + if (StringUtils.isBlank(contextPath)) { + contextPath = "/"; + } + String hostAddress = "localhost"; + try { + hostAddress = InetAddress.getLocalHost().getHostAddress(); + } catch (UnknownHostException e) { + log.warn("The host name could not be determined, using `localhost` as fallback"); + } + log.info("\n----------------------------------------------------------\n\t" + + "Application '{}' is running! Access URLs:\n\t" + + "Local: \t\t{}://localhost:{}{}\n\t" + + "External: \t{}://{}:{}{}\n\t" + + "Profile(s): \t{}\n----------------------------------------------------------", + env.getProperty("spring.application.name"), + protocol, + serverPort, + contextPath, + protocol, + hostAddress, + serverPort, + contextPath, + env.getActiveProfiles()); + } +} diff --git a/src/main/java/it/cnr/isti/epasmed/aop/logging/LoggingAspect.java b/src/main/java/it/cnr/isti/epasmed/aop/logging/LoggingAspect.java new file mode 100644 index 0000000..00de3ba --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/aop/logging/LoggingAspect.java @@ -0,0 +1,113 @@ +package it.cnr.isti.epasmed.aop.logging; + +import io.github.jhipster.config.JHipsterConstants; + +import org.aspectj.lang.JoinPoint; +import org.aspectj.lang.ProceedingJoinPoint; +import org.aspectj.lang.annotation.AfterThrowing; +import org.aspectj.lang.annotation.Around; +import org.aspectj.lang.annotation.Aspect; +import org.aspectj.lang.annotation.Pointcut; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; +import org.springframework.core.env.Environment; +import org.springframework.core.env.Profiles; + +import java.util.Arrays; + +/** + * Aspect for logging execution of service and repository Spring components. + * + * By default, it only runs with the "dev" profile. + */ +@Aspect +public class LoggingAspect { + + private final Environment env; + + public LoggingAspect(Environment env) { + this.env = env; + } + + /** + * Pointcut that matches all repositories, services and Web REST endpoints. + */ + @Pointcut("within(@org.springframework.stereotype.Repository *)" + + " || within(@org.springframework.stereotype.Service *)" + + " || within(@org.springframework.web.bind.annotation.RestController *)") + public void springBeanPointcut() { + // Method is empty as this is just a Pointcut, the implementations are in the advices. + } + + /** + * Pointcut that matches all Spring beans in the application's main packages. + */ + @Pointcut("within(it.cnr.isti.epasmed.repository..*)"+ + " || within(it.cnr.isti.epasmed.service..*)"+ + " || within(it.cnr.isti.epasmed.web.rest..*)") + public void applicationPackagePointcut() { + // Method is empty as this is just a Pointcut, the implementations are in the advices. + } + + /** + * Retrieves the {@link Logger} associated to the given {@link JoinPoint}. + * + * @param joinPoint join point we want the logger for. + * @return {@link Logger} associated to the given {@link JoinPoint}. + */ + private Logger logger(JoinPoint joinPoint) { + return LoggerFactory.getLogger(joinPoint.getSignature().getDeclaringTypeName()); + } + + /** + * Advice that logs methods throwing exceptions. + * + * @param joinPoint join point for advice. + * @param e exception. + */ + @AfterThrowing(pointcut = "applicationPackagePointcut() && springBeanPointcut()", throwing = "e") + public void logAfterThrowing(JoinPoint joinPoint, Throwable e) { + if (env.acceptsProfiles(Profiles.of(JHipsterConstants.SPRING_PROFILE_DEVELOPMENT))) { + logger(joinPoint) + .error( + "Exception in {}() with cause = \'{}\' and exception = \'{}\'", + joinPoint.getSignature().getName(), + e.getCause() != null ? e.getCause() : "NULL", + e.getMessage(), + e + ); + } else { + logger(joinPoint) + .error( + "Exception in {}() with cause = {}", + joinPoint.getSignature().getName(), + e.getCause() != null ? e.getCause() : "NULL" + ); + } + } + + /** + * Advice that logs when a method is entered and exited. + * + * @param joinPoint join point for advice. + * @return result. + * @throws Throwable throws {@link IllegalArgumentException}. + */ + @Around("applicationPackagePointcut() && springBeanPointcut()") + public Object logAround(ProceedingJoinPoint joinPoint) throws Throwable { + Logger log = logger(joinPoint); + if (log.isDebugEnabled()) { + log.debug("Enter: {}() with argument[s] = {}", joinPoint.getSignature().getName(), Arrays.toString(joinPoint.getArgs())); + } + try { + Object result = joinPoint.proceed(); + if (log.isDebugEnabled()) { + log.debug("Exit: {}() with result = {}", joinPoint.getSignature().getName(), result); + } + return result; + } catch (IllegalArgumentException e) { + log.error("Illegal argument: {} in {}()", Arrays.toString(joinPoint.getArgs()), joinPoint.getSignature().getName()); + throw e; + } + } +} diff --git a/src/main/java/it/cnr/isti/epasmed/config/ApplicationProperties.java b/src/main/java/it/cnr/isti/epasmed/config/ApplicationProperties.java new file mode 100644 index 0000000..8da134d --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/config/ApplicationProperties.java @@ -0,0 +1,942 @@ +package it.cnr.isti.epasmed.config; + +import java.io.File; +import java.nio.charset.Charset; +import java.util.LinkedHashMap; +import java.util.List; +import java.util.Map; +import java.util.UUID; + +import javax.sql.DataSource; + +import org.springframework.beans.factory.BeanClassLoaderAware; +import org.springframework.beans.factory.BeanCreationException; +import org.springframework.beans.factory.InitializingBean; +import org.springframework.boot.context.properties.ConfigurationProperties; +import org.springframework.boot.jdbc.DataSourceBuilder; +import org.springframework.boot.jdbc.DataSourceInitializationMode; +import org.springframework.boot.jdbc.DatabaseDriver; +import org.springframework.boot.jdbc.EmbeddedDatabaseConnection; +import org.springframework.util.Assert; +import org.springframework.util.ClassUtils; +import org.springframework.util.StringUtils; + +/** + * Properties specific to Epasmed. + *

+ * Properties are configured in the {@code application.yml} file. See + * {@link io.github.jhipster.config.JHipsterProperties} for a good example. + */ +@ConfigurationProperties(prefix = "application", ignoreUnknownFields = false) +public class ApplicationProperties { + + private final DatasourceEpasmed datasourceEpasmed = new DatasourceEpasmed(); + private final DatasourceSistemainformativo datasourceSistemainformativo = new DatasourceSistemainformativo(); + private final DatasourceEpasRest datasourceEpasRest = new DatasourceEpasRest(); + + public static class DatasourceEpasmed { + private final DataSourceProperties datasource = new DataSourceProperties(); + private final LiquibaseProperties liquibase = new LiquibaseProperties(); + private final HikariProperties hikari = new HikariProperties(); + + public DataSourceProperties getDatasource() { + return datasource; + } + + public LiquibaseProperties getLiquibase() { + return liquibase; + } + + public HikariProperties getHikari() { + return hikari; + } + + } + + public static class DatasourceSistemainformativo { + private final DataSourceProperties datasource = new DataSourceProperties(); + private final HikariProperties hikari = new HikariProperties(); + + public DataSourceProperties getDatasource() { + return datasource; + } + + public HikariProperties getHikari() { + return hikari; + } + + } + + public static class DatasourceEpasRest { + private String restUrl; + private String restUsername1; + private String restPassword1; + private String restUsername2; + private String restPassword2; + + public String getRestUrl() { + return restUrl; + } + + public void setRestUrl(String restUrl) { + this.restUrl = restUrl; + } + + public String getRestUsername1() { + return restUsername1; + } + + public void setRestUsername1(String restUsername1) { + this.restUsername1 = restUsername1; + } + + public String getRestPassword1() { + return restPassword1; + } + + public void setRestPassword1(String restPassword1) { + this.restPassword1 = restPassword1; + } + + public String getRestUsername2() { + return restUsername2; + } + + public void setRestUsername2(String restUsername2) { + this.restUsername2 = restUsername2; + } + + public String getRestPassword2() { + return restPassword2; + } + + public void setRestPassword2(String restPassword2) { + this.restPassword2 = restPassword2; + } + + @Override + public String toString() { + return "DatasourceEpasRest [restUrl=" + restUrl + ", restUsername1=" + restUsername1 + ", restPassword1=" + + restPassword1 + ", restUsername2=" + restUsername2 + ", restPassword2=" + restPassword2 + "]"; + } + + } + + public static class HikariProperties { + private String jdbc_url; + private String database_username; + private String database_password; + private boolean cachePrepStmts = true; + private int prepStmtCacheSize = 250; + private int prepStmtCacheSqlLimit = 2048; + private String poolName = "Hikari1"; + private boolean autoCommit = false; + private int maximumPoolSize = 10; + + public String getPoolName() { + return poolName; + } + + public void setPoolName(String poolName) { + this.poolName = poolName; + } + + public boolean isAutoCommit() { + return autoCommit; + } + + public void setAutoCommit(boolean autoCommit) { + this.autoCommit = autoCommit; + } + + public int getMaximumPoolSize() { + return maximumPoolSize; + } + + public void setMaximumPoolSize(int maximumPoolSize) { + this.maximumPoolSize = maximumPoolSize; + } + + public String getJdbc_url() { + return jdbc_url; + } + + public String getDatabase_username() { + return database_username; + } + + public void setDatabase_username(String database_username) { + this.database_username = database_username; + } + + public String getDatabase_password() { + return database_password; + } + + public void setDatabase_password(String database_password) { + this.database_password = database_password; + } + + public boolean isCachePrepStmts() { + return cachePrepStmts; + } + + public void setCachePrepStmts(boolean cachePrepStmts) { + this.cachePrepStmts = cachePrepStmts; + } + + public int getPrepStmtCacheSize() { + return prepStmtCacheSize; + } + + public void setPrepStmtCacheSize(int prepStmtCacheSize) { + this.prepStmtCacheSize = prepStmtCacheSize; + } + + public int getPrepStmtCacheSqlLimit() { + return prepStmtCacheSqlLimit; + } + + public void setPrepStmtCacheSqlLimit(int prepStmtCacheSqlLimit) { + this.prepStmtCacheSqlLimit = prepStmtCacheSqlLimit; + } + + public void setJdbc_url(String jdbc_url) { + this.jdbc_url = jdbc_url; + } + + @Override + public String toString() { + return "HikariProperties [jdbc_url=" + jdbc_url + ", database_username=" + database_username + + ", database_password=" + database_password + ", cachePrepStmts=" + cachePrepStmts + + ", prepStmtCacheSize=" + prepStmtCacheSize + ", prepStmtCacheSqlLimit=" + prepStmtCacheSqlLimit + + ", poolName=" + poolName + ", autoCommit=" + autoCommit + ", maximumPoolSize=" + maximumPoolSize + + "]"; + } + + } + + public static class DataSourceProperties implements BeanClassLoaderAware, InitializingBean { + + private ClassLoader classLoader; + + /** + * Name of the datasource. Default to "testdb" when using an embedded database. + */ + private String name; + + /** + * Whether to generate a random datasource name. + */ + private boolean generateUniqueName; + + /** + * Fully qualified name of the connection pool implementation to use. By + * default, it is auto-detected from the classpath. + */ + private Class type; + + /** + * Fully qualified name of the JDBC driver. Auto-detected based on the URL by + * default. + */ + private String driverClassName; + + /** + * JDBC URL of the database. + */ + private String url; + + /** + * Login username of the database. + */ + private String username; + + /** + * Login password of the database. + */ + private String password; + + /** + * JNDI location of the datasource. Class, url, username & password are ignored + * when set. + */ + private String jndiName; + + /** + * Initialize the datasource with available DDL and DML scripts. + */ + private DataSourceInitializationMode initializationMode = DataSourceInitializationMode.EMBEDDED; + + /** + * Platform to use in the DDL or DML scripts (such as schema-${platform}.sql or + * data-${platform}.sql). + */ + private String platform = "all"; + + /** + * Schema (DDL) script resource references. + */ + private List schema; + + /** + * Username of the database to execute DDL scripts (if different). + */ + private String schemaUsername; + + /** + * Password of the database to execute DDL scripts (if different). + */ + private String schemaPassword; + + /** + * Data (DML) script resource references. + */ + private List data; + + /** + * Username of the database to execute DML scripts (if different). + */ + private String dataUsername; + + /** + * Password of the database to execute DML scripts (if different). + */ + private String dataPassword; + + /** + * Whether to stop if an error occurs while initializing the database. + */ + private boolean continueOnError = false; + + /** + * Statement separator in SQL initialization scripts. + */ + private String separator = ";"; + + /** + * SQL scripts encoding. + */ + private Charset sqlScriptEncoding; + + private EmbeddedDatabaseConnection embeddedDatabaseConnection = EmbeddedDatabaseConnection.NONE; + + private Xa xa = new Xa(); + + private String uniqueName; + + @Override + public void setBeanClassLoader(ClassLoader classLoader) { + this.classLoader = classLoader; + } + + @Override + public void afterPropertiesSet() throws Exception { + this.embeddedDatabaseConnection = EmbeddedDatabaseConnection.get(this.classLoader); + } + + /** + * Initialize a {@link DataSourceBuilder} with the state of this instance. + * + * @return a {@link DataSourceBuilder} initialized with the customizations + * defined on this instance + */ + public DataSourceBuilder initializeDataSourceBuilder() { + return DataSourceBuilder.create(getClassLoader()).type(getType()) + .driverClassName(determineDriverClassName()).url(determineUrl()).username(determineUsername()) + .password(determinePassword()); + } + + public String getName() { + return this.name; + } + + public void setName(String name) { + this.name = name; + } + + public boolean isGenerateUniqueName() { + return this.generateUniqueName; + } + + public void setGenerateUniqueName(boolean generateUniqueName) { + this.generateUniqueName = generateUniqueName; + } + + public Class getType() { + return this.type; + } + + public void setType(Class type) { + this.type = type; + } + + /** + * Return the configured driver or {@code null} if none was configured. + * + * @return the configured driver + * @see #determineDriverClassName() + */ + public String getDriverClassName() { + return this.driverClassName; + } + + public void setDriverClassName(String driverClassName) { + this.driverClassName = driverClassName; + } + + /** + * Determine the driver to use based on this configuration and the environment. + * + * @return the driver to use + * @since 1.4.0 + */ + public String determineDriverClassName() { + if (StringUtils.hasText(this.driverClassName)) { + Assert.state(driverClassIsLoadable(), () -> "Cannot load driver class: " + this.driverClassName); + return this.driverClassName; + } + String driverClassName = null; + if (StringUtils.hasText(this.url)) { + driverClassName = DatabaseDriver.fromJdbcUrl(this.url).getDriverClassName(); + } + if (!StringUtils.hasText(driverClassName)) { + driverClassName = this.embeddedDatabaseConnection.getDriverClassName(); + } + if (!StringUtils.hasText(driverClassName)) { + throw new DataSourceBeanCreationException("Failed to determine a suitable driver class", this, + this.embeddedDatabaseConnection); + } + return driverClassName; + } + + private boolean driverClassIsLoadable() { + try { + ClassUtils.forName(this.driverClassName, null); + return true; + } catch (UnsupportedClassVersionError ex) { + // Driver library has been compiled with a later JDK, propagate error + throw ex; + } catch (Throwable ex) { + return false; + } + } + + /** + * Return the configured url or {@code null} if none was configured. + * + * @return the configured url + * @see #determineUrl() + */ + public String getUrl() { + return this.url; + } + + public void setUrl(String url) { + this.url = url; + } + + /** + * Determine the url to use based on this configuration and the environment. + * + * @return the url to use + * @since 1.4.0 + */ + public String determineUrl() { + if (StringUtils.hasText(this.url)) { + return this.url; + } + String databaseName = determineDatabaseName(); + String url = (databaseName != null) ? this.embeddedDatabaseConnection.getUrl(databaseName) : null; + if (!StringUtils.hasText(url)) { + throw new DataSourceBeanCreationException("Failed to determine suitable jdbc url", this, + this.embeddedDatabaseConnection); + } + return url; + } + + /** + * Determine the name to used based on this configuration. + * + * @return the database name to use or {@code null} + * @since 2.0.0 + */ + public String determineDatabaseName() { + if (this.generateUniqueName) { + if (this.uniqueName == null) { + this.uniqueName = UUID.randomUUID().toString(); + } + return this.uniqueName; + } + if (StringUtils.hasLength(this.name)) { + return this.name; + } + if (this.embeddedDatabaseConnection != EmbeddedDatabaseConnection.NONE) { + return "testdb"; + } + return null; + } + + /** + * Return the configured username or {@code null} if none was configured. + * + * @return the configured username + * @see #determineUsername() + */ + public String getUsername() { + return this.username; + } + + public void setUsername(String username) { + this.username = username; + } + + /** + * Determine the username to use based on this configuration and the + * environment. + * + * @return the username to use + * @since 1.4.0 + */ + public String determineUsername() { + if (StringUtils.hasText(this.username)) { + return this.username; + } + if (EmbeddedDatabaseConnection.isEmbedded(determineDriverClassName())) { + return "sa"; + } + return null; + } + + /** + * Return the configured password or {@code null} if none was configured. + * + * @return the configured password + * @see #determinePassword() + */ + public String getPassword() { + return this.password; + } + + public void setPassword(String password) { + this.password = password; + } + + /** + * Determine the password to use based on this configuration and the + * environment. + * + * @return the password to use + * @since 1.4.0 + */ + public String determinePassword() { + if (StringUtils.hasText(this.password)) { + return this.password; + } + if (EmbeddedDatabaseConnection.isEmbedded(determineDriverClassName())) { + return ""; + } + return null; + } + + public String getJndiName() { + return this.jndiName; + } + + /** + * Allows the DataSource to be managed by the container and obtained via JNDI. + * The {@code URL}, {@code driverClassName}, {@code username} and + * {@code password} fields will be ignored when using JNDI lookups. + * + * @param jndiName the JNDI name + */ + public void setJndiName(String jndiName) { + this.jndiName = jndiName; + } + + public DataSourceInitializationMode getInitializationMode() { + return this.initializationMode; + } + + public void setInitializationMode(DataSourceInitializationMode initializationMode) { + this.initializationMode = initializationMode; + } + + public String getPlatform() { + return this.platform; + } + + public void setPlatform(String platform) { + this.platform = platform; + } + + public List getSchema() { + return this.schema; + } + + public void setSchema(List schema) { + this.schema = schema; + } + + public String getSchemaUsername() { + return this.schemaUsername; + } + + public void setSchemaUsername(String schemaUsername) { + this.schemaUsername = schemaUsername; + } + + public String getSchemaPassword() { + return this.schemaPassword; + } + + public void setSchemaPassword(String schemaPassword) { + this.schemaPassword = schemaPassword; + } + + public List getData() { + return this.data; + } + + public void setData(List data) { + this.data = data; + } + + public String getDataUsername() { + return this.dataUsername; + } + + public void setDataUsername(String dataUsername) { + this.dataUsername = dataUsername; + } + + public String getDataPassword() { + return this.dataPassword; + } + + public void setDataPassword(String dataPassword) { + this.dataPassword = dataPassword; + } + + public boolean isContinueOnError() { + return this.continueOnError; + } + + public void setContinueOnError(boolean continueOnError) { + this.continueOnError = continueOnError; + } + + public String getSeparator() { + return this.separator; + } + + public void setSeparator(String separator) { + this.separator = separator; + } + + public Charset getSqlScriptEncoding() { + return this.sqlScriptEncoding; + } + + public void setSqlScriptEncoding(Charset sqlScriptEncoding) { + this.sqlScriptEncoding = sqlScriptEncoding; + } + + public ClassLoader getClassLoader() { + return this.classLoader; + } + + public Xa getXa() { + return this.xa; + } + + public void setXa(Xa xa) { + this.xa = xa; + } + + /** + * XA Specific datasource settings. + */ + public class Xa { + + /** + * XA datasource fully qualified name. + */ + private String dataSourceClassName; + + /** + * Properties to pass to the XA data source. + */ + private Map properties = new LinkedHashMap<>(); + + public String getDataSourceClassName() { + return this.dataSourceClassName; + } + + public void setDataSourceClassName(String dataSourceClassName) { + this.dataSourceClassName = dataSourceClassName; + } + + public Map getProperties() { + return this.properties; + } + + public void setProperties(Map properties) { + this.properties = properties; + } + + } + + public class DataSourceBeanCreationException extends BeanCreationException { + + private static final long serialVersionUID = 1L; + + private final DataSourceProperties properties; + + private final EmbeddedDatabaseConnection connection; + + DataSourceBeanCreationException(String message, DataSourceProperties properties, + EmbeddedDatabaseConnection connection) { + super(message); + this.properties = properties; + this.connection = connection; + } + + DataSourceProperties getProperties() { + return this.properties; + } + + EmbeddedDatabaseConnection getConnection() { + return this.connection; + } + + } + + } + + public static class LiquibaseProperties { + + /** + * Change log configuration path. + */ + private String changeLog = "classpath:/db/changelog/db.changelog-master.yaml"; + + /** + * Comma-separated list of runtime contexts to use. + */ + private String contexts; + + /** + * Default database schema. + */ + private String defaultSchema; + + /** + * Schema to use for Liquibase objects. + */ + private String liquibaseSchema; + + /** + * Tablespace to use for Liquibase objects. + */ + private String liquibaseTablespace; + + /** + * Name of table to use for tracking change history. + */ + private String databaseChangeLogTable = "DATABASECHANGELOG"; + + /** + * Name of table to use for tracking concurrent Liquibase usage. + */ + private String databaseChangeLogLockTable = "DATABASECHANGELOGLOCK"; + + /** + * Whether to first drop the database schema. + */ + private boolean dropFirst; + + /** + * Whether to enable Liquibase support. + */ + private boolean enabled = true; + + /** + * Login user of the database to migrate. + */ + private String user; + + /** + * Login password of the database to migrate. + */ + private String password; + + /** + * JDBC URL of the database to migrate. If not set, the primary configured data + * source is used. + */ + private String url; + + /** + * Comma-separated list of runtime labels to use. + */ + private String labels; + + /** + * Change log parameters. + */ + private Map parameters; + + /** + * File to which rollback SQL is written when an update is performed. + */ + private File rollbackFile; + + /** + * Whether rollback should be tested before update is performed. + */ + private boolean testRollbackOnUpdate; + + public String getChangeLog() { + return this.changeLog; + } + + public void setChangeLog(String changeLog) { + Assert.notNull(changeLog, "ChangeLog must not be null"); + this.changeLog = changeLog; + } + + public String getContexts() { + return this.contexts; + } + + public void setContexts(String contexts) { + this.contexts = contexts; + } + + public String getDefaultSchema() { + return this.defaultSchema; + } + + public void setDefaultSchema(String defaultSchema) { + this.defaultSchema = defaultSchema; + } + + public String getLiquibaseSchema() { + return this.liquibaseSchema; + } + + public void setLiquibaseSchema(String liquibaseSchema) { + this.liquibaseSchema = liquibaseSchema; + } + + public String getLiquibaseTablespace() { + return this.liquibaseTablespace; + } + + public void setLiquibaseTablespace(String liquibaseTablespace) { + this.liquibaseTablespace = liquibaseTablespace; + } + + public String getDatabaseChangeLogTable() { + return this.databaseChangeLogTable; + } + + public void setDatabaseChangeLogTable(String databaseChangeLogTable) { + this.databaseChangeLogTable = databaseChangeLogTable; + } + + public String getDatabaseChangeLogLockTable() { + return this.databaseChangeLogLockTable; + } + + public void setDatabaseChangeLogLockTable(String databaseChangeLogLockTable) { + this.databaseChangeLogLockTable = databaseChangeLogLockTable; + } + + public boolean isDropFirst() { + return this.dropFirst; + } + + public void setDropFirst(boolean dropFirst) { + this.dropFirst = dropFirst; + } + + public boolean isEnabled() { + return this.enabled; + } + + public void setEnabled(boolean enabled) { + this.enabled = enabled; + } + + public String getUser() { + return this.user; + } + + public void setUser(String user) { + this.user = user; + } + + public String getPassword() { + return this.password; + } + + public void setPassword(String password) { + this.password = password; + } + + public String getUrl() { + return this.url; + } + + public void setUrl(String url) { + this.url = url; + } + + public String getLabels() { + return this.labels; + } + + public void setLabels(String labels) { + this.labels = labels; + } + + public Map getParameters() { + return this.parameters; + } + + public void setParameters(Map parameters) { + this.parameters = parameters; + } + + public File getRollbackFile() { + return this.rollbackFile; + } + + public void setRollbackFile(File rollbackFile) { + this.rollbackFile = rollbackFile; + } + + public boolean isTestRollbackOnUpdate() { + return this.testRollbackOnUpdate; + } + + public void setTestRollbackOnUpdate(boolean testRollbackOnUpdate) { + this.testRollbackOnUpdate = testRollbackOnUpdate; + } + + } + + public DatasourceEpasmed getDatasourceEpasmed() { + return datasourceEpasmed; + } + + public DatasourceSistemainformativo getDatasourceSistemainformativo() { + return datasourceSistemainformativo; + } + + public DatasourceEpasRest getDatasourceEpasRest() { + return datasourceEpasRest; + } +} diff --git a/src/main/java/it/cnr/isti/epasmed/config/AsyncConfiguration.java b/src/main/java/it/cnr/isti/epasmed/config/AsyncConfiguration.java new file mode 100644 index 0000000..b7d84f9 --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/config/AsyncConfiguration.java @@ -0,0 +1,47 @@ +package it.cnr.isti.epasmed.config; + +import io.github.jhipster.async.ExceptionHandlingAsyncTaskExecutor; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; +import org.springframework.aop.interceptor.AsyncUncaughtExceptionHandler; +import org.springframework.aop.interceptor.SimpleAsyncUncaughtExceptionHandler; +import org.springframework.boot.autoconfigure.task.TaskExecutionProperties; +import org.springframework.context.annotation.Bean; +import org.springframework.context.annotation.Configuration; +import org.springframework.scheduling.annotation.AsyncConfigurer; +import org.springframework.scheduling.annotation.EnableAsync; +import org.springframework.scheduling.annotation.EnableScheduling; +import org.springframework.scheduling.concurrent.ThreadPoolTaskExecutor; + +import java.util.concurrent.Executor; + +@Configuration +@EnableAsync +@EnableScheduling +public class AsyncConfiguration implements AsyncConfigurer { + + private final Logger log = LoggerFactory.getLogger(AsyncConfiguration.class); + + private final TaskExecutionProperties taskExecutionProperties; + + public AsyncConfiguration(TaskExecutionProperties taskExecutionProperties) { + this.taskExecutionProperties = taskExecutionProperties; + } + + @Override + @Bean(name = "taskExecutor") + public Executor getAsyncExecutor() { + log.debug("Creating Async Task Executor"); + ThreadPoolTaskExecutor executor = new ThreadPoolTaskExecutor(); + executor.setCorePoolSize(taskExecutionProperties.getPool().getCoreSize()); + executor.setMaxPoolSize(taskExecutionProperties.getPool().getMaxSize()); + executor.setQueueCapacity(taskExecutionProperties.getPool().getQueueCapacity()); + executor.setThreadNamePrefix(taskExecutionProperties.getThreadNamePrefix()); + return new ExceptionHandlingAsyncTaskExecutor(executor); + } + + @Override + public AsyncUncaughtExceptionHandler getAsyncUncaughtExceptionHandler() { + return new SimpleAsyncUncaughtExceptionHandler(); + } +} diff --git a/src/main/java/it/cnr/isti/epasmed/config/CloudDatabaseConfiguration.java b/src/main/java/it/cnr/isti/epasmed/config/CloudDatabaseConfiguration.java new file mode 100644 index 0000000..30cda4f --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/config/CloudDatabaseConfiguration.java @@ -0,0 +1,28 @@ +package it.cnr.isti.epasmed.config; + +import io.github.jhipster.config.JHipsterConstants; + +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; +import org.springframework.cloud.config.java.AbstractCloudConfig; +import org.springframework.context.annotation.*; + +import javax.sql.DataSource; +import org.springframework.boot.context.properties.ConfigurationProperties; + + +@Configuration +@Profile(JHipsterConstants.SPRING_PROFILE_CLOUD) +public class CloudDatabaseConfiguration extends AbstractCloudConfig { + + private final Logger log = LoggerFactory.getLogger(CloudDatabaseConfiguration.class); + + private static final String CLOUD_CONFIGURATION_HIKARI_PREFIX = "spring.datasource.hikari"; + + @Bean + @ConfigurationProperties(CLOUD_CONFIGURATION_HIKARI_PREFIX) + public DataSource dataSource() { + log.info("Configuring JDBC datasource from a cloud provider"); + return connectionFactory().dataSource(); + } +} diff --git a/src/main/java/it/cnr/isti/epasmed/config/Constants.java b/src/main/java/it/cnr/isti/epasmed/config/Constants.java new file mode 100644 index 0000000..9e255f9 --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/config/Constants.java @@ -0,0 +1,17 @@ +package it.cnr.isti.epasmed.config; + +/** + * Application constants. + */ +public final class Constants { + + // Regex for acceptable logins + public static final String LOGIN_REGEX = "^(?>[a-zA-Z0-9!$&*+=?^_`{|}~.-]+@[a-zA-Z0-9-]+(?:\\.[a-zA-Z0-9-]+)*)|(?>[_.@A-Za-z0-9-]+)$"; + + public static final String SYSTEM_ACCOUNT = "system"; + public static final String DEFAULT_LANGUAGE = "en"; + public static final String ANONYMOUS_USER = "anonymoususer"; + + private Constants() { + } +} diff --git a/src/main/java/it/cnr/isti/epasmed/config/DateTimeFormatConfiguration.java b/src/main/java/it/cnr/isti/epasmed/config/DateTimeFormatConfiguration.java new file mode 100644 index 0000000..9c22050 --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/config/DateTimeFormatConfiguration.java @@ -0,0 +1,22 @@ +package it.cnr.isti.epasmed.config; + +import org.springframework.context.annotation.Configuration; +import org.springframework.format.FormatterRegistry; +import org.springframework.format.datetime.standard.DateTimeFormatterRegistrar; +import org.springframework.web.servlet.config.annotation.WebMvcConfigurer; + +/** + * Configure the converters to use the ISO format for dates by default. + */ +@Configuration +public class DateTimeFormatConfiguration implements WebMvcConfigurer { + + @Override + public void addFormatters(FormatterRegistry registry) { + DateTimeFormatterRegistrar registrar = new DateTimeFormatterRegistrar(); + registrar.setUseIsoFormat(true); + registrar.registerFormatters(registry); + } + + +} diff --git a/src/main/java/it/cnr/isti/epasmed/config/EpasMedDataSourceConfiguration.java b/src/main/java/it/cnr/isti/epasmed/config/EpasMedDataSourceConfiguration.java new file mode 100644 index 0000000..c0b677f --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/config/EpasMedDataSourceConfiguration.java @@ -0,0 +1,202 @@ +package it.cnr.isti.epasmed.config; + +import java.sql.SQLException; +import java.util.HashMap; + +import javax.sql.DataSource; + +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; +import org.springframework.beans.factory.annotation.Qualifier; +import org.springframework.boot.orm.jpa.EntityManagerFactoryBuilder; +import org.springframework.context.annotation.Bean; +import org.springframework.context.annotation.Configuration; +import org.springframework.context.annotation.Primary; +import org.springframework.context.annotation.Profile; +import org.springframework.core.env.Environment; +import org.springframework.core.env.Profiles; +import org.springframework.data.jpa.repository.config.EnableJpaAuditing; +import org.springframework.data.jpa.repository.config.EnableJpaRepositories; +import org.springframework.orm.jpa.JpaTransactionManager; +import org.springframework.orm.jpa.LocalContainerEntityManagerFactoryBean; +import org.springframework.orm.jpa.vendor.HibernateJpaVendorAdapter; +import org.springframework.transaction.PlatformTransactionManager; +import org.springframework.transaction.annotation.EnableTransactionManagement; + +import com.zaxxer.hikari.HikariConfig; +import com.zaxxer.hikari.HikariDataSource; + +import io.github.jhipster.config.JHipsterConstants; +import io.github.jhipster.config.h2.H2ConfigurationHelper; +import it.cnr.isti.epasmed.config.ApplicationProperties.DataSourceProperties; +import it.cnr.isti.epasmed.config.ApplicationProperties.DatasourceEpasmed; +import it.cnr.isti.epasmed.config.ApplicationProperties.HikariProperties; +import it.cnr.isti.epasmed.config.ApplicationProperties.LiquibaseProperties; +import liquibase.integration.spring.SpringLiquibase; + +@Configuration +@EnableJpaRepositories(basePackages = "it.cnr.isti.epasmed.repository", entityManagerFactoryRef = "epasMedEntityManagerFactory", transactionManagerRef = "epasMedTransactionManager") +@EnableJpaAuditing(auditorAwareRef = "springSecurityAuditorAware") +@EnableTransactionManagement +public class EpasMedDataSourceConfiguration { + + private final Logger log = LoggerFactory.getLogger(EpasMedDataSourceConfiguration.class); + + private final Environment env; + private final ApplicationProperties applicationProperties; + + public EpasMedDataSourceConfiguration(Environment env, ApplicationProperties applicationProperties) { + this.env = env; + this.applicationProperties = applicationProperties; + } + + /** + * Open the TCP port for the H2 database, so it is available remotely. + * + * @return the H2 database TCP server. + * @throws SQLException if the server failed to start. + */ + @Bean(initMethod = "start", destroyMethod = "stop") + @Profile(JHipsterConstants.SPRING_PROFILE_DEVELOPMENT) + public Object h2TCPServer() throws SQLException { + String port = getValidPortForH2(); + log.debug("H2 database is available on port {}", port); + return H2ConfigurationHelper.createServer(port); + } + + private String getValidPortForH2() { + int port = Integer.parseInt(env.getProperty("server.port")); + if (port < 10000) { + port = 10000 + port; + } else { + if (port < 63536) { + port = port + 2000; + } else { + port = port - 2000; + } + } + return String.valueOf(port); + } + + @Bean + @Primary + public DatasourceEpasmed epasMedDataSourceProperties() { + return applicationProperties.getDatasourceEpasmed(); + } + + @Bean + @Primary + public DataSource epasMedDataSource() { + log.info("Setup ePASMedDataSource"); + DataSourceProperties dsp = epasMedDataSourceProperties().getDatasource(); + log.info("Properties: {}",dsp); + HikariProperties hikariProperties = epasMedDataSourceProperties().getHikari(); + log.info("Hikari: {}",hikariProperties); + + HikariConfig config = new HikariConfig(); + config.setJdbcUrl(dsp.getUrl()); + config.setUsername(dsp.getUsername()); + config.setPassword(dsp.getPassword()); + config.addDataSourceProperty("cachePrepStmts", hikariProperties.isCachePrepStmts()); + config.addDataSourceProperty("prepStmtCacheSize", hikariProperties.getPrepStmtCacheSize()); + config.addDataSourceProperty("prepStmtCacheSqlLimit", hikariProperties.getPrepStmtCacheSqlLimit()); + config.setAutoCommit(hikariProperties.isAutoCommit()); + config.setPoolName(hikariProperties.getPoolName()); + config.setMaximumPoolSize(hikariProperties.getMaximumPoolSize()); + log.info("HikariConfig epasmed: {}", dsp.getUrl()); + HikariDataSource ds = new HikariDataSource(config); + return ds; + } + + @Bean(name = "epasMedEntityManagerFactory") + @Primary + public LocalContainerEntityManagerFactoryBean epasMedEntityManagerFactory(EntityManagerFactoryBuilder builder) { + LocalContainerEntityManagerFactoryBean em = builder.dataSource(epasMedDataSource()).persistenceUnit("epasMedPU") + .packages("it.cnr.isti.epasmed.domain").build(); + + HibernateJpaVendorAdapter vendorAdapter = new HibernateJpaVendorAdapter(); + // vendorAdapter.setDatabase(Database.DB2); + vendorAdapter.setShowSql(true); + vendorAdapter.setDatabasePlatform(epasMedDataSourceProperties().getDatasource().getPlatform()); + + + em.setJpaVendorAdapter(vendorAdapter); + HashMap properties = new HashMap<>(); + properties.put("hibernate.jdbc.time_zone", "UTC"); + properties.put("hibernate.id.new_generator_mappings", true); + properties.put("hibernate.connection.provider_disables_autocommit", true); + properties.put("hibernate.cache.use_second_level_cache", false); + properties.put("hibernate.cache.use_query_cache", false); + properties.put("hibernate.generate_statistics", false); + // modify batch size as necessary + properties.put("hibernate.jdbc.batch_size", 25); + properties.put("hibernate.order_inserts", true); + properties.put("hibernate.order_updates", true); + properties.put("hibernate.jdbc.lob.non_contextual_creation", true); + properties.put("hibernate.query.fail_on_pagination_over_collection_fetch", true); + properties.put("hibernate.query.in_clause_parameter_padding", true); + + // Naming props + properties.put("hibernate.ddl-auto", "none"); + properties.put("hibernate.physical_naming_strategy", + "org.springframework.boot.orm.jpa.hibernate.SpringPhysicalNamingStrategy"); + properties.put("hibernate.implicit_naming_strategy", + "org.springframework.boot.orm.jpa.hibernate.SpringImplicitNamingStrategy"); + //org.hibernate.boot.model.naming.ImplicitNamingStrategyLegacyJpaImpl + + + properties.put("hibernate.hbm2ddl.auto","none"); + // properties.put("hibernate.dialect", env.getProperty("hibernate.dialect")); + em.setJpaPropertyMap(properties); + return em; + } + + @Bean + @Primary + public PlatformTransactionManager epasMedTransactionManager( + final @Qualifier("epasMedEntityManagerFactory") LocalContainerEntityManagerFactoryBean epasMedEntityManagerFactory) { + return new JpaTransactionManager(epasMedEntityManagerFactory.getObject()); + } + + @Bean + public LiquibaseProperties epasMedLiquibaseProperties() { + return applicationProperties.getDatasourceEpasmed().getLiquibase(); + } + + @Bean + public SpringLiquibase epasMedLiquibase() { + return liquibase(epasMedLiquibaseProperties(), epasMedDataSource()); + } + + private SpringLiquibase liquibase(LiquibaseProperties epasMedLiquibaseProperties, DataSource epasMedDataSource) { + log.debug("Create EpasMed Liquibase DataSource"); + SpringLiquibase liquibase = new SpringLiquibase(); + liquibase.setDataSource(epasMedDataSource); + // SpringLiquibaseUtil.createAsyncSpringLiquibase(env, + // asyncConfiguration.getAsyncExecutor(), liquibaseDataSource, + // epasMedLiquibaseProperties, + // epasMedDataSource, epasMedDataSourceProperties); + log.debug("Set params"); + liquibase.setContexts(epasMedLiquibaseProperties.getContexts()); + liquibase.setChangeLog(epasMedLiquibaseProperties.getChangeLog()); + liquibase.setDefaultSchema(epasMedLiquibaseProperties.getDefaultSchema()); + liquibase.setLiquibaseSchema(epasMedLiquibaseProperties.getLiquibaseSchema()); + liquibase.setLiquibaseTablespace(epasMedLiquibaseProperties.getLiquibaseTablespace()); + liquibase.setDatabaseChangeLogLockTable(epasMedLiquibaseProperties.getDatabaseChangeLogLockTable()); + liquibase.setDatabaseChangeLogTable(epasMedLiquibaseProperties.getDatabaseChangeLogTable()); + liquibase.setDropFirst(epasMedLiquibaseProperties.isDropFirst()); + liquibase.setLabels(epasMedLiquibaseProperties.getLabels()); + liquibase.setChangeLogParameters(epasMedLiquibaseProperties.getParameters()); + liquibase.setRollbackFile(epasMedLiquibaseProperties.getRollbackFile()); + liquibase.setTestRollbackOnUpdate(epasMedLiquibaseProperties.isTestRollbackOnUpdate()); + if (env.acceptsProfiles(Profiles.of(JHipsterConstants.SPRING_PROFILE_NO_LIQUIBASE))) { + liquibase.setShouldRun(false); + } else { + liquibase.setShouldRun(epasMedLiquibaseProperties.isEnabled()); + log.debug("Configuring epasMed Liquibase"); + } + log.debug("Created epasMed Libquibase"); + return liquibase; + } + +} diff --git a/src/main/java/it/cnr/isti/epasmed/config/EpasRestTemplateConfig.java b/src/main/java/it/cnr/isti/epasmed/config/EpasRestTemplateConfig.java new file mode 100644 index 0000000..ab585cd --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/config/EpasRestTemplateConfig.java @@ -0,0 +1,36 @@ +package it.cnr.isti.epasmed.config; + +import org.springframework.boot.web.client.RestTemplateBuilder; +import org.springframework.context.annotation.Bean; +import org.springframework.context.annotation.Configuration; +import org.springframework.context.annotation.Primary; +import org.springframework.http.MediaType; +import org.springframework.web.client.RestTemplate; + +@Configuration +public class EpasRestTemplateConfig { + + private final ApplicationProperties applicationProperties; + + public EpasRestTemplateConfig(ApplicationProperties applicationProperties) { + super(); + this.applicationProperties = applicationProperties; + } + + @Bean(name = "RestTemplateForFirstUser") + @Primary + public RestTemplate restTemplateFirst(RestTemplateBuilder builder) { + return builder.defaultHeader("Accept", MediaType.APPLICATION_JSON.toString()) + .basicAuthentication(applicationProperties.getDatasourceEpasRest().getRestUsername1(), + applicationProperties.getDatasourceEpasRest().getRestPassword1()) + .build(); + } + + @Bean(name = "RestTemplateForSecondUser") + public RestTemplate restTemplateSecond(RestTemplateBuilder builder) { + return builder.defaultHeader("Accept", MediaType.APPLICATION_JSON.toString()) + .basicAuthentication(applicationProperties.getDatasourceEpasRest().getRestUsername2(), + applicationProperties.getDatasourceEpasRest().getRestPassword2()) + .build(); + } +} diff --git a/src/main/java/it/cnr/isti/epasmed/config/JacksonConfiguration.java b/src/main/java/it/cnr/isti/epasmed/config/JacksonConfiguration.java new file mode 100644 index 0000000..d6e792e --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/config/JacksonConfiguration.java @@ -0,0 +1,52 @@ +package it.cnr.isti.epasmed.config; + +import com.fasterxml.jackson.datatype.hibernate5.Hibernate5Module; +import com.fasterxml.jackson.datatype.jdk8.Jdk8Module; +import com.fasterxml.jackson.datatype.jsr310.JavaTimeModule; + +import org.springframework.context.annotation.Bean; +import org.springframework.context.annotation.Configuration; +import org.zalando.problem.ProblemModule; +import org.zalando.problem.violations.ConstraintViolationProblemModule; + +@Configuration +public class JacksonConfiguration { + + /** + * Support for Java date and time API. + * @return the corresponding Jackson module. + */ + @Bean + public JavaTimeModule javaTimeModule() { + return new JavaTimeModule(); + } + + @Bean + public Jdk8Module jdk8TimeModule() { + return new Jdk8Module(); + } + + /* + * Support for Hibernate types in Jackson. + */ + @Bean + public Hibernate5Module hibernate5Module() { + return new Hibernate5Module(); + } + + /* + * Module for serialization/deserialization of RFC7807 Problem. + */ + @Bean + public ProblemModule problemModule() { + return new ProblemModule(); + } + + /* + * Module for serialization/deserialization of ConstraintViolationProblem. + */ + @Bean + public ConstraintViolationProblemModule constraintViolationProblemModule() { + return new ConstraintViolationProblemModule(); + } +} diff --git a/src/main/java/it/cnr/isti/epasmed/config/LocaleConfiguration.java b/src/main/java/it/cnr/isti/epasmed/config/LocaleConfiguration.java new file mode 100644 index 0000000..0fbe886 --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/config/LocaleConfiguration.java @@ -0,0 +1,27 @@ +package it.cnr.isti.epasmed.config; + +import io.github.jhipster.config.locale.AngularCookieLocaleResolver; + +import org.springframework.context.annotation.Bean; +import org.springframework.context.annotation.Configuration; +import org.springframework.web.servlet.LocaleResolver; +import org.springframework.web.servlet.config.annotation.*; +import org.springframework.web.servlet.i18n.LocaleChangeInterceptor; + +@Configuration +public class LocaleConfiguration implements WebMvcConfigurer { + + @Bean + public LocaleResolver localeResolver() { + AngularCookieLocaleResolver cookieLocaleResolver = new AngularCookieLocaleResolver(); + cookieLocaleResolver.setCookieName("NG_TRANSLATE_LANG_KEY"); + return cookieLocaleResolver; + } + + @Override + public void addInterceptors(InterceptorRegistry registry) { + LocaleChangeInterceptor localeChangeInterceptor = new LocaleChangeInterceptor(); + localeChangeInterceptor.setParamName("language"); + registry.addInterceptor(localeChangeInterceptor); + } +} diff --git a/src/main/java/it/cnr/isti/epasmed/config/LoggingAspectConfiguration.java b/src/main/java/it/cnr/isti/epasmed/config/LoggingAspectConfiguration.java new file mode 100644 index 0000000..5a438ab --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/config/LoggingAspectConfiguration.java @@ -0,0 +1,19 @@ +package it.cnr.isti.epasmed.config; + +import it.cnr.isti.epasmed.aop.logging.LoggingAspect; + +import io.github.jhipster.config.JHipsterConstants; + +import org.springframework.context.annotation.*; +import org.springframework.core.env.Environment; + +@Configuration +@EnableAspectJAutoProxy +public class LoggingAspectConfiguration { + + @Bean + @Profile(JHipsterConstants.SPRING_PROFILE_DEVELOPMENT) + public LoggingAspect loggingAspect(Environment env) { + return new LoggingAspect(env); + } +} diff --git a/src/main/java/it/cnr/isti/epasmed/config/LoggingConfiguration.java b/src/main/java/it/cnr/isti/epasmed/config/LoggingConfiguration.java new file mode 100644 index 0000000..7d56ea4 --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/config/LoggingConfiguration.java @@ -0,0 +1,50 @@ +package it.cnr.isti.epasmed.config; + +import ch.qos.logback.classic.LoggerContext; +import com.fasterxml.jackson.core.JsonProcessingException; +import com.fasterxml.jackson.databind.ObjectMapper; +import io.github.jhipster.config.JHipsterProperties; +import org.slf4j.LoggerFactory; +import org.springframework.beans.factory.annotation.Value; +import org.springframework.context.annotation.Configuration; + +import java.util.HashMap; +import java.util.Map; + +import static io.github.jhipster.config.logging.LoggingUtils.*; + +/* + * Configures the console and Logstash log appenders from the app properties + */ +@Configuration +public class LoggingConfiguration { + + public LoggingConfiguration(@Value("${spring.application.name}") String appName, + @Value("${server.port}") String serverPort, + JHipsterProperties jHipsterProperties, + ObjectMapper mapper) throws JsonProcessingException { + + LoggerContext context = (LoggerContext) LoggerFactory.getILoggerFactory(); + + Map map = new HashMap<>(); + map.put("app_name", appName); + map.put("app_port", serverPort); + String customFields = mapper.writeValueAsString(map); + + JHipsterProperties.Logging loggingProperties = jHipsterProperties.getLogging(); + JHipsterProperties.Logging.Logstash logstashProperties = loggingProperties.getLogstash(); + + if (loggingProperties.isUseJsonFormat()) { + addJsonConsoleAppender(context, customFields); + } + if (logstashProperties.isEnabled()) { + addLogstashTcpSocketAppender(context, customFields, logstashProperties); + } + if (loggingProperties.isUseJsonFormat() || logstashProperties.isEnabled()) { + addContextListener(context, customFields, loggingProperties); + } + if (jHipsterProperties.getMetrics().getLogs().isEnabled()) { + setMetricsMarkerLogbackFilter(context, loggingProperties.isUseJsonFormat()); + } + } +} diff --git a/src/main/java/it/cnr/isti/epasmed/config/OpenApiConfiguration.java b/src/main/java/it/cnr/isti/epasmed/config/OpenApiConfiguration.java new file mode 100644 index 0000000..2a9827a --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/config/OpenApiConfiguration.java @@ -0,0 +1,72 @@ +package it.cnr.isti.epasmed.config; + +import com.google.common.base.Predicates; +import io.github.jhipster.config.JHipsterConstants; +import io.github.jhipster.config.JHipsterProperties; +import io.github.jhipster.config.apidoc.customizer.SwaggerCustomizer; +import org.springframework.context.annotation.Bean; +import org.springframework.context.annotation.Configuration; +import org.springframework.context.annotation.Profile; +import org.springframework.data.domain.Pageable; +import org.springframework.http.ResponseEntity; +import springfox.documentation.builders.RequestHandlerSelectors; +import springfox.documentation.service.ApiInfo; +import springfox.documentation.service.Contact; +import springfox.documentation.spi.DocumentationType; +import springfox.documentation.spring.web.plugins.Docket; + +import java.nio.ByteBuffer; +import java.util.ArrayList; +import java.util.Arrays; +import java.util.HashSet; + +import static springfox.documentation.builders.PathSelectors.regex; + +@Configuration +@Profile(JHipsterConstants.SPRING_PROFILE_SWAGGER) +public class OpenApiConfiguration { + + @Bean + public SwaggerCustomizer noApiFirstCustomizer() { + return docket -> docket.select() + .apis(Predicates.not(RequestHandlerSelectors.basePackage("it.cnr.isti.epasmed.web.api"))); + } + + @Bean + public Docket apiFirstDocket(JHipsterProperties jHipsterProperties) { + JHipsterProperties.Swagger properties = jHipsterProperties.getSwagger(); + Contact contact = new Contact( + properties.getContactName(), + properties.getContactUrl(), + properties.getContactEmail() + ); + + ApiInfo apiInfo = new ApiInfo( + "API First " + properties.getTitle(), + properties.getDescription(), + properties.getVersion(), + properties.getTermsOfServiceUrl(), + contact, + properties.getLicense(), + properties.getLicenseUrl(), + new ArrayList<>() + ); + + return new Docket(DocumentationType.SWAGGER_2) + .groupName("openapi") + .host(properties.getHost()) + .protocols(new HashSet<>(Arrays.asList(properties.getProtocols()))) + .apiInfo(apiInfo) + .useDefaultResponseMessages(properties.isUseDefaultResponseMessages()) + .forCodeGeneration(true) + .directModelSubstitute(ByteBuffer.class, String.class) + .genericModelSubstitutes(ResponseEntity.class) + .ignoredParameterTypes(Pageable.class) + .select() + .apis(RequestHandlerSelectors.basePackage("it.cnr.isti.epasmed.web.api")) + .paths(regex(properties.getDefaultIncludePattern())) + .build(); + } + + +} diff --git a/src/main/java/it/cnr/isti/epasmed/config/SecurityConfiguration.java b/src/main/java/it/cnr/isti/epasmed/config/SecurityConfiguration.java new file mode 100644 index 0000000..f6fa1af --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/config/SecurityConfiguration.java @@ -0,0 +1,98 @@ +package it.cnr.isti.epasmed.config; + +import it.cnr.isti.epasmed.security.*; + +import io.github.jhipster.config.JHipsterProperties; +import io.github.jhipster.security.*; + +import org.springframework.context.annotation.Bean; +import org.springframework.context.annotation.Import; +import org.springframework.http.HttpMethod; +import org.springframework.security.config.annotation.method.configuration.EnableGlobalMethodSecurity; +import org.springframework.security.config.annotation.web.builders.HttpSecurity; +import org.springframework.security.config.annotation.web.builders.WebSecurity; +import org.springframework.security.config.annotation.web.configuration.EnableWebSecurity; +import org.springframework.security.config.annotation.web.configuration.WebSecurityConfigurerAdapter; +import org.springframework.security.crypto.bcrypt.BCryptPasswordEncoder; +import org.springframework.security.crypto.password.PasswordEncoder; +import org.springframework.security.web.authentication.RememberMeServices; +import org.springframework.security.web.csrf.CookieCsrfTokenRepository; +import org.springframework.security.web.csrf.CsrfFilter; +import org.springframework.security.web.header.writers.ReferrerPolicyHeaderWriter; +import org.springframework.web.filter.CorsFilter; +import org.zalando.problem.spring.web.advice.security.SecurityProblemSupport; + +@EnableWebSecurity +@EnableGlobalMethodSecurity(prePostEnabled = true, securedEnabled = true) +@Import(SecurityProblemSupport.class) +public class SecurityConfiguration extends WebSecurityConfigurerAdapter { + + private final JHipsterProperties jHipsterProperties; + + private final RememberMeServices rememberMeServices; + + private final CorsFilter corsFilter; + private final SecurityProblemSupport problemSupport; + + public SecurityConfiguration(JHipsterProperties jHipsterProperties, RememberMeServices rememberMeServices, + CorsFilter corsFilter, SecurityProblemSupport problemSupport) { + this.jHipsterProperties = jHipsterProperties; + this.rememberMeServices = rememberMeServices; + this.corsFilter = corsFilter; + this.problemSupport = problemSupport; + } + + @Bean + public AjaxAuthenticationSuccessHandler ajaxAuthenticationSuccessHandler() { + return new AjaxAuthenticationSuccessHandler(); + } + + @Bean + public AjaxAuthenticationFailureHandler ajaxAuthenticationFailureHandler() { + return new AjaxAuthenticationFailureHandler(); + } + + @Bean + public AjaxLogoutSuccessHandler ajaxLogoutSuccessHandler() { + return new AjaxLogoutSuccessHandler(); + } + + @Bean + public PasswordEncoder passwordEncoder() { + return new BCryptPasswordEncoder(); + } + + @Override + public void configure(WebSecurity web) { + web.ignoring().antMatchers(HttpMethod.OPTIONS, "/**").antMatchers("/app/**/*.{js,html}").antMatchers("/i18n/**") + .antMatchers("/content/**").antMatchers("/h2-console/**").antMatchers("/swagger-ui/index.html") + .antMatchers("/test/**"); + } + + @Override + public void configure(HttpSecurity http) throws Exception { + // @formatter:off + http.csrf().csrfTokenRepository(CookieCsrfTokenRepository.withHttpOnlyFalse()).and() + .addFilterBefore(corsFilter, CsrfFilter.class).exceptionHandling() + .authenticationEntryPoint(problemSupport).accessDeniedHandler(problemSupport).and().rememberMe() + .rememberMeServices(rememberMeServices).rememberMeParameter("remember-me") + .key(jHipsterProperties.getSecurity().getRememberMe().getKey()).and().formLogin() + .loginProcessingUrl("/api/authentication").successHandler(ajaxAuthenticationSuccessHandler()) + .failureHandler(ajaxAuthenticationFailureHandler()).permitAll().and().logout().logoutUrl("/api/logout") + .logoutSuccessHandler(ajaxLogoutSuccessHandler()).permitAll().and().headers() + .contentSecurityPolicy( + "default-src 'self'; frame-src 'self' data:; script-src 'self' 'unsafe-inline' 'unsafe-eval' https://storage.googleapis.com; style-src 'self' 'unsafe-inline'; img-src 'self' data:; font-src 'self' data:") + .and().referrerPolicy(ReferrerPolicyHeaderWriter.ReferrerPolicy.STRICT_ORIGIN_WHEN_CROSS_ORIGIN).and() + .featurePolicy( + "geolocation 'none'; midi 'none'; sync-xhr 'none'; microphone 'none'; camera 'none'; magnetometer 'none'; gyroscope 'none'; speaker 'none'; fullscreen 'self'; payment 'none'") + .and().frameOptions().deny().and().authorizeRequests().antMatchers("/api/authenticate").permitAll() + .antMatchers("/api/register").hasAnyAuthority(AuthoritiesConstants.ADMIN).antMatchers("/api/activate") + .permitAll().antMatchers("/api/account/reset-password/init").permitAll() + .antMatchers("/api/account/reset-password/finish").permitAll().antMatchers("/api/**").authenticated() + .antMatchers("/management/health").permitAll().antMatchers("/management/info").permitAll() + .antMatchers("/management/prometheus") + .hasAnyAuthority(AuthoritiesConstants.ADMIN, AuthoritiesConstants.PROMETHEUS) + .antMatchers("/management/**").hasAuthority(AuthoritiesConstants.ADMIN); + // @formatter:on + } +} diff --git a/src/main/java/it/cnr/isti/epasmed/config/SistemaInformativoDataSourceConfiguration.java b/src/main/java/it/cnr/isti/epasmed/config/SistemaInformativoDataSourceConfiguration.java new file mode 100644 index 0000000..be109a6 --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/config/SistemaInformativoDataSourceConfiguration.java @@ -0,0 +1,88 @@ +package it.cnr.isti.epasmed.config; + +import javax.sql.DataSource; + +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; +import org.springframework.beans.factory.annotation.Qualifier; +import org.springframework.context.annotation.Bean; +import org.springframework.context.annotation.Configuration; +import org.springframework.data.jdbc.repository.config.AbstractJdbcConfiguration; +import org.springframework.data.jdbc.repository.config.EnableJdbcRepositories; +import org.springframework.jdbc.core.namedparam.NamedParameterJdbcOperations; +import org.springframework.jdbc.core.namedparam.NamedParameterJdbcTemplate; +import org.springframework.jdbc.datasource.DataSourceTransactionManager; +import org.springframework.jdbc.datasource.DriverManagerDataSource; +import org.springframework.transaction.TransactionManager; +import org.springframework.transaction.annotation.EnableTransactionManagement; + +import com.zaxxer.hikari.HikariConfig; +import com.zaxxer.hikari.HikariDataSource; + +import it.cnr.isti.epasmed.config.ApplicationProperties.DataSourceProperties; +import it.cnr.isti.epasmed.config.ApplicationProperties.DatasourceSistemainformativo; +import it.cnr.isti.epasmed.config.ApplicationProperties.HikariProperties; + +@Configuration +@EnableJdbcRepositories(basePackages = "it.cnr.isti.epasmed.sistemainformativo.repository") +//@EnableJpaRepositories(basePackages = "it.cnr.isti.epasmed.sistemainformativo.repository", entityManagerFactoryRef = "sistemaInformativoEntityManagerFactory", transactionManagerRef = "sistemaInformativoTransactionManager") +//@EnableJpaAuditing(auditorAwareRef = "springSecurityAuditorAware") +@EnableTransactionManagement +public class SistemaInformativoDataSourceConfiguration extends AbstractJdbcConfiguration { + + private final Logger log = LoggerFactory.getLogger(SistemaInformativoDataSourceConfiguration.class); + + private final ApplicationProperties applicationProperties; + + public SistemaInformativoDataSourceConfiguration(ApplicationProperties applicationProperties) { + this.applicationProperties = applicationProperties; + } + + @Bean + public DatasourceSistemainformativo sistemaInformativoDataSourceProperties() { + return applicationProperties.getDatasourceSistemainformativo(); + } + + @Bean(name = "sistemaInformativoDataSource") + public DataSource sistemaInformativoDataSource() { + DataSourceProperties dsp = sistemaInformativoDataSourceProperties().getDatasource(); + HikariProperties hikariProperties = sistemaInformativoDataSourceProperties().getHikari(); + + //DriverManagerDataSource dataSource = new DriverManagerDataSource(); + //dataSource.setDriverClassName(dsp.getDriverClassName()); + //dataSource.setUrl(dsp.getUrl()); + //dataSource.setUsername(dsp.getUsername()); + //dataSource.setPassword(dsp.getPassword()); + //log.info("SistemaInformativoDataSource created"); + + + HikariConfig config = new HikariConfig(); + config.setJdbcUrl(dsp.getUrl()); + config.setDriverClassName(dsp.getDriverClassName()); + config.setUsername(dsp.getUsername()); + config.setPassword(dsp.getPassword()); + config.addDataSourceProperty("cachePrepStmts", hikariProperties.isCachePrepStmts()); + config.addDataSourceProperty("prepStmtCacheSize", hikariProperties.getPrepStmtCacheSize()); + config.addDataSourceProperty("prepStmtCacheSqlLimit", hikariProperties.getPrepStmtCacheSqlLimit()); + config.setAutoCommit(hikariProperties.isAutoCommit()); + config.setPoolName(hikariProperties.getPoolName()); + config.setMaximumPoolSize(hikariProperties.getMaximumPoolSize()); + log.debug("HikariConfig sistemainformativo: " + dsp.getUrl()); + HikariDataSource dataSource = new HikariDataSource(config); + return dataSource; + } + + + + @Bean + NamedParameterJdbcOperations namedParameterJdbcOperations( + final @Qualifier("sistemaInformativoDataSource") DataSource dataSource) { + return new NamedParameterJdbcTemplate(dataSource); + } + + @Bean + TransactionManager sistemaInformativoTransactionManager( + final @Qualifier("sistemaInformativoDataSource") DataSource dataSource) { + return new DataSourceTransactionManager(dataSource); + } +} diff --git a/src/main/java/it/cnr/isti/epasmed/config/StaticResourcesWebConfiguration.java b/src/main/java/it/cnr/isti/epasmed/config/StaticResourcesWebConfiguration.java new file mode 100644 index 0000000..f7f9ea1 --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/config/StaticResourcesWebConfiguration.java @@ -0,0 +1,49 @@ +package it.cnr.isti.epasmed.config; + +import io.github.jhipster.config.JHipsterConstants; +import io.github.jhipster.config.JHipsterProperties; +import org.springframework.context.annotation.Configuration; +import org.springframework.context.annotation.Profile; +import org.springframework.http.CacheControl; +import org.springframework.web.servlet.config.annotation.ResourceHandlerRegistration; +import org.springframework.web.servlet.config.annotation.ResourceHandlerRegistry; +import org.springframework.web.servlet.config.annotation.WebMvcConfigurer; + +import java.util.concurrent.TimeUnit; + +@Configuration +@Profile({JHipsterConstants.SPRING_PROFILE_PRODUCTION}) +public class StaticResourcesWebConfiguration implements WebMvcConfigurer { + + protected static final String[] RESOURCE_LOCATIONS = new String[]{"classpath:/static/app/", "classpath:/static/content/", "classpath:/static/i18n/"}; + protected static final String[] RESOURCE_PATHS = new String[]{"/app/*", "/content/*", "/i18n/*"}; + + private final JHipsterProperties jhipsterProperties; + + public StaticResourcesWebConfiguration(JHipsterProperties jHipsterProperties) { + this.jhipsterProperties = jHipsterProperties; + } + + @Override + public void addResourceHandlers(ResourceHandlerRegistry registry) { + ResourceHandlerRegistration resourceHandlerRegistration = appendResourceHandler(registry); + initializeResourceHandler(resourceHandlerRegistration); + } + + protected ResourceHandlerRegistration appendResourceHandler(ResourceHandlerRegistry registry) { + return registry.addResourceHandler(RESOURCE_PATHS); + } + + protected void initializeResourceHandler(ResourceHandlerRegistration resourceHandlerRegistration) { + resourceHandlerRegistration.addResourceLocations(RESOURCE_LOCATIONS).setCacheControl(getCacheControl()); + } + + protected CacheControl getCacheControl() { + return CacheControl.maxAge(getJHipsterHttpCacheProperty(), TimeUnit.DAYS).cachePublic(); + } + + private int getJHipsterHttpCacheProperty() { + return jhipsterProperties.getHttp().getCache().getTimeToLiveInDays(); + } + +} diff --git a/src/main/java/it/cnr/isti/epasmed/config/WebConfigurer.java b/src/main/java/it/cnr/isti/epasmed/config/WebConfigurer.java new file mode 100644 index 0000000..9ed4911 --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/config/WebConfigurer.java @@ -0,0 +1,138 @@ +package it.cnr.isti.epasmed.config; + +import static java.net.URLDecoder.decode; + +import java.io.File; +import java.io.UnsupportedEncodingException; +import java.nio.charset.StandardCharsets; +import java.nio.file.Paths; + +import javax.servlet.ServletContext; +import javax.servlet.ServletException; + +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; +import org.springframework.boot.web.server.MimeMappings; +import org.springframework.boot.web.server.WebServerFactory; +import org.springframework.boot.web.server.WebServerFactoryCustomizer; +import org.springframework.boot.web.servlet.ServletContextInitializer; +import org.springframework.boot.web.servlet.server.ConfigurableServletWebServerFactory; +import org.springframework.context.annotation.Bean; +import org.springframework.context.annotation.Configuration; +import org.springframework.core.env.Environment; +import org.springframework.core.env.Profiles; +import org.springframework.http.MediaType; +import org.springframework.web.cors.CorsConfiguration; +import org.springframework.web.cors.UrlBasedCorsConfigurationSource; +import org.springframework.web.filter.CorsFilter; + +import io.github.jhipster.config.JHipsterConstants; +import io.github.jhipster.config.JHipsterProperties; +import io.github.jhipster.config.h2.H2ConfigurationHelper; + +/** + * Configuration of web application with Servlet 3.0 APIs. + */ +@Configuration +public class WebConfigurer implements ServletContextInitializer, WebServerFactoryCustomizer { + + private final Logger log = LoggerFactory.getLogger(WebConfigurer.class); + + private final Environment env; + + private final JHipsterProperties jHipsterProperties; + + public WebConfigurer(Environment env, JHipsterProperties jHipsterProperties) { + this.env = env; + this.jHipsterProperties = jHipsterProperties; + } + + @Override + public void onStartup(ServletContext servletContext) throws ServletException { + if (env.getActiveProfiles().length != 0) { + log.info("Web application configuration, using profiles: {}", (Object[]) env.getActiveProfiles()); + } + + if (env.acceptsProfiles(Profiles.of(JHipsterConstants.SPRING_PROFILE_DEVELOPMENT))) { + initH2Console(servletContext); + } + log.info("Web application fully configured"); + } + + /** + * Customize the Servlet engine: Mime types, the document root, the cache. + */ + @Override + public void customize(WebServerFactory server) { + setMimeMappings(server); + // When running in an IDE or with ./mvnw spring-boot:run, set location of the static web assets. + setLocationForStaticAssets(server); + } + + private void setMimeMappings(WebServerFactory server) { + if (server instanceof ConfigurableServletWebServerFactory) { + MimeMappings mappings = new MimeMappings(MimeMappings.DEFAULT); + // IE issue, see https://github.com/jhipster/generator-jhipster/pull/711 + mappings.add("html", MediaType.TEXT_HTML_VALUE + ";charset=" + StandardCharsets.UTF_8.name().toLowerCase()); + // CloudFoundry issue, see https://github.com/cloudfoundry/gorouter/issues/64 + mappings.add("json", MediaType.TEXT_HTML_VALUE + ";charset=" + StandardCharsets.UTF_8.name().toLowerCase()); + ConfigurableServletWebServerFactory servletWebServer = (ConfigurableServletWebServerFactory) server; + servletWebServer.setMimeMappings(mappings); + } + } + + private void setLocationForStaticAssets(WebServerFactory server) { + if (server instanceof ConfigurableServletWebServerFactory) { + ConfigurableServletWebServerFactory servletWebServer = (ConfigurableServletWebServerFactory) server; + File root; + String prefixPath = resolvePathPrefix(); + root = new File(prefixPath + "target/classes/static/"); + if (root.exists() && root.isDirectory()) { + servletWebServer.setDocumentRoot(root); + } + } + } + + /** + * Resolve path prefix to static resources. + */ + private String resolvePathPrefix() { + String fullExecutablePath; + try { + fullExecutablePath = decode(this.getClass().getResource("").getPath(), StandardCharsets.UTF_8.name()); + } catch (UnsupportedEncodingException e) { + /* try without decoding if this ever happens */ + fullExecutablePath = this.getClass().getResource("").getPath(); + } + String rootPath = Paths.get(".").toUri().normalize().getPath(); + String extractedPath = fullExecutablePath.replace(rootPath, ""); + int extractionEndIndex = extractedPath.indexOf("target/"); + if (extractionEndIndex <= 0) { + return ""; + } + return extractedPath.substring(0, extractionEndIndex); + } + + + @Bean + public CorsFilter corsFilter() { + UrlBasedCorsConfigurationSource source = new UrlBasedCorsConfigurationSource(); + CorsConfiguration config = jHipsterProperties.getCors(); + if (config.getAllowedOrigins() != null && !config.getAllowedOrigins().isEmpty()) { + log.debug("Registering CORS filter"); + source.registerCorsConfiguration("/api/**", config); + source.registerCorsConfiguration("/management/**", config); + source.registerCorsConfiguration("/v2/api-docs", config); + } + return new CorsFilter(source); + } + + /** + * Initializes H2 console. + */ + private void initH2Console(ServletContext servletContext) { + log.debug("Initialize H2 console"); + H2ConfigurationHelper.initH2Console(servletContext); + } + +} diff --git a/src/main/java/it/cnr/isti/epasmed/config/audit/AuditEventConverter.java b/src/main/java/it/cnr/isti/epasmed/config/audit/AuditEventConverter.java new file mode 100644 index 0000000..1937bbd --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/config/audit/AuditEventConverter.java @@ -0,0 +1,86 @@ +package it.cnr.isti.epasmed.config.audit; + +import it.cnr.isti.epasmed.domain.PersistentAuditEvent; + +import org.springframework.boot.actuate.audit.AuditEvent; +import org.springframework.security.web.authentication.WebAuthenticationDetails; +import org.springframework.stereotype.Component; + +import java.util.*; + +@Component +public class AuditEventConverter { + + /** + * Convert a list of {@link PersistentAuditEvent}s to a list of {@link AuditEvent}s. + * + * @param persistentAuditEvents the list to convert. + * @return the converted list. + */ + public List convertToAuditEvent(Iterable persistentAuditEvents) { + if (persistentAuditEvents == null) { + return Collections.emptyList(); + } + List auditEvents = new ArrayList<>(); + for (PersistentAuditEvent persistentAuditEvent : persistentAuditEvents) { + auditEvents.add(convertToAuditEvent(persistentAuditEvent)); + } + return auditEvents; + } + + /** + * Convert a {@link PersistentAuditEvent} to an {@link AuditEvent}. + * + * @param persistentAuditEvent the event to convert. + * @return the converted list. + */ + public AuditEvent convertToAuditEvent(PersistentAuditEvent persistentAuditEvent) { + if (persistentAuditEvent == null) { + return null; + } + return new AuditEvent(persistentAuditEvent.getAuditEventDate(), persistentAuditEvent.getPrincipal(), + persistentAuditEvent.getAuditEventType(), convertDataToObjects(persistentAuditEvent.getData())); + } + + /** + * Internal conversion. This is needed to support the current SpringBoot actuator {@code AuditEventRepository} interface. + * + * @param data the data to convert. + * @return a map of {@link String}, {@link Object}. + */ + public Map convertDataToObjects(Map data) { + Map results = new HashMap<>(); + + if (data != null) { + for (Map.Entry entry : data.entrySet()) { + results.put(entry.getKey(), entry.getValue()); + } + } + return results; + } + + /** + * Internal conversion. This method will allow to save additional data. + * By default, it will save the object as string. + * + * @param data the data to convert. + * @return a map of {@link String}, {@link String}. + */ + public Map convertDataToStrings(Map data) { + Map results = new HashMap<>(); + + if (data != null) { + for (Map.Entry entry : data.entrySet()) { + // Extract the data that will be saved. + if (entry.getValue() instanceof WebAuthenticationDetails) { + WebAuthenticationDetails authenticationDetails = (WebAuthenticationDetails) entry.getValue(); + results.put("remoteAddress", authenticationDetails.getRemoteAddress()); + results.put("sessionId", authenticationDetails.getSessionId()); + } else { + results.put(entry.getKey(), Objects.toString(entry.getValue())); + } + } + } + return results; + } +} diff --git a/src/main/java/it/cnr/isti/epasmed/config/audit/package-info.java b/src/main/java/it/cnr/isti/epasmed/config/audit/package-info.java new file mode 100644 index 0000000..867d38e --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/config/audit/package-info.java @@ -0,0 +1,4 @@ +/** + * Audit specific code. + */ +package it.cnr.isti.epasmed.config.audit; diff --git a/src/main/java/it/cnr/isti/epasmed/config/package-info.java b/src/main/java/it/cnr/isti/epasmed/config/package-info.java new file mode 100644 index 0000000..c5538b7 --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/config/package-info.java @@ -0,0 +1,4 @@ +/** + * Spring Framework configuration files. + */ +package it.cnr.isti.epasmed.config; diff --git a/src/main/java/it/cnr/isti/epasmed/domain/AbstractAuditingEntity.java b/src/main/java/it/cnr/isti/epasmed/domain/AbstractAuditingEntity.java new file mode 100644 index 0000000..2371fc2 --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/domain/AbstractAuditingEntity.java @@ -0,0 +1,77 @@ +package it.cnr.isti.epasmed.domain; + +import com.fasterxml.jackson.annotation.JsonIgnore; +import org.springframework.data.annotation.CreatedBy; +import org.springframework.data.annotation.CreatedDate; +import org.springframework.data.annotation.LastModifiedBy; +import org.springframework.data.annotation.LastModifiedDate; +import org.springframework.data.jpa.domain.support.AuditingEntityListener; + +import java.io.Serializable; +import java.time.Instant; +import javax.persistence.Column; +import javax.persistence.EntityListeners; +import javax.persistence.MappedSuperclass; + +/** + * Base abstract class for entities which will hold definitions for created, last modified, created by, + * last modified by attributes. + */ +@MappedSuperclass +@EntityListeners(AuditingEntityListener.class) +public abstract class AbstractAuditingEntity implements Serializable { + + private static final long serialVersionUID = 1L; + + @CreatedBy + @Column(name = "created_by", nullable = false, length = 50, updatable = false) + @JsonIgnore + private String createdBy; + + @CreatedDate + @Column(name = "created_date", updatable = false) + @JsonIgnore + private Instant createdDate = Instant.now(); + + @LastModifiedBy + @Column(name = "last_modified_by", length = 50) + @JsonIgnore + private String lastModifiedBy; + + @LastModifiedDate + @Column(name = "last_modified_date") + @JsonIgnore + private Instant lastModifiedDate = Instant.now(); + + public String getCreatedBy() { + return createdBy; + } + + public void setCreatedBy(String createdBy) { + this.createdBy = createdBy; + } + + public Instant getCreatedDate() { + return createdDate; + } + + public void setCreatedDate(Instant createdDate) { + this.createdDate = createdDate; + } + + public String getLastModifiedBy() { + return lastModifiedBy; + } + + public void setLastModifiedBy(String lastModifiedBy) { + this.lastModifiedBy = lastModifiedBy; + } + + public Instant getLastModifiedDate() { + return lastModifiedDate; + } + + public void setLastModifiedDate(Instant lastModifiedDate) { + this.lastModifiedDate = lastModifiedDate; + } +} diff --git a/src/main/java/it/cnr/isti/epasmed/domain/Authority.java b/src/main/java/it/cnr/isti/epasmed/domain/Authority.java new file mode 100644 index 0000000..3bc42ae --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/domain/Authority.java @@ -0,0 +1,58 @@ +package it.cnr.isti.epasmed.domain; + +import javax.persistence.Entity; +import javax.persistence.Id; +import javax.persistence.Table; +import javax.persistence.Column; +import javax.validation.constraints.NotNull; +import javax.validation.constraints.Size; +import java.io.Serializable; +import java.util.Objects; + +/** + * An authority (a security role) used by Spring Security. + */ +@Entity +@Table(name = "jhi_authority") +public class Authority implements Serializable { + + private static final long serialVersionUID = 1L; + + @NotNull + @Size(max = 50) + @Id + @Column(length = 50) + private String name; + + public String getName() { + return name; + } + + public void setName(String name) { + this.name = name; + } + + @Override + public boolean equals(Object o) { + if (this == o) { + return true; + } + if (!(o instanceof Authority)) { + return false; + } + return Objects.equals(name, ((Authority) o).name); + } + + @Override + public int hashCode() { + return Objects.hashCode(name); + } + + // prettier-ignore + @Override + public String toString() { + return "Authority{" + + "name='" + name + '\'' + + "}"; + } +} diff --git a/src/main/java/it/cnr/isti/epasmed/domain/PersistentAuditEvent.java b/src/main/java/it/cnr/isti/epasmed/domain/PersistentAuditEvent.java new file mode 100644 index 0000000..bc6f6ec --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/domain/PersistentAuditEvent.java @@ -0,0 +1,108 @@ +package it.cnr.isti.epasmed.domain; + +import javax.persistence.*; +import javax.validation.constraints.NotNull; +import java.io.Serializable; +import java.time.Instant; +import java.util.HashMap; +import java.util.Map; + +/** + * Persist AuditEvent managed by the Spring Boot actuator. + * + * @see org.springframework.boot.actuate.audit.AuditEvent + */ +@Entity +@Table(name = "jhi_persistent_audit_event") +public class PersistentAuditEvent implements Serializable { + + private static final long serialVersionUID = 1L; + + @Id + @GeneratedValue(strategy = GenerationType.SEQUENCE, generator = "persistentAuditEventSequenceGenerator") + @SequenceGenerator(name = "persistentAuditEventSequenceGenerator", allocationSize = 1) + @Column(name = "event_id") + private Long id; + + @NotNull + @Column(nullable = false) + private String principal; + + @Column(name = "event_date") + private Instant auditEventDate; + + @Column(name = "event_type") + private String auditEventType; + + @ElementCollection + @MapKeyColumn(name = "name") + @Column(name = "value") + @CollectionTable(name = "jhi_persistent_audit_evt_data", joinColumns=@JoinColumn(name="event_id")) + private Map data = new HashMap<>(); + + public Long getId() { + return id; + } + + public void setId(Long id) { + this.id = id; + } + + public String getPrincipal() { + return principal; + } + + public void setPrincipal(String principal) { + this.principal = principal; + } + + public Instant getAuditEventDate() { + return auditEventDate; + } + + public void setAuditEventDate(Instant auditEventDate) { + this.auditEventDate = auditEventDate; + } + + public String getAuditEventType() { + return auditEventType; + } + + public void setAuditEventType(String auditEventType) { + this.auditEventType = auditEventType; + } + + public Map getData() { + return data; + } + + public void setData(Map data) { + this.data = data; + } + + @Override + public boolean equals(Object o) { + if (this == o) { + return true; + } + if (!(o instanceof PersistentAuditEvent)) { + return false; + } + return id != null && id.equals(((PersistentAuditEvent) o).id); + } + + @Override + public int hashCode() { + return 31; + } + + // prettier-ignore + @Override + public String toString() { + return "PersistentAuditEvent{" + + "principal='" + principal + '\'' + + ", auditEventDate=" + auditEventDate + + ", auditEventType='" + auditEventType + '\'' + + '}'; + } +} diff --git a/src/main/java/it/cnr/isti/epasmed/domain/PersistentToken.java b/src/main/java/it/cnr/isti/epasmed/domain/PersistentToken.java new file mode 100644 index 0000000..6bc3485 --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/domain/PersistentToken.java @@ -0,0 +1,128 @@ +package it.cnr.isti.epasmed.domain; + +import com.fasterxml.jackson.annotation.JsonIgnore; + +import javax.persistence.*; +import javax.validation.constraints.NotNull; +import javax.validation.constraints.Size; + +import java.io.Serializable; +import java.util.Objects; +import java.time.LocalDate; + +/** + * Persistent tokens are used by Spring Security to automatically log in users. + * + * @see it.cnr.isti.epasmed.security.PersistentTokenRememberMeServices + */ +@Entity +@Table(name = "jhi_persistent_token") +public class PersistentToken implements Serializable { + + private static final long serialVersionUID = 1L; + + private static final int MAX_USER_AGENT_LEN = 255; + + @Id + private String series; + + @JsonIgnore + @NotNull + @Column(name = "token_value", nullable = false) + private String tokenValue; + + @Column(name = "token_date") + private LocalDate tokenDate; + + //an IPV6 address max length is 39 characters + @Size(min = 0, max = 39) + @Column(name = "ip_address", length = 39) + private String ipAddress; + + @Column(name = "user_agent") + private String userAgent; + + @JsonIgnore + @ManyToOne + private User user; + + public String getSeries() { + return series; + } + + public void setSeries(String series) { + this.series = series; + } + + public String getTokenValue() { + return tokenValue; + } + + public void setTokenValue(String tokenValue) { + this.tokenValue = tokenValue; + } + + public LocalDate getTokenDate() { + return tokenDate; + } + + public void setTokenDate(LocalDate tokenDate) { + this.tokenDate = tokenDate; + } + + public String getIpAddress() { + return ipAddress; + } + + public void setIpAddress(String ipAddress) { + this.ipAddress = ipAddress; + } + + public String getUserAgent() { + return userAgent; + } + + public void setUserAgent(String userAgent) { + if (userAgent.length() >= MAX_USER_AGENT_LEN) { + this.userAgent = userAgent.substring(0, MAX_USER_AGENT_LEN - 1); + } else { + this.userAgent = userAgent; + } + } + + public User getUser() { + return user; + } + + public void setUser(User user) { + this.user = user; + } + + @Override + public boolean equals(Object o) { + if (this == o) { + return true; + } + if (!(o instanceof PersistentToken)) { + return false; + } + return Objects.equals(series, ((PersistentToken) o).series); + } + + @Override + public int hashCode() { + return Objects.hashCode(series); + } + + // prettier-ignore + @Override + public String toString() { + return "PersistentToken{" + + "series='" + series + '\'' + + ", tokenValue='" + tokenValue + '\'' + + ", tokenDate=" + tokenDate + + ", ipAddress='" + ipAddress + '\'' + + ", userAgent='" + userAgent + '\'' + + "}"; + } +} diff --git a/src/main/java/it/cnr/isti/epasmed/domain/TabsSI.java b/src/main/java/it/cnr/isti/epasmed/domain/TabsSI.java new file mode 100644 index 0000000..b21cac0 --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/domain/TabsSI.java @@ -0,0 +1,106 @@ +package it.cnr.isti.epasmed.domain; + +import java.io.Serializable; +import java.time.LocalDateTime; + +import javax.persistence.Column; +import javax.persistence.Entity; +import javax.persistence.GeneratedValue; +import javax.persistence.GenerationType; +import javax.persistence.Id; +import javax.persistence.SequenceGenerator; +import javax.persistence.Table; +import javax.validation.constraints.NotNull; + + + +/** + * A user. + */ +@Entity +@Table(name = "tabssi") +public class TabsSI implements Serializable { + + private static final long serialVersionUID = 1L; + + @Id + @GeneratedValue(strategy = GenerationType.SEQUENCE, generator = "tabssiSequenceGenerator") + @SequenceGenerator(name = "tabssiSequenceGenerator", allocationSize = 1) + private Long id; + + @NotNull + @Column(unique = true, nullable = false) + public String nome; + + @NotNull + public String operazioni; + + @NotNull + public Long idFlusso; + + @NotNull + @Column(columnDefinition = "TIMESTAMP" ) + private LocalDateTime lastUpdate; + + public Long getId() { + return id; + } + + public void setId(Long id) { + this.id = id; + } + + public String getNome() { + return nome; + } + + public void setNome(String nome) { + this.nome = nome; + } + + public String getOperazioni() { + return operazioni; + } + + public void setOperazioni(String operazioni) { + this.operazioni = operazioni; + } + + public Long getIdFlusso() { + return idFlusso; + } + + public void setIdFlusso(Long idFlusso) { + this.idFlusso = idFlusso; + } + + public LocalDateTime getLastUpdate() { + return lastUpdate; + } + + public void setLastUpdate(LocalDateTime lastUpdate) { + this.lastUpdate = lastUpdate; + } + + public static long getSerialversionuid() { + return serialVersionUID; + } + + @Override + public boolean equals(Object o) { + if (this == o) { + return true; + } + if (!(o instanceof TabsSI)) { + return false; + } + return id != null && id.equals(((TabsSI) o).id); + } + + @Override + public String toString() { + return "TabsSI [id=" + id + ", nome=" + nome + ", operazioni=" + operazioni + ", idFlusso=" + idFlusso + + ", lastUpdate=" + lastUpdate + "]"; + } + +} diff --git a/src/main/java/it/cnr/isti/epasmed/domain/User.java b/src/main/java/it/cnr/isti/epasmed/domain/User.java new file mode 100644 index 0000000..54b7da2 --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/domain/User.java @@ -0,0 +1,240 @@ +package it.cnr.isti.epasmed.domain; + +import it.cnr.isti.epasmed.config.Constants; + +import com.fasterxml.jackson.annotation.JsonIgnore; +import org.apache.commons.lang3.StringUtils; +import org.hibernate.annotations.BatchSize; + +import javax.persistence.*; +import javax.validation.constraints.Email; +import javax.validation.constraints.NotNull; +import javax.validation.constraints.Pattern; +import javax.validation.constraints.Size; +import java.io.Serializable; +import java.time.Instant; +import java.util.HashSet; +import java.util.Locale; +import java.util.Set; + +/** + * A user. + */ +@Entity +@Table(name = "jhi_user") +public class User extends AbstractAuditingEntity implements Serializable { + + private static final long serialVersionUID = 1L; + + @Id + @GeneratedValue(strategy = GenerationType.SEQUENCE, generator = "userSequenceGenerator") + @SequenceGenerator(name = "userSequenceGenerator", allocationSize = 1) + private Long id; + + @NotNull + @Pattern(regexp = Constants.LOGIN_REGEX) + @Size(min = 1, max = 50) + @Column(length = 50, unique = true, nullable = false) + private String login; + + @JsonIgnore + @NotNull + @Size(min = 60, max = 60) + @Column(name = "password_hash", length = 60, nullable = false) + private String password; + + @Size(max = 50) + @Column(name = "first_name", length = 50) + private String firstName; + + @Size(max = 50) + @Column(name = "last_name", length = 50) + private String lastName; + + @Email + @Size(min = 5, max = 254) + @Column(length = 254, unique = true) + private String email; + + @NotNull + @Column(nullable = false) + private boolean activated = false; + + @Size(min = 2, max = 10) + @Column(name = "lang_key", length = 10) + private String langKey; + + @Size(max = 256) + @Column(name = "image_url", length = 256) + private String imageUrl; + + @Size(max = 20) + @Column(name = "activation_key", length = 20) + @JsonIgnore + private String activationKey; + + @Size(max = 20) + @Column(name = "reset_key", length = 20) + @JsonIgnore + private String resetKey; + + @Column(name = "reset_date") + private Instant resetDate = null; + + @JsonIgnore + @ManyToMany + @JoinTable( + name = "jhi_user_authority", + joinColumns = {@JoinColumn(name = "user_id", referencedColumnName = "id")}, + inverseJoinColumns = {@JoinColumn(name = "authority_name", referencedColumnName = "name")}) + @BatchSize(size = 20) + private Set authorities = new HashSet<>(); + + @JsonIgnore + @OneToMany(cascade = CascadeType.ALL, orphanRemoval = true, mappedBy = "user") + private Set persistentTokens = new HashSet<>(); + + public Long getId() { + return id; + } + + public void setId(Long id) { + this.id = id; + } + + public String getLogin() { + return login; + } + + // Lowercase the login before saving it in database + public void setLogin(String login) { + this.login = StringUtils.lowerCase(login, Locale.ENGLISH); + } + + public String getPassword() { + return password; + } + + public void setPassword(String password) { + this.password = password; + } + + public String getFirstName() { + return firstName; + } + + public void setFirstName(String firstName) { + this.firstName = firstName; + } + + public String getLastName() { + return lastName; + } + + public void setLastName(String lastName) { + this.lastName = lastName; + } + + public String getEmail() { + return email; + } + + public void setEmail(String email) { + this.email = email; + } + + public String getImageUrl() { + return imageUrl; + } + + public void setImageUrl(String imageUrl) { + this.imageUrl = imageUrl; + } + + public boolean getActivated() { + return activated; + } + + public void setActivated(boolean activated) { + this.activated = activated; + } + + public String getActivationKey() { + return activationKey; + } + + public void setActivationKey(String activationKey) { + this.activationKey = activationKey; + } + + public String getResetKey() { + return resetKey; + } + + public void setResetKey(String resetKey) { + this.resetKey = resetKey; + } + + public Instant getResetDate() { + return resetDate; + } + + public void setResetDate(Instant resetDate) { + this.resetDate = resetDate; + } + + public String getLangKey() { + return langKey; + } + + public void setLangKey(String langKey) { + this.langKey = langKey; + } + + public Set getAuthorities() { + return authorities; + } + + public void setAuthorities(Set authorities) { + this.authorities = authorities; + } + + public Set getPersistentTokens() { + return persistentTokens; + } + + public void setPersistentTokens(Set persistentTokens) { + this.persistentTokens = persistentTokens; + } + + @Override + public boolean equals(Object o) { + if (this == o) { + return true; + } + if (!(o instanceof User)) { + return false; + } + return id != null && id.equals(((User) o).id); + } + + @Override + public int hashCode() { + return 31; + } + + // prettier-ignore + @Override + public String toString() { + return "User{" + + "login='" + login + '\'' + + ", firstName='" + firstName + '\'' + + ", lastName='" + lastName + '\'' + + ", email='" + email + '\'' + + ", imageUrl='" + imageUrl + '\'' + + ", activated='" + activated + '\'' + + ", langKey='" + langKey + '\'' + + ", activationKey='" + activationKey + '\'' + + "}"; + } +} diff --git a/src/main/java/it/cnr/isti/epasmed/domain/package-info.java b/src/main/java/it/cnr/isti/epasmed/domain/package-info.java new file mode 100644 index 0000000..0647e34 --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/domain/package-info.java @@ -0,0 +1,4 @@ +/** + * JPA domain objects. + */ +package it.cnr.isti.epasmed.domain; diff --git a/src/main/java/it/cnr/isti/epasmed/epas/client/EPASAffiliationsCient.java b/src/main/java/it/cnr/isti/epasmed/epas/client/EPASAffiliationsCient.java new file mode 100644 index 0000000..e43736b --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/epas/client/EPASAffiliationsCient.java @@ -0,0 +1,108 @@ +package it.cnr.isti.epasmed.epas.client; + +import java.util.HashMap; +import java.util.List; +import java.util.Map; + +import javax.validation.Valid; + +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.beans.factory.annotation.Qualifier; +import org.springframework.core.ParameterizedTypeReference; +import org.springframework.http.HttpMethod; +import org.springframework.http.ResponseEntity; +import org.springframework.stereotype.Component; +import org.springframework.web.client.RestTemplate; + +import it.cnr.isti.epasmed.config.ApplicationProperties; +import it.cnr.isti.epasmed.epas.dto.EPASAffiliationsDTO; +import it.cnr.isti.epasmed.epas.model.EPASAffiliations; + +@Component +public class EPASAffiliationsCient { + + private final Logger log = LoggerFactory.getLogger(getClass()); + + @Autowired + @Qualifier("RestTemplateForFirstUser") + RestTemplate rt; + + @Autowired + ApplicationProperties appProps; + + public EPASAffiliations getById(String id) { + EPASAffiliations epasAffiliations = rt.getForObject( + appProps.getDatasourceEpasRest().getRestUrl() + "/v2/affiliations/show?id={id}", EPASAffiliations.class, + id); + log.info("Retrieved Affiliations: {}", epasAffiliations); + return epasAffiliations; + } + + public List getByGroup(String groupId, boolean includeInactive) { + ResponseEntity> responseEntity = rt.exchange( + appProps.getDatasourceEpasRest().getRestUrl() + + "/v2/affiliations/byGroup?id={groupId}&includeInactive={includeInactive}", + HttpMethod.GET, null, new ParameterizedTypeReference>() { + }, groupId,includeInactive); + List listEPASAffiliations = responseEntity.getBody(); + log.info("Retrieved Affiliations byGroup: {}", listEPASAffiliations); + return listEPASAffiliations; + } + + public List getByPersonId(String personId) { + ResponseEntity> responseEntity = rt.exchange( + appProps.getDatasourceEpasRest().getRestUrl() + + "/v2/affiliations/byPerson?id={personId}", + HttpMethod.GET, null, new ParameterizedTypeReference>() { + }, personId); + List listEPASAffiliations = responseEntity.getBody(); + log.info("Retrieved Affiliations by Person id: {}", listEPASAffiliations); + return listEPASAffiliations; + } + + public List getByPersonFiscalcode(String fc) { + ResponseEntity> responseEntity = rt.exchange( + appProps.getDatasourceEpasRest().getRestUrl() + + "/v2/affiliations/byPerson?fiscalCode={fc}", + HttpMethod.GET, null, new ParameterizedTypeReference>() { + }, fc); + List listEPASAffiliations = responseEntity.getBody(); + log.info("Retrieved Affiliations by Person fiscalcode: {}", listEPASAffiliations); + return listEPASAffiliations; + } + + + + public EPASAffiliations create(EPASAffiliationsDTO epasAffiliationsDTO) { + ResponseEntity response = rt.postForEntity( + appProps.getDatasourceEpasRest().getRestUrl() + "/v2/affiliations/create", epasAffiliationsDTO, + EPASAffiliations.class); + + EPASAffiliations createdEPASAffiliations = response.getBody(); + + log.info("Created Affiliations: {}", createdEPASAffiliations); + return createdEPASAffiliations; + } + + public void updateById(String id, @Valid EPASAffiliationsDTO epasAffiliationsDTO) { + Map uriVariables = new HashMap<>(); + uriVariables.put("id", id); + + rt.put(appProps.getDatasourceEpasRest().getRestUrl() + "/v2/affiliations/update?id={id}", epasAffiliationsDTO, + uriVariables); + + log.info("Updated Affiliations id: {}", id); + return; + } + + public void deleteById(String id) { + rt.delete(appProps.getDatasourceEpasRest().getRestUrl() + "/v2/affiliations/delete?id={id}", id); + log.info("Deleted Group id: {}", id); + return; + } + + + +} diff --git a/src/main/java/it/cnr/isti/epasmed/epas/client/EPASContractsCient.java b/src/main/java/it/cnr/isti/epasmed/epas/client/EPASContractsCient.java new file mode 100644 index 0000000..9619211 --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/epas/client/EPASContractsCient.java @@ -0,0 +1,117 @@ +package it.cnr.isti.epasmed.epas.client; + +import java.util.HashMap; +import java.util.List; +import java.util.Map; + +import javax.validation.Valid; + +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.beans.factory.annotation.Qualifier; +import org.springframework.core.ParameterizedTypeReference; +import org.springframework.http.HttpMethod; +import org.springframework.http.ResponseEntity; +import org.springframework.stereotype.Component; +import org.springframework.web.client.RestTemplate; + +import it.cnr.isti.epasmed.config.ApplicationProperties; +import it.cnr.isti.epasmed.epas.dto.EPASContractsDTO; +import it.cnr.isti.epasmed.epas.model.EPASContracts; + +@Component +public class EPASContractsCient { + + private final Logger log = LoggerFactory.getLogger(getClass()); + + @Autowired + @Qualifier("RestTemplateForFirstUser") + RestTemplate rt; + + @Autowired + ApplicationProperties appProps; + + public EPASContracts getById(String id) { + EPASContracts epasContracts = rt.getForObject( + appProps.getDatasourceEpasRest().getRestUrl() + "/v2/contracts/show/{id}", EPASContracts.class, id); + log.info("Retrieved Contracts: {}", epasContracts); + return epasContracts; + } + + public List getByPersonId(String id) { + ResponseEntity> responseEntity = rt.exchange( + appProps.getDatasourceEpasRest().getRestUrl() + "/v2/contracts/byPerson?id={id}", HttpMethod.GET, null, + new ParameterizedTypeReference>() { + }, id); + List listEPASContracts = responseEntity.getBody(); + log.info("Retrieved Contracts: {}", listEPASContracts); + return listEPASContracts; + + } + + public List getByPersonFiscalcode(String fc) { + ResponseEntity> responseEntity = rt.exchange( + appProps.getDatasourceEpasRest().getRestUrl() + "/v2/contracts/byPerson?fiscalCode={fc}", HttpMethod.GET, + null, new ParameterizedTypeReference>() { + }, fc); + List listEPASContracts = responseEntity.getBody(); + log.info("Retrieved Contracts: {}", listEPASContracts); + return listEPASContracts; + } + + public EPASContracts create(EPASContractsDTO epasContractsDTO) { + ResponseEntity response = rt.postForEntity( + appProps.getDatasourceEpasRest().getRestUrl() + "/v2/contracts/create", epasContractsDTO, + EPASContracts.class); + + EPASContracts createdEPASContracts = response.getBody(); + + log.info("Created Contracts: {}", createdEPASContracts); + return createdEPASContracts; + } + + public void updateById(String id, @Valid EPASContractsDTO epasContractsDTO) { + Map uriVariables = new HashMap<>(); + uriVariables.put("id", id); + + rt.put(appProps.getDatasourceEpasRest().getRestUrl() + "/v2/contracts/update?id={id}", epasContractsDTO, + uriVariables); + + log.info("Updated Contracts id: {}", id); + return; + } + + // Continuazione di due contratti consecutivi + // E' possibile impostare che un contratto è continuativo rispetto al precedente + // e ne eredità le ferie non godute precedenti. Questa funzionalità è utilizzare + // per esempio per alcune stabilizzazioni dove il dipendente si porta dietro dal + // precedente contratto la situazione delle ferie non godute. Per impostare e + // rimuovere che un contratto è continuativo rispetto al precedente è possibile + // utilizzare i metodi + /// rest/v2/contract/setPreviousContract + /// rest/v2/contract/unsetPreviousContract + // $ http -a isti_registry_manager DELETE + // https://epas-demo.devel.iit.cnr.it/rest/v2/contract/setPreviousContract?id=4678 + // $ http -a isti_registry_manager DELETE + // https://epas-demo.devel.iit.cnr.it/rest/v2/contract/unsetPreviousContract?id=4678 + + public void setPreviousContract(String id) { + rt.delete(appProps.getDatasourceEpasRest().getRestUrl() + "/v2/contracts/setPreviousContract?id={id}", id); + log.info("Set Previous Contract for Contract id: {}", id); + return; + } + + public void setUnsetPreviousContract(String id) { + rt.delete(appProps.getDatasourceEpasRest().getRestUrl() + "/v2/contracts/unsetPreviousContract?id={id}", id); + log.info("Set Unset Previous Contract for Contract id: {}", id); + return; + } + + public void deleteById(String id) { + rt.delete(appProps.getDatasourceEpasRest().getRestUrl() + "/v2/contracts/delete?id={id}", id); + log.info("Deleted Contract id: {}", id); + return; + } + +} diff --git a/src/main/java/it/cnr/isti/epasmed/epas/client/EPASGroupsCient.java b/src/main/java/it/cnr/isti/epasmed/epas/client/EPASGroupsCient.java new file mode 100644 index 0000000..d9ee272 --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/epas/client/EPASGroupsCient.java @@ -0,0 +1,83 @@ +package it.cnr.isti.epasmed.epas.client; + +import java.util.HashMap; +import java.util.List; +import java.util.Map; + +import javax.validation.Valid; + +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.beans.factory.annotation.Qualifier; +import org.springframework.core.ParameterizedTypeReference; +import org.springframework.http.HttpMethod; +import org.springframework.http.ResponseEntity; +import org.springframework.stereotype.Component; +import org.springframework.web.client.RestTemplate; + +import it.cnr.isti.epasmed.config.ApplicationProperties; +import it.cnr.isti.epasmed.epas.dto.EPASGroupsDTO; +import it.cnr.isti.epasmed.epas.model.EPASGroups; + +@Component +public class EPASGroupsCient { + + private final Logger log = LoggerFactory.getLogger(getClass()); + + @Autowired + @Qualifier("RestTemplateForFirstUser") + RestTemplate rt; + + @Autowired + ApplicationProperties appProps; + + public EPASGroups getById(String id) { + EPASGroups epasGroups = rt.getForObject( + appProps.getDatasourceEpasRest().getRestUrl() + "/v2/groups/show?id={id}", EPASGroups.class, id); + log.info("Retrieved Groups: {}", epasGroups); + return epasGroups; + } + + public List getList(String officeId) { + ResponseEntity> responseEntity = rt.exchange( + appProps.getDatasourceEpasRest().getRestUrl() + "/v2/groups/list?id={officeId}", HttpMethod.GET, null, + new ParameterizedTypeReference>() { + }, officeId); + List listEPASGroups = responseEntity.getBody(); + log.info("Retrieved Groups: {}", listEPASGroups); + return listEPASGroups; + } + + public EPASGroups create(EPASGroupsDTO epasGroupsDTO) { + ResponseEntity response = rt.postForEntity( + appProps.getDatasourceEpasRest().getRestUrl() + "/v2/groups/create", epasGroupsDTO, EPASGroups.class); + + EPASGroups createdEPASGroups = response.getBody(); + + log.info("Created Groups: {}", createdEPASGroups); + return createdEPASGroups; + } + + public void updateById(String id, @Valid EPASGroupsDTO epasGroupsDTO) { + Map uriVariables = new HashMap<>(); + uriVariables.put("id", id); + + rt.put(appProps.getDatasourceEpasRest().getRestUrl() + "/v2/groups/update?id={id}", epasGroupsDTO, uriVariables); + + log.info("Updated Groups id: {}", id); + return; + } + + public void deleteById(String id) { + rt.delete(appProps.getDatasourceEpasRest().getRestUrl() + "/v2/groups/delete?id={id}", id); + log.info("Deleted Group id: {}", id); + return; + } + + // $ http -a isti_registry_manager POST + // https://epas-demo.devel.iit.cnr.it/rest/v2/groups/create + // name="Gruppo Test" description="Gruppo ISTI di test" + // officeId=201 managerId=4222 externalId="pimpaExternalId" + +} diff --git a/src/main/java/it/cnr/isti/epasmed/epas/client/EPASPersonsCient.java b/src/main/java/it/cnr/isti/epasmed/epas/client/EPASPersonsCient.java new file mode 100644 index 0000000..e8279be --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/epas/client/EPASPersonsCient.java @@ -0,0 +1,127 @@ +package it.cnr.isti.epasmed.epas.client; + +import java.util.HashMap; +import java.util.List; +import java.util.Map; + +import javax.validation.Valid; + +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.beans.factory.annotation.Qualifier; +import org.springframework.core.ParameterizedTypeReference; +import org.springframework.http.HttpMethod; +import org.springframework.http.ResponseEntity; +import org.springframework.stereotype.Component; +import org.springframework.web.client.RestTemplate; + +import it.cnr.isti.epasmed.config.ApplicationProperties; +import it.cnr.isti.epasmed.epas.dto.EPASPersonsDTO; +import it.cnr.isti.epasmed.epas.model.EPASPersons; + +@Component +public class EPASPersonsCient { + + private final Logger log = LoggerFactory.getLogger(getClass()); + + @Autowired + @Qualifier("RestTemplateForFirstUser") + RestTemplate rt; + + @Autowired + ApplicationProperties appProps; + + public EPASPersons getById(String id) { + EPASPersons epasPersons = rt.getForObject( + appProps.getDatasourceEpasRest().getRestUrl() + "/v2/persons/show/{id}", EPASPersons.class, id); + log.info("Retrieved Persons: {}", epasPersons); + return epasPersons; + } + + public EPASPersons getByFiscalCode(String fc) { + EPASPersons epasPersons = rt.getForObject( + appProps.getDatasourceEpasRest().getRestUrl() + "/v2/persons/show?fiscalCode={fc}", EPASPersons.class, fc); + log.info("Retrieved Persons: {}", epasPersons); + return epasPersons; + } + + public List getList(String officeId) { + ResponseEntity> responseEntity = rt.exchange( + appProps.getDatasourceEpasRest().getRestUrl() + "/v2/persons/list?id={officeId}", HttpMethod.GET, null, + new ParameterizedTypeReference>() { + }, officeId); + List listEPASPersons = responseEntity.getBody(); + log.info("Retrieved Persons: {}", listEPASPersons); + return listEPASPersons; + } + + public EPASPersons create(EPASPersonsDTO epasPersonsDTO) { + ResponseEntity response = rt.postForEntity( + appProps.getDatasourceEpasRest().getRestUrl() + "/v2/persons/create", epasPersonsDTO, EPASPersons.class); + + EPASPersons createdEPASPersons = response.getBody(); + + log.info("Created Persons: {}", createdEPASPersons); + return createdEPASPersons; + } + + public void updateById(String id, @Valid EPASPersonsDTO epasPersonsDTO) { + Map uriVariables = new HashMap<>(); + uriVariables.put("id", id); + + rt.put(appProps.getDatasourceEpasRest().getRestUrl() + "/v2/persons/update?id={id}", epasPersonsDTO, uriVariables); + + log.info("Updated Persons: {}", id); + return; + } + + public void updateByFiscalCode(String fc, EPASPersonsDTO epasPersonsDTO) { + Map uriVariables = new HashMap<>(); + uriVariables.put("fiscalCode", fc); + + rt.put(appProps.getDatasourceEpasRest().getRestUrl() + "/v2/persons/update?fiscalCode={fiscalCode}", + epasPersonsDTO, uriVariables); + + log.info("Updated Person: {}", fc); + return; + } + + public void deleteById(String id) { + rt.delete(appProps.getDatasourceEpasRest().getRestUrl() + "/v2/persons/delete?id={id}", id); + log.info("Deleted Person with id: {}", id); + return; + } + + public void deleteByFiscalCode(String fc) { + rt.delete(appProps.getDatasourceEpasRest().getRestUrl() + "/v2/persons/delete?fiscalCode={fc}", fc); + log.info("Delete Persons with fiscalcode: {}", fc); + return; + } + + /* + * HttpHeaders headers = new HttpHeaders(); + * headers.setContentType(MediaType.APPLICATION_FORM_URLENCODED); + * + * MultiValueMap map= new LinkedMultiValueMap(); + * map.add("name", epasPersons.getName()); map.add("surname", + * epasPersons.getSurname()); map.add("fiscalCode", + * epasPersons.getFiscalCode()); map.add("email", epasPersons.getEmail()); + * map.add("qualification", epasPersons.getQualifcation()); map.add("officeId", + * epasPersons.getOffice().getId()); + * + * HttpEntity> request = new + * HttpEntity>(map, headers); + * + * ResponseEntity response = rt.postForEntity( + * appProps.getDatasourceEpasRest().getRestUrl() + "/persons/create", request , + * EPASPersons.class ); + */ + // ResponseEntity response = rt.postForEntity( + // appProps.getDatasourceEpasRest().getRestUrl() + // + "/persons/create?name={name}&surname={surname}&email={email}" + // + + // "&qualification={qualification}&officeId={officeId}&fiscalCode={fiscalCode}", + // null, EPASPersons.class, uriVariables); + +} diff --git a/src/main/java/it/cnr/isti/epasmed/epas/client/EPASWorkinTipeTypesCient.java b/src/main/java/it/cnr/isti/epasmed/epas/client/EPASWorkinTipeTypesCient.java new file mode 100644 index 0000000..358d46e --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/epas/client/EPASWorkinTipeTypesCient.java @@ -0,0 +1,48 @@ +package it.cnr.isti.epasmed.epas.client; + +import java.util.List; + +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.beans.factory.annotation.Qualifier; +import org.springframework.core.ParameterizedTypeReference; +import org.springframework.http.HttpMethod; +import org.springframework.http.ResponseEntity; +import org.springframework.stereotype.Component; +import org.springframework.web.client.RestTemplate; + +import it.cnr.isti.epasmed.config.ApplicationProperties; +import it.cnr.isti.epasmed.epas.model.EPASWorkingTimeTypes; + +@Component +public class EPASWorkinTipeTypesCient { + + private final Logger log = LoggerFactory.getLogger(getClass()); + + @Autowired + @Qualifier("RestTemplateForFirstUser") + RestTemplate rt; + + @Autowired + ApplicationProperties appProps; + + public EPASWorkingTimeTypes getById(String id) { + EPASWorkingTimeTypes epasWorkingTimeTypes = rt.getForObject( + appProps.getDatasourceEpasRest().getRestUrl() + "/v2/workingtimetypes/show?id={id}", + EPASWorkingTimeTypes.class, id); + log.info("Retrieved Working Time Types : {}", epasWorkingTimeTypes); + return epasWorkingTimeTypes; + } + + public List getList(String officeId) { + ResponseEntity> responseEntity = rt.exchange( + appProps.getDatasourceEpasRest().getRestUrl() + "/v2/workingtimetypes/list?id={officeId}", HttpMethod.GET, + null, new ParameterizedTypeReference>() { + }, officeId); + List listEPASWorkingTimeTypes = responseEntity.getBody(); + log.info("Retrieved Retrieved Working Time Types: {}", listEPASWorkingTimeTypes); + return listEPASWorkingTimeTypes; + } + +} diff --git a/src/main/java/it/cnr/isti/epasmed/epas/dto/EPASAffiliationsDTO.java b/src/main/java/it/cnr/isti/epasmed/epas/dto/EPASAffiliationsDTO.java new file mode 100644 index 0000000..036ff8c --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/epas/dto/EPASAffiliationsDTO.java @@ -0,0 +1,25 @@ +package it.cnr.isti.epasmed.epas.dto; + +import java.io.Serializable; + +import lombok.Data; +import lombok.AllArgsConstructor; +import lombok.NoArgsConstructor; + +@Data +@AllArgsConstructor +@NoArgsConstructor +public class EPASAffiliationsDTO implements Serializable { + + private static final long serialVersionUID = 1L; + + private String beginDate; + private String endDate; + private String groupId; + private String id; + private String percentage; + private String personId; + private String externalId; + +} + diff --git a/src/main/java/it/cnr/isti/epasmed/epas/dto/EPASContractsDTO.java b/src/main/java/it/cnr/isti/epasmed/epas/dto/EPASContractsDTO.java new file mode 100644 index 0000000..cfad0d4 --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/epas/dto/EPASContractsDTO.java @@ -0,0 +1,26 @@ +package it.cnr.isti.epasmed.epas.dto; + +import java.io.Serializable; + +import lombok.AllArgsConstructor; +import lombok.Data; +import lombok.NoArgsConstructor; + +@Data +@AllArgsConstructor +@NoArgsConstructor +public class EPASContractsDTO implements Serializable { + + + private static final long serialVersionUID = 1L; + + private String beginDate; + private String endContract; + private String endDate; + private String externalId; + private String id; + private boolean onCertificate; + private String personId; + private String previousContract; + +} \ No newline at end of file diff --git a/src/main/java/it/cnr/isti/epasmed/epas/dto/EPASGroupsDTO.java b/src/main/java/it/cnr/isti/epasmed/epas/dto/EPASGroupsDTO.java new file mode 100644 index 0000000..447c75d --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/epas/dto/EPASGroupsDTO.java @@ -0,0 +1,24 @@ +package it.cnr.isti.epasmed.epas.dto; + +import java.io.Serializable; + +import lombok.Data; +import lombok.AllArgsConstructor; +import lombok.NoArgsConstructor; + +@Data +@AllArgsConstructor +@NoArgsConstructor +public class EPASGroupsDTO implements Serializable { + + private static final long serialVersionUID = 1L; + + private String description; + private String endDate; + private String externalId; + private String id; + private String managerId; + private String name; + private String officeId; + +} \ No newline at end of file diff --git a/src/main/java/it/cnr/isti/epasmed/epas/dto/EPASPersonsDTO.java b/src/main/java/it/cnr/isti/epasmed/epas/dto/EPASPersonsDTO.java new file mode 100644 index 0000000..246440f --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/epas/dto/EPASPersonsDTO.java @@ -0,0 +1,31 @@ +package it.cnr.isti.epasmed.epas.dto; + +import java.io.Serializable; + +import lombok.AllArgsConstructor; +import lombok.Data; +import lombok.NoArgsConstructor; + +@Data +@AllArgsConstructor +@NoArgsConstructor +public class EPASPersonsDTO implements Serializable { + + private static final long serialVersionUID = 1L; + + private String id; + private String name; + private String surname; + private String othersSurnames; + private String telephone; + private String fax; + private String mobile; + private String qualification; + private String[] badges; + private String officeId; + private String fullname; + private String fiscalCode; + private String email; + private String number; + private String eppn; + } \ No newline at end of file diff --git a/src/main/java/it/cnr/isti/epasmed/epas/mapper/EPASAffiliationsMapper.java b/src/main/java/it/cnr/isti/epasmed/epas/mapper/EPASAffiliationsMapper.java new file mode 100644 index 0000000..c8d304d --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/epas/mapper/EPASAffiliationsMapper.java @@ -0,0 +1,47 @@ +package it.cnr.isti.epasmed.epas.mapper; + +import java.util.List; +import java.util.Objects; +import java.util.stream.Collectors; + +import org.springframework.stereotype.Service; + +import it.cnr.isti.epasmed.epas.dto.EPASAffiliationsDTO; +import it.cnr.isti.epasmed.epas.model.EPASAffiliations; + +/** + * Mapper for the entity {@link EPASAffiliations} and its DTO called + * {@link EPASAffiliationsDTO}. + * + * Normal mappers are generated using MapStruct, this one is hand-coded as + * MapStruct support is still in beta, and requires a manual step with an IDE. + */ +@Service +public class EPASAffiliationsMapper { + + public List epasAffiliationsToEPASAffiliationsDTOs( + List epasAffiliationsList) { + return epasAffiliationsList.stream().filter(Objects::nonNull).map(this::epasAffiliationsToEPASAffiliationsDTO) + .collect(Collectors.toList()); + } + + public EPASAffiliationsDTO epasAffiliationsToEPASAffiliationsDTO(EPASAffiliations epasAffiliations) { + if (epasAffiliations == null) { + return null; + } else { + EPASAffiliationsDTO epasAffiliationsDTO = new EPASAffiliationsDTO(); + epasAffiliationsDTO.setId(epasAffiliations.getId()); + epasAffiliationsDTO.setExternalId(epasAffiliations.getExternalId()); + epasAffiliationsDTO.setBeginDate(epasAffiliations.getBeginDate()); + epasAffiliationsDTO.setEndDate(epasAffiliations.getEndDate()); + if (epasAffiliations.getGroup() != null) { + epasAffiliationsDTO.setGroupId(epasAffiliations.getGroup().getId()); + } + if (epasAffiliations.getPerson() != null) { + epasAffiliationsDTO.setPersonId(epasAffiliations.getPerson().getId()); + } + epasAffiliationsDTO.setPercentage(epasAffiliations.getPercentage()); + return epasAffiliationsDTO; + } + } +} diff --git a/src/main/java/it/cnr/isti/epasmed/epas/mapper/EPASContractsMapper.java b/src/main/java/it/cnr/isti/epasmed/epas/mapper/EPASContractsMapper.java new file mode 100644 index 0000000..2aa9d80 --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/epas/mapper/EPASContractsMapper.java @@ -0,0 +1,45 @@ +package it.cnr.isti.epasmed.epas.mapper; + +import java.util.List; +import java.util.Objects; +import java.util.stream.Collectors; + +import org.springframework.stereotype.Service; + +import it.cnr.isti.epasmed.epas.dto.EPASContractsDTO; +import it.cnr.isti.epasmed.epas.model.EPASContracts; + +/** + * Mapper for the entity {@link EPASContracts} and its DTO called + * {@link EPASContractsDTO}. + * + * Normal mappers are generated using MapStruct, this one is hand-coded as + * MapStruct support is still in beta, and requires a manual step with an IDE. + */ +@Service +public class EPASContractsMapper { + + public List epasContractsToEPASContractsDTOs(List epasContractsList) { + return epasContractsList.stream().filter(Objects::nonNull).map(this::epasContractsToEPASContractsDTO) + .collect(Collectors.toList()); + } + + public EPASContractsDTO epasContractsToEPASContractsDTO(EPASContracts epasContracts) { + if (epasContracts == null) { + return null; + } else { + EPASContractsDTO epasContractsDTO = new EPASContractsDTO(); + epasContractsDTO.setId(epasContracts.getId()); + epasContractsDTO.setExternalId(epasContracts.getExternalId()); + epasContractsDTO.setBeginDate(epasContracts.getBeginDate()); + epasContractsDTO.setEndDate(epasContracts.getEndDate()); + epasContractsDTO.setEndContract(epasContracts.getEndContract()); + epasContractsDTO.setOnCertificate(epasContracts.isOnCertificate()); + if (epasContracts.getPerson() != null) { + epasContractsDTO.setPersonId(epasContracts.getPerson().getId()); + } + epasContractsDTO.setPreviousContract(epasContracts.getPreviousContract()); + return epasContractsDTO; + } + } +} diff --git a/src/main/java/it/cnr/isti/epasmed/epas/mapper/EPASGroupsMapper.java b/src/main/java/it/cnr/isti/epasmed/epas/mapper/EPASGroupsMapper.java new file mode 100644 index 0000000..6030c6a --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/epas/mapper/EPASGroupsMapper.java @@ -0,0 +1,47 @@ +package it.cnr.isti.epasmed.epas.mapper; + +import java.util.List; +import java.util.Objects; +import java.util.stream.Collectors; + +import org.springframework.stereotype.Service; + +import it.cnr.isti.epasmed.epas.dto.EPASGroupsDTO; +import it.cnr.isti.epasmed.epas.model.EPASGroups; + +/** + * Mapper for the entity {@link EPASGroups} and its DTO called + * {@link EPASGroupsDTO}. + * + * Normal mappers are generated using MapStruct, this one is hand-coded as + * MapStruct support is still in beta, and requires a manual step with an IDE. + */ +@Service +public class EPASGroupsMapper { + + public List epasGroupsToEPASGroupsDTOs(List epasGroupsList) { + return epasGroupsList.stream().filter(Objects::nonNull).map(this::epasGroupsToEPASGroupsDTO) + .collect(Collectors.toList()); + } + + public EPASGroupsDTO epasGroupsToEPASGroupsDTO(EPASGroups epasGroups) { + if (epasGroups == null) { + return null; + } else { + EPASGroupsDTO epasGroupsDTO = new EPASGroupsDTO(); + epasGroupsDTO.setId(epasGroups.getId()); + epasGroupsDTO.setName(epasGroups.getName()); + epasGroupsDTO.setDescription(epasGroups.getDescription()); + epasGroupsDTO.setEndDate(epasGroups.getEndDate()); + epasGroupsDTO.setExternalId(epasGroups.getExternalId()); + if (epasGroups.getManager() != null) { + epasGroupsDTO.setManagerId(epasGroups.getManager().getId()); + } + if (epasGroups.getOffice() != null) { + epasGroupsDTO.setOfficeId(epasGroups.getOffice().getId()); + } + + return epasGroupsDTO; + } + } +} diff --git a/src/main/java/it/cnr/isti/epasmed/epas/mapper/EPASPersonsMapper.java b/src/main/java/it/cnr/isti/epasmed/epas/mapper/EPASPersonsMapper.java new file mode 100644 index 0000000..3261729 --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/epas/mapper/EPASPersonsMapper.java @@ -0,0 +1,51 @@ +package it.cnr.isti.epasmed.epas.mapper; + +import java.util.List; +import java.util.Objects; +import java.util.stream.Collectors; + +import org.springframework.stereotype.Service; + +import it.cnr.isti.epasmed.epas.dto.EPASPersonsDTO; +import it.cnr.isti.epasmed.epas.model.EPASPersons; + +/** + * Mapper for the entity {@link EPASPersons} and its DTO called + * {@link EPASPersonsDTO}. + * + * Normal mappers are generated using MapStruct, this one is hand-coded as + * MapStruct support is still in beta, and requires a manual step with an IDE. + */ +@Service +public class EPASPersonsMapper { + + public List epasPersonsToEPASPersonsDTOs(List epasPersonsList) { + return epasPersonsList.stream().filter(Objects::nonNull).map(this::epasPersonsToEPASPersonsDTO) + .collect(Collectors.toList()); + } + + public EPASPersonsDTO epasPersonsToEPASPersonsDTO(EPASPersons epasPersons) { + if (epasPersons == null) { + return null; + } else { + EPASPersonsDTO epasPersonsDTO = new EPASPersonsDTO(); + epasPersonsDTO.setId(epasPersons.getId()); + epasPersonsDTO.setName(epasPersons.getName()); + epasPersonsDTO.setSurname(epasPersons.getSurname()); + epasPersonsDTO.setOthersSurnames(epasPersons.getOthersSurnames()); + epasPersonsDTO.setTelephone(epasPersons.getTelephone()); + epasPersonsDTO.setFax(epasPersons.getFax()); + epasPersonsDTO.setQualification(epasPersons.getQualification()); + epasPersonsDTO.setBadges(epasPersons.getBadges()); + if (epasPersons.getOffice() != null) { + epasPersonsDTO.setOfficeId(epasPersons.getOffice().getId()); + } + epasPersonsDTO.setFullname(epasPersons.getFullname()); + epasPersonsDTO.setFiscalCode(epasPersons.getFiscalCode()); + epasPersonsDTO.setEmail(epasPersons.getEmail()); + epasPersonsDTO.setNumber(epasPersons.getNumber()); + epasPersonsDTO.setEppn(epasPersons.getEppn()); + return epasPersonsDTO; + } + } +} diff --git a/src/main/java/it/cnr/isti/epasmed/epas/model/EPASAffiliations.java b/src/main/java/it/cnr/isti/epasmed/epas/model/EPASAffiliations.java new file mode 100644 index 0000000..a32ee98 --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/epas/model/EPASAffiliations.java @@ -0,0 +1,26 @@ +package it.cnr.isti.epasmed.epas.model; + +import java.io.Serializable; + +import lombok.Data; +import lombok.AllArgsConstructor; +import lombok.NoArgsConstructor; + +@Data +@AllArgsConstructor +@NoArgsConstructor +public class EPASAffiliations implements Serializable { + + private static final long serialVersionUID = 1L; + + private String beginDate; + private String endDate; + private EPASGroups group; + private String id; + private String percentage; + private EPASPersons person; + private String externalId; + private String updatedAt; + +} + diff --git a/src/main/java/it/cnr/isti/epasmed/epas/model/EPASContracts.java b/src/main/java/it/cnr/isti/epasmed/epas/model/EPASContracts.java new file mode 100644 index 0000000..7f368f9 --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/epas/model/EPASContracts.java @@ -0,0 +1,28 @@ +package it.cnr.isti.epasmed.epas.model; + +import java.io.Serializable; + +import lombok.Data; +import lombok.AllArgsConstructor; +import lombok.NoArgsConstructor; + +@Data +@AllArgsConstructor +@NoArgsConstructor +public class EPASContracts implements Serializable { + + private static final long serialVersionUID = 1L; + + private String beginDate; + private String endContract; + private String endDate; + private String externalId; + private String id; + private boolean onCertificate; + private EPASPersons person; + private String previousContract; + private EPASWorkingTimeForPerson[] workingTimeTypes; + private String updatedAt; + +} + diff --git a/src/main/java/it/cnr/isti/epasmed/epas/model/EPASGroups.java b/src/main/java/it/cnr/isti/epasmed/epas/model/EPASGroups.java new file mode 100644 index 0000000..4f6439b --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/epas/model/EPASGroups.java @@ -0,0 +1,25 @@ +package it.cnr.isti.epasmed.epas.model; + +import java.io.Serializable; + +import lombok.Data; +import lombok.AllArgsConstructor; +import lombok.NoArgsConstructor; + +@Data +@AllArgsConstructor +@NoArgsConstructor +public class EPASGroups implements Serializable { + + private static final long serialVersionUID = 1L; + + private String description; + private String endDate; + private String externalId; + private String id; + private EPASGroupsManager manager; + private String name; + private EPASOffice office; + private EPASPersons[] people; + private String updatedAt; +} \ No newline at end of file diff --git a/src/main/java/it/cnr/isti/epasmed/epas/model/EPASGroupsManager.java b/src/main/java/it/cnr/isti/epasmed/epas/model/EPASGroupsManager.java new file mode 100644 index 0000000..6249e89 --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/epas/model/EPASGroupsManager.java @@ -0,0 +1,24 @@ +package it.cnr.isti.epasmed.epas.model; + +import java.io.Serializable; + +import lombok.AllArgsConstructor; +import lombok.Data; +import lombok.NoArgsConstructor; + + +@Data +@AllArgsConstructor +@NoArgsConstructor +public class EPASGroupsManager implements Serializable { + + private static final long serialVersionUID = 1L; + + private String email; + private String eppn; + private String fiscalCode; + private String fullname; + private String id; + private String number; + +} diff --git a/src/main/java/it/cnr/isti/epasmed/epas/model/EPASOffice.java b/src/main/java/it/cnr/isti/epasmed/epas/model/EPASOffice.java new file mode 100644 index 0000000..e1171e4 --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/epas/model/EPASOffice.java @@ -0,0 +1,22 @@ +package it.cnr.isti.epasmed.epas.model; + +import java.io.Serializable; + +import lombok.AllArgsConstructor; +import lombok.Data; +import lombok.NoArgsConstructor; + + +@Data +@AllArgsConstructor +@NoArgsConstructor +public class EPASOffice implements Serializable { + + private static final long serialVersionUID = 1L; + + private String id; + private String name; + private String code; + private String codeId; + +} diff --git a/src/main/java/it/cnr/isti/epasmed/epas/model/EPASPersons.java b/src/main/java/it/cnr/isti/epasmed/epas/model/EPASPersons.java new file mode 100644 index 0000000..747b445 --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/epas/model/EPASPersons.java @@ -0,0 +1,31 @@ +package it.cnr.isti.epasmed.epas.model; + +import java.io.Serializable; + +import lombok.AllArgsConstructor; +import lombok.Data; +import lombok.NoArgsConstructor; + +@Data +@AllArgsConstructor +@NoArgsConstructor +public class EPASPersons implements Serializable { + + private static final long serialVersionUID = 1L; + + private String id; + private String name; + private String surname; + private String othersSurnames; + private String telephone; + private String fax; + private String mobile; + private String qualification; + private String[] badges; + private EPASOffice office; + private String fullname; + private String fiscalCode; + private String email; + private String number; + private String eppn; + } \ No newline at end of file diff --git a/src/main/java/it/cnr/isti/epasmed/epas/model/EPASWorkingTimeForPerson.java b/src/main/java/it/cnr/isti/epasmed/epas/model/EPASWorkingTimeForPerson.java new file mode 100644 index 0000000..7d45c4e --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/epas/model/EPASWorkingTimeForPerson.java @@ -0,0 +1,20 @@ +package it.cnr.isti.epasmed.epas.model; + +import java.io.Serializable; + +import lombok.Data; +import lombok.AllArgsConstructor; +import lombok.NoArgsConstructor; + +@Data +@AllArgsConstructor +@NoArgsConstructor +public class EPASWorkingTimeForPerson implements Serializable { + + private static final long serialVersionUID = 1L; + + private String beginDate; + private String endDate; + private EPASWorkingTimeTypes workingTimeType; + +} \ No newline at end of file diff --git a/src/main/java/it/cnr/isti/epasmed/epas/model/EPASWorkingTimeTypeDays.java b/src/main/java/it/cnr/isti/epasmed/epas/model/EPASWorkingTimeTypeDays.java new file mode 100644 index 0000000..4b02ece --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/epas/model/EPASWorkingTimeTypeDays.java @@ -0,0 +1,27 @@ +package it.cnr.isti.epasmed.epas.model; + +import java.io.Serializable; + +import lombok.Data; +import lombok.AllArgsConstructor; +import lombok.NoArgsConstructor; + +@Data +@AllArgsConstructor +@NoArgsConstructor +public class EPASWorkingTimeTypeDays implements Serializable { + + private static final long serialVersionUID = 1L; + + private String breakTicketTime; + private String dayOfWeek; + private String holiday; + private String id; + private String ticketAfternoonThreshold; + private String ticketAfternoonWorkingTime; + private String timeMealFrom; + private String timeMealTo; + private String updatedAt; + private String workingTime; + +} \ No newline at end of file diff --git a/src/main/java/it/cnr/isti/epasmed/epas/model/EPASWorkingTimeTypes.java b/src/main/java/it/cnr/isti/epasmed/epas/model/EPASWorkingTimeTypes.java new file mode 100644 index 0000000..a52b137 --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/epas/model/EPASWorkingTimeTypes.java @@ -0,0 +1,25 @@ +package it.cnr.isti.epasmed.epas.model; + +import java.io.Serializable; + +import lombok.Data; +import lombok.AllArgsConstructor; +import lombok.NoArgsConstructor; + +@Data +@AllArgsConstructor +@NoArgsConstructor +public class EPASWorkingTimeTypes implements Serializable { + + private static final long serialVersionUID = 1L; + + private String description; + private String disabled; + private String externalId; + private String horizontal; + private String id; + private String office; + private String updatedAt; + private EPASWorkingTimeTypeDays[] workingTimeTypeDays; + +} \ No newline at end of file diff --git a/src/main/java/it/cnr/isti/epasmed/epas/service/EPASAffiliationsService.java b/src/main/java/it/cnr/isti/epasmed/epas/service/EPASAffiliationsService.java new file mode 100644 index 0000000..98a4d45 --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/epas/service/EPASAffiliationsService.java @@ -0,0 +1,48 @@ +package it.cnr.isti.epasmed.epas.service; + +import java.util.List; + +import javax.validation.Valid; + +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.stereotype.Service; + +import it.cnr.isti.epasmed.epas.client.EPASAffiliationsCient; +import it.cnr.isti.epasmed.epas.dto.EPASAffiliationsDTO; +import it.cnr.isti.epasmed.epas.model.EPASAffiliations; + +@Service +public class EPASAffiliationsService { + + @Autowired + EPASAffiliationsCient epasAffiliationsClient; + + public EPASAffiliations getById(String id) { + return epasAffiliationsClient.getById(id); + } + + public List getByGroup(String groupId, boolean includeInactive ) { + return epasAffiliationsClient.getByGroup(groupId,includeInactive); + } + + public List getByPersonId(String personId) { + return epasAffiliationsClient.getByPersonId(personId); + } + + public List getByPersonFiscalcode(String fc) { + return epasAffiliationsClient.getByPersonFiscalcode(fc); + } + + public EPASAffiliations create(EPASAffiliationsDTO epasAffiliationsDTO) { + return epasAffiliationsClient.create(epasAffiliationsDTO); + } + + public void updateById(String id, @Valid EPASAffiliationsDTO epasAffiliationsDTO) { + epasAffiliationsClient.updateById(id,epasAffiliationsDTO); + } + + public void deleteById(String id) { + epasAffiliationsClient.deleteById(id); + } + +} diff --git a/src/main/java/it/cnr/isti/epasmed/epas/service/EPASContractsService.java b/src/main/java/it/cnr/isti/epasmed/epas/service/EPASContractsService.java new file mode 100644 index 0000000..a980b03 --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/epas/service/EPASContractsService.java @@ -0,0 +1,52 @@ +package it.cnr.isti.epasmed.epas.service; + +import java.util.List; + +import javax.validation.Valid; + +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.stereotype.Service; + +import it.cnr.isti.epasmed.epas.client.EPASContractsCient; +import it.cnr.isti.epasmed.epas.dto.EPASContractsDTO; +import it.cnr.isti.epasmed.epas.model.EPASContracts; + +@Service +public class EPASContractsService { + + @Autowired + EPASContractsCient epasContractsClient; + + public EPASContracts getById(String id) { + return epasContractsClient.getById(id); + } + + public List getByPersonId(String id) { + return epasContractsClient.getByPersonId(id); + } + + public List getByPersonFiscalcode(String fc) { + return epasContractsClient.getByPersonFiscalcode(fc); + } + + public EPASContracts create(EPASContractsDTO epasContractsDTO) { + return epasContractsClient.create(epasContractsDTO); + } + + public void updateById(String id, @Valid EPASContractsDTO epasContractsDTO) { + epasContractsClient.updateById(id, epasContractsDTO); + } + + public void setPreviousContract(String id) { + epasContractsClient.setPreviousContract(id); + } + + public void setUnsetPreviousContract(String id) { + epasContractsClient.setUnsetPreviousContract(id); + } + + public void deleteById(String id) { + epasContractsClient.deleteById(id); + } + +} diff --git a/src/main/java/it/cnr/isti/epasmed/epas/service/EPASGroupsService.java b/src/main/java/it/cnr/isti/epasmed/epas/service/EPASGroupsService.java new file mode 100644 index 0000000..eb7e087 --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/epas/service/EPASGroupsService.java @@ -0,0 +1,40 @@ +package it.cnr.isti.epasmed.epas.service; + +import java.util.List; + +import javax.validation.Valid; + +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.stereotype.Service; + +import it.cnr.isti.epasmed.epas.client.EPASGroupsCient; +import it.cnr.isti.epasmed.epas.dto.EPASGroupsDTO; +import it.cnr.isti.epasmed.epas.model.EPASGroups; + +@Service +public class EPASGroupsService { + + @Autowired + EPASGroupsCient epasGroupsClient; + + public EPASGroups getById(String id) { + return epasGroupsClient.getById(id); + } + + public List getList(String officeId) { + return epasGroupsClient.getList(officeId); + } + + public EPASGroups create(EPASGroupsDTO epasGroupsDTO) { + return epasGroupsClient.create(epasGroupsDTO); + } + + public void updateById(String id, @Valid EPASGroupsDTO epasGroupsDTO) { + epasGroupsClient.updateById(id,epasGroupsDTO); + } + + public void deleteById(String id) { + epasGroupsClient.deleteById(id); + } + +} diff --git a/src/main/java/it/cnr/isti/epasmed/epas/service/EPASPersonsService.java b/src/main/java/it/cnr/isti/epasmed/epas/service/EPASPersonsService.java new file mode 100644 index 0000000..6ccb738 --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/epas/service/EPASPersonsService.java @@ -0,0 +1,54 @@ +package it.cnr.isti.epasmed.epas.service; + +import java.util.List; + +import javax.validation.Valid; + +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.stereotype.Service; + +import it.cnr.isti.epasmed.epas.client.EPASPersonsCient; +import it.cnr.isti.epasmed.epas.dto.EPASPersonsDTO; +import it.cnr.isti.epasmed.epas.model.EPASPersons; + +@Service +public class EPASPersonsService { + + @Autowired + EPASPersonsCient epasPersonsClient; + + public EPASPersons getById(String id) { + return epasPersonsClient.getById(id); + } + + public EPASPersons getByFiscalCode(String fc) { + return epasPersonsClient.getByFiscalCode(fc); + } + + public List getList(String officeId) { + return epasPersonsClient.getList(officeId); + } + + public EPASPersons create(EPASPersonsDTO epasPersonsDTO) { + return epasPersonsClient.create(epasPersonsDTO); + } + + public void updateById(String id, @Valid EPASPersonsDTO epasPersonsDTO) { + epasPersonsClient.updateById(id,epasPersonsDTO); + } + + public void updateByFiscalCode(String fc, EPASPersonsDTO epasPersonsDTO) { + epasPersonsClient.updateByFiscalCode(fc,epasPersonsDTO); + } + + public void deleteById(String id) { + epasPersonsClient.deleteById(id); + } + + public void deleteByFiscalCode(String fc) { + epasPersonsClient.deleteByFiscalCode(fc); + + } + + +} diff --git a/src/main/java/it/cnr/isti/epasmed/epas/service/EPASWorkingTimeTypesService.java b/src/main/java/it/cnr/isti/epasmed/epas/service/EPASWorkingTimeTypesService.java new file mode 100644 index 0000000..793179c --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/epas/service/EPASWorkingTimeTypesService.java @@ -0,0 +1,26 @@ +package it.cnr.isti.epasmed.epas.service; + +import java.util.List; + +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.stereotype.Service; + +import it.cnr.isti.epasmed.epas.client.EPASWorkinTipeTypesCient; +import it.cnr.isti.epasmed.epas.model.EPASWorkingTimeTypes; + + +@Service +public class EPASWorkingTimeTypesService { + + @Autowired + EPASWorkinTipeTypesCient epasWorkingTimeTypesClient; + + public EPASWorkingTimeTypes getById(String id) { + return epasWorkingTimeTypesClient.getById(id); + } + + public List getList(String officeId) { + return epasWorkingTimeTypesClient.getList(officeId); + } + +} diff --git a/src/main/java/it/cnr/isti/epasmed/repository/AuthorityRepository.java b/src/main/java/it/cnr/isti/epasmed/repository/AuthorityRepository.java new file mode 100644 index 0000000..9b98673 --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/repository/AuthorityRepository.java @@ -0,0 +1,11 @@ +package it.cnr.isti.epasmed.repository; + +import it.cnr.isti.epasmed.domain.Authority; + +import org.springframework.data.jpa.repository.JpaRepository; + +/** + * Spring Data JPA repository for the {@link Authority} entity. + */ +public interface AuthorityRepository extends JpaRepository { +} diff --git a/src/main/java/it/cnr/isti/epasmed/repository/CustomAuditEventRepository.java b/src/main/java/it/cnr/isti/epasmed/repository/CustomAuditEventRepository.java new file mode 100644 index 0000000..6f2ed4e --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/repository/CustomAuditEventRepository.java @@ -0,0 +1,89 @@ +package it.cnr.isti.epasmed.repository; + +import it.cnr.isti.epasmed.config.Constants; +import it.cnr.isti.epasmed.config.audit.AuditEventConverter; +import it.cnr.isti.epasmed.domain.PersistentAuditEvent; + +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; +import org.springframework.boot.actuate.audit.AuditEvent; +import org.springframework.boot.actuate.audit.AuditEventRepository; +import org.springframework.stereotype.Repository; +import org.springframework.transaction.annotation.Propagation; +import org.springframework.transaction.annotation.Transactional; + +import java.time.Instant; +import java.util.*; + +/** + * An implementation of Spring Boot's {@link AuditEventRepository}. + */ +@Repository +public class CustomAuditEventRepository implements AuditEventRepository { + + private static final String AUTHORIZATION_FAILURE = "AUTHORIZATION_FAILURE"; + + /** + * Should be the same as in Liquibase migration. + */ + protected static final int EVENT_DATA_COLUMN_MAX_LENGTH = 255; + + private final PersistenceAuditEventRepository persistenceAuditEventRepository; + + private final AuditEventConverter auditEventConverter; + + private final Logger log = LoggerFactory.getLogger(getClass()); + + public CustomAuditEventRepository(PersistenceAuditEventRepository persistenceAuditEventRepository, + AuditEventConverter auditEventConverter) { + + this.persistenceAuditEventRepository = persistenceAuditEventRepository; + this.auditEventConverter = auditEventConverter; + } + + @Override + public List find(String principal, Instant after, String type) { + Iterable persistentAuditEvents = + persistenceAuditEventRepository.findByPrincipalAndAuditEventDateAfterAndAuditEventType(principal, after, type); + return auditEventConverter.convertToAuditEvent(persistentAuditEvents); + } + + @Override + @Transactional(value="epasMedTransactionManager",propagation = Propagation.REQUIRES_NEW) + public void add(AuditEvent event) { + if (!AUTHORIZATION_FAILURE.equals(event.getType()) && + !Constants.ANONYMOUS_USER.equals(event.getPrincipal())) { + + PersistentAuditEvent persistentAuditEvent = new PersistentAuditEvent(); + persistentAuditEvent.setPrincipal(event.getPrincipal()); + persistentAuditEvent.setAuditEventType(event.getType()); + persistentAuditEvent.setAuditEventDate(event.getTimestamp()); + Map eventData = auditEventConverter.convertDataToStrings(event.getData()); + persistentAuditEvent.setData(truncate(eventData)); + persistenceAuditEventRepository.save(persistentAuditEvent); + } + } + + /** + * Truncate event data that might exceed column length. + */ + private Map truncate(Map data) { + Map results = new HashMap<>(); + + if (data != null) { + for (Map.Entry entry : data.entrySet()) { + String value = entry.getValue(); + if (value != null) { + int length = value.length(); + if (length > EVENT_DATA_COLUMN_MAX_LENGTH) { + value = value.substring(0, EVENT_DATA_COLUMN_MAX_LENGTH); + log.warn("Event data for {} too long ({}) has been truncated to {}. Consider increasing column width.", + entry.getKey(), length, EVENT_DATA_COLUMN_MAX_LENGTH); + } + } + results.put(entry.getKey(), value); + } + } + return results; + } +} diff --git a/src/main/java/it/cnr/isti/epasmed/repository/PersistenceAuditEventRepository.java b/src/main/java/it/cnr/isti/epasmed/repository/PersistenceAuditEventRepository.java new file mode 100644 index 0000000..4f926b2 --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/repository/PersistenceAuditEventRepository.java @@ -0,0 +1,23 @@ +package it.cnr.isti.epasmed.repository; + +import it.cnr.isti.epasmed.domain.PersistentAuditEvent; +import org.springframework.data.domain.Page; +import org.springframework.data.domain.Pageable; +import org.springframework.data.jpa.repository.JpaRepository; + +import java.time.Instant; +import java.util.List; + +/** + * Spring Data JPA repository for the {@link PersistentAuditEvent} entity. + */ +public interface PersistenceAuditEventRepository extends JpaRepository { + + List findByPrincipal(String principal); + + List findByPrincipalAndAuditEventDateAfterAndAuditEventType(String principal, Instant after, String type); + + Page findAllByAuditEventDateBetween(Instant fromDate, Instant toDate, Pageable pageable); + + List findByAuditEventDateBefore(Instant before); +} diff --git a/src/main/java/it/cnr/isti/epasmed/repository/PersistentTokenRepository.java b/src/main/java/it/cnr/isti/epasmed/repository/PersistentTokenRepository.java new file mode 100644 index 0000000..058a2a0 --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/repository/PersistentTokenRepository.java @@ -0,0 +1,19 @@ +package it.cnr.isti.epasmed.repository; + +import it.cnr.isti.epasmed.domain.PersistentToken; +import it.cnr.isti.epasmed.domain.User; +import java.time.LocalDate; +import org.springframework.data.jpa.repository.JpaRepository; + +import java.util.List; + +/** + * Spring Data JPA repository for the {@link PersistentToken} entity. + */ +public interface PersistentTokenRepository extends JpaRepository { + + List findByUser(User user); + + List findByTokenDateBefore(LocalDate localDate); + +} diff --git a/src/main/java/it/cnr/isti/epasmed/repository/TabsSIRepository.java b/src/main/java/it/cnr/isti/epasmed/repository/TabsSIRepository.java new file mode 100644 index 0000000..36980a4 --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/repository/TabsSIRepository.java @@ -0,0 +1,19 @@ +package it.cnr.isti.epasmed.repository; + +import java.util.Optional; + +import org.springframework.data.jpa.repository.JpaRepository; +import org.springframework.stereotype.Repository; + +import it.cnr.isti.epasmed.domain.TabsSI; +import it.cnr.isti.epasmed.domain.User; + +/** + * Spring Data JPA repository for the {@link User} entity. + */ +@Repository +public interface TabsSIRepository extends JpaRepository { + + Optional findOneByNome(String nome); + + } diff --git a/src/main/java/it/cnr/isti/epasmed/repository/UserRepository.java b/src/main/java/it/cnr/isti/epasmed/repository/UserRepository.java new file mode 100644 index 0000000..5df2edc --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/repository/UserRepository.java @@ -0,0 +1,38 @@ +package it.cnr.isti.epasmed.repository; + +import it.cnr.isti.epasmed.domain.User; + +import org.springframework.data.domain.Page; +import org.springframework.data.domain.Pageable; +import org.springframework.data.jpa.repository.EntityGraph; +import org.springframework.data.jpa.repository.JpaRepository; +import org.springframework.stereotype.Repository; + +import java.util.List; +import java.util.Optional; +import java.time.Instant; + +/** + * Spring Data JPA repository for the {@link User} entity. + */ +@Repository +public interface UserRepository extends JpaRepository { + + Optional findOneByActivationKey(String activationKey); + + List findAllByActivatedIsFalseAndActivationKeyIsNotNullAndCreatedDateBefore(Instant dateTime); + + Optional findOneByResetKey(String resetKey); + + Optional findOneByEmailIgnoreCase(String email); + + Optional findOneByLogin(String login); + + @EntityGraph(attributePaths = "authorities") + Optional findOneWithAuthoritiesByLogin(String login); + + @EntityGraph(attributePaths = "authorities") + Optional findOneWithAuthoritiesByEmailIgnoreCase(String email); + + Page findAllByLoginNot(Pageable pageable, String login); +} diff --git a/src/main/java/it/cnr/isti/epasmed/repository/package-info.java b/src/main/java/it/cnr/isti/epasmed/repository/package-info.java new file mode 100644 index 0000000..ecfd4df --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/repository/package-info.java @@ -0,0 +1,4 @@ +/** + * Spring Data JPA repositories. + */ +package it.cnr.isti.epasmed.repository; diff --git a/src/main/java/it/cnr/isti/epasmed/security/AuthoritiesConstants.java b/src/main/java/it/cnr/isti/epasmed/security/AuthoritiesConstants.java new file mode 100644 index 0000000..62c6209 --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/security/AuthoritiesConstants.java @@ -0,0 +1,18 @@ +package it.cnr.isti.epasmed.security; + +/** + * Constants for Spring Security authorities. + */ +public final class AuthoritiesConstants { + + public static final String ADMIN = "ROLE_ADMIN"; + + public static final String USER = "ROLE_USER"; + + public static final String PROMETHEUS = "ROLE_PROMETHEUS"; + + public static final String ANONYMOUS = "ROLE_ANONYMOUS"; + + private AuthoritiesConstants() { + } +} diff --git a/src/main/java/it/cnr/isti/epasmed/security/DomainUserDetailsService.java b/src/main/java/it/cnr/isti/epasmed/security/DomainUserDetailsService.java new file mode 100644 index 0000000..8863d15 --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/security/DomainUserDetailsService.java @@ -0,0 +1,62 @@ +package it.cnr.isti.epasmed.security; + +import it.cnr.isti.epasmed.domain.User; +import it.cnr.isti.epasmed.repository.UserRepository; +import org.hibernate.validator.internal.constraintvalidators.hv.EmailValidator; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; +import org.springframework.security.core.GrantedAuthority; +import org.springframework.security.core.authority.SimpleGrantedAuthority; +import org.springframework.security.core.userdetails.UserDetails; +import org.springframework.security.core.userdetails.UserDetailsService; +import org.springframework.security.core.userdetails.UsernameNotFoundException; +import org.springframework.stereotype.Component; +import org.springframework.transaction.annotation.Transactional; + +import java.util.*; +import java.util.stream.Collectors; + +/** + * Authenticate a user from the database. + */ +@Component("userDetailsService") +public class DomainUserDetailsService implements UserDetailsService { + + private final Logger log = LoggerFactory.getLogger(DomainUserDetailsService.class); + + private final UserRepository userRepository; + + public DomainUserDetailsService(UserRepository userRepository) { + this.userRepository = userRepository; + } + + @Override + @Transactional("epasMedTransactionManager") + public UserDetails loadUserByUsername(final String login) { + log.debug("Authenticating {}", login); + + if (new EmailValidator().isValid(login, null)) { + return userRepository.findOneWithAuthoritiesByEmailIgnoreCase(login) + .map(user -> createSpringSecurityUser(login, user)) + .orElseThrow(() -> new UsernameNotFoundException("User with email " + login + " was not found in the database")); + } + + String lowercaseLogin = login.toLowerCase(Locale.ENGLISH); + return userRepository.findOneWithAuthoritiesByLogin(lowercaseLogin) + .map(user -> createSpringSecurityUser(lowercaseLogin, user)) + .orElseThrow(() -> new UsernameNotFoundException("User " + lowercaseLogin + " was not found in the database")); + + } + + private org.springframework.security.core.userdetails.User createSpringSecurityUser(String lowercaseLogin, User user) { + if (!user.getActivated()) { + throw new UserNotActivatedException("User " + lowercaseLogin + " was not activated"); + } + List grantedAuthorities = user.getAuthorities().stream() + .map(authority -> new SimpleGrantedAuthority(authority.getName())) + .collect(Collectors.toList()); + return new org.springframework.security.core.userdetails.User(user.getLogin(), + user.getPassword(), + grantedAuthorities); + } +} diff --git a/src/main/java/it/cnr/isti/epasmed/security/PersistentTokenRememberMeServices.java b/src/main/java/it/cnr/isti/epasmed/security/PersistentTokenRememberMeServices.java new file mode 100644 index 0000000..e6a2633 --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/security/PersistentTokenRememberMeServices.java @@ -0,0 +1,228 @@ +package it.cnr.isti.epasmed.security; + +import it.cnr.isti.epasmed.domain.PersistentToken; +import it.cnr.isti.epasmed.repository.PersistentTokenRepository; +import it.cnr.isti.epasmed.repository.UserRepository; + + +import io.github.jhipster.config.JHipsterProperties; +import io.github.jhipster.security.PersistentTokenCache; +import io.github.jhipster.security.RandomUtil; + +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; +import org.springframework.dao.DataAccessException; +import org.springframework.security.core.Authentication; +import org.springframework.security.core.userdetails.UserDetails; +import org.springframework.security.core.userdetails.UsernameNotFoundException; +import org.springframework.security.web.authentication.rememberme.*; +import org.springframework.stereotype.Service; + +import javax.servlet.http.HttpServletRequest; +import javax.servlet.http.HttpServletResponse; +import java.io.Serializable; +import java.time.LocalDate; +import java.util.*; + +/** + * Custom implementation of Spring Security's RememberMeServices. + *

+ * Persistent tokens are used by Spring Security to automatically log in users. + *

+ * This is a specific implementation of Spring Security's remember-me authentication, but it is much + * more powerful than the standard implementations: + *

    + *
  • It allows a user to see the list of his currently opened sessions, and invalidate them
  • + *
  • It stores more information, such as the IP address and the user agent, for audit purposes
  • + *
  • When a user logs out, only his current session is invalidated, and not all of his sessions
  • + *
+ *

+ * Please note that it allows the use of the same token for 5 seconds, and this value stored in a specific + * cache during that period. This is to allow concurrent requests from the same user: otherwise, two + * requests being sent at the same time could invalidate each other's token. + *

+ * This is inspired by: + *

+ *

+ * The main algorithm comes from Spring Security's {@code PersistentTokenBasedRememberMeServices}, but this class + * couldn't be cleanly extended. + */ +@Service +public class PersistentTokenRememberMeServices extends + AbstractRememberMeServices { + + private final Logger log = LoggerFactory.getLogger(PersistentTokenRememberMeServices.class); + + // Token is valid for one month + private static final int TOKEN_VALIDITY_DAYS = 31; + + private static final int TOKEN_VALIDITY_SECONDS = 60 * 60 * 24 * TOKEN_VALIDITY_DAYS; + + private static final long UPGRADED_TOKEN_VALIDITY_MILLIS = 5000l; + + private final PersistentTokenCache upgradedTokenCache; + + private final PersistentTokenRepository persistentTokenRepository; + + private final UserRepository userRepository; + + public PersistentTokenRememberMeServices(JHipsterProperties jHipsterProperties, + org.springframework.security.core.userdetails.UserDetailsService userDetailsService, + PersistentTokenRepository persistentTokenRepository, UserRepository userRepository) { + + super(jHipsterProperties.getSecurity().getRememberMe().getKey(), userDetailsService); + this.persistentTokenRepository = persistentTokenRepository; + this.userRepository = userRepository; + upgradedTokenCache = new PersistentTokenCache<>(UPGRADED_TOKEN_VALIDITY_MILLIS); + } + + @Override + protected UserDetails processAutoLoginCookie(String[] cookieTokens, HttpServletRequest request, + HttpServletResponse response) { + + synchronized (this) { // prevent 2 authentication requests from the same user in parallel + String login = null; + UpgradedRememberMeToken upgradedToken = upgradedTokenCache.get(cookieTokens[0]); + if (upgradedToken != null) { + login = upgradedToken.getUserLoginIfValid(cookieTokens); + log.debug("Detected previously upgraded login token for user '{}'", login); + } + + if (login == null) { + PersistentToken token = getPersistentToken(cookieTokens); + login = token.getUser().getLogin(); + + // Token also matches, so login is valid. Update the token value, keeping the *same* series number. + log.debug("Refreshing persistent login token for user '{}', series '{}'", login, token.getSeries()); + token.setTokenDate(LocalDate.now()); + token.setTokenValue(RandomUtil.generateRandomAlphanumericString()); + token.setIpAddress(request.getRemoteAddr()); + token.setUserAgent(request.getHeader("User-Agent")); + try { + persistentTokenRepository.saveAndFlush(token); + } catch (DataAccessException e) { + log.error("Failed to update token: ", e); + throw new RememberMeAuthenticationException("Autologin failed due to data access problem", e); + } + addCookie(token, request, response); + upgradedTokenCache.put(cookieTokens[0], new UpgradedRememberMeToken(cookieTokens, login)); + } + return getUserDetailsService().loadUserByUsername(login); + } + } + + @Override + protected void onLoginSuccess(HttpServletRequest request, HttpServletResponse response, Authentication + successfulAuthentication) { + + String login = successfulAuthentication.getName(); + + log.debug("Creating new persistent login for user {}", login); + PersistentToken token = userRepository.findOneByLogin(login).map(u -> { + PersistentToken t = new PersistentToken(); + t.setSeries(RandomUtil.generateRandomAlphanumericString()); + t.setUser(u); + t.setTokenValue(RandomUtil.generateRandomAlphanumericString()); + t.setTokenDate(LocalDate.now()); + t.setIpAddress(request.getRemoteAddr()); + t.setUserAgent(request.getHeader("User-Agent")); + return t; + }).orElseThrow(() -> new UsernameNotFoundException("User " + login + " was not found in the database")); + try { + persistentTokenRepository.saveAndFlush(token); + addCookie(token, request, response); + } catch (DataAccessException e) { + log.error("Failed to save persistent token ", e); + } + } + + /** + * When logout occurs, only invalidate the current token, and not all user sessions. + *

+ * The standard Spring Security implementations are too basic: they invalidate all tokens for the + * current user, so when he logs out from one browser, all his other sessions are destroyed. + * + * @param request the request. + * @param response the response. + * @param authentication the authentication. + */ + @Override + public void logout(HttpServletRequest request, HttpServletResponse response, Authentication authentication) { + String rememberMeCookie = extractRememberMeCookie(request); + if (rememberMeCookie != null && rememberMeCookie.length() != 0) { + try { + String[] cookieTokens = decodeCookie(rememberMeCookie); + PersistentToken token = getPersistentToken(cookieTokens); + persistentTokenRepository.deleteById(token.getSeries()); + } catch (InvalidCookieException ice) { + log.info("Invalid cookie, no persistent token could be deleted", ice); + } catch (RememberMeAuthenticationException rmae) { + log.debug("No persistent token found, so no token could be deleted", rmae); + } + } + super.logout(request, response, authentication); + } + + /** + * Validate the token and return it. + */ + private PersistentToken getPersistentToken(String[] cookieTokens) { + if (cookieTokens.length != 2) { + throw new InvalidCookieException("Cookie token did not contain " + 2 + + " tokens, but contained '" + Arrays.asList(cookieTokens) + "'"); + } + String presentedSeries = cookieTokens[0]; + String presentedToken = cookieTokens[1]; + Optional optionalToken = persistentTokenRepository.findById(presentedSeries); + if (!optionalToken.isPresent()) { + // No series match, so we can't authenticate using this cookie + throw new RememberMeAuthenticationException("No persistent token found for series id: " + presentedSeries); + } + PersistentToken token = optionalToken.get(); + // We have a match for this user/series combination + log.info("presentedToken={} / tokenValue={}", presentedToken, token.getTokenValue()); + if (!presentedToken.equals(token.getTokenValue())) { + // Token doesn't match series value. Delete this session and throw an exception. + persistentTokenRepository.deleteById(token.getSeries()); + throw new CookieTheftException("Invalid remember-me token (Series/token) mismatch. Implies previous " + + "cookie theft attack."); + } + if (token.getTokenDate().plusDays(TOKEN_VALIDITY_DAYS).isBefore(LocalDate.now())) { + persistentTokenRepository.deleteById(token.getSeries()); + throw new RememberMeAuthenticationException("Remember-me login has expired"); + } + return token; + } + + private void addCookie(PersistentToken token, HttpServletRequest request, HttpServletResponse response) { + setCookie( + new String[]{token.getSeries(), token.getTokenValue()}, + TOKEN_VALIDITY_SECONDS, request, response); + } + + private static class UpgradedRememberMeToken implements Serializable { + + private static final long serialVersionUID = 1L; + + private final String[] upgradedToken; + + private final String userLogin; + + UpgradedRememberMeToken(String[] upgradedToken, String userLogin) { + this.upgradedToken = upgradedToken; + this.userLogin = userLogin; + } + + String getUserLoginIfValid(String[] currentToken) { + if (currentToken[0].equals(this.upgradedToken[0]) && + currentToken[1].equals(this.upgradedToken[1])) { + return this.userLogin; + } + return null; + } + } +} diff --git a/src/main/java/it/cnr/isti/epasmed/security/SecurityUtils.java b/src/main/java/it/cnr/isti/epasmed/security/SecurityUtils.java new file mode 100644 index 0000000..b7dc7d4 --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/security/SecurityUtils.java @@ -0,0 +1,73 @@ +package it.cnr.isti.epasmed.security; + +import org.springframework.security.core.Authentication; +import org.springframework.security.core.GrantedAuthority; +import org.springframework.security.core.context.SecurityContext; +import org.springframework.security.core.context.SecurityContextHolder; +import org.springframework.security.core.userdetails.UserDetails; + +import java.util.Optional; +import java.util.stream.Stream; + +/** + * Utility class for Spring Security. + */ +public final class SecurityUtils { + + private SecurityUtils() { + } + + /** + * Get the login of the current user. + * + * @return the login of the current user. + */ + public static Optional getCurrentUserLogin() { + SecurityContext securityContext = SecurityContextHolder.getContext(); + return Optional.ofNullable(extractPrincipal(securityContext.getAuthentication())); + } + + private static String extractPrincipal(Authentication authentication) { + if (authentication == null) { + return null; + } else if (authentication.getPrincipal() instanceof UserDetails) { + UserDetails springSecurityUser = (UserDetails) authentication.getPrincipal(); + return springSecurityUser.getUsername(); + } else if (authentication.getPrincipal() instanceof String) { + return (String) authentication.getPrincipal(); + } + return null; + } + + + /** + * Check if a user is authenticated. + * + * @return true if the user is authenticated, false otherwise. + */ + public static boolean isAuthenticated() { + Authentication authentication = SecurityContextHolder.getContext().getAuthentication(); + return authentication != null && + getAuthorities(authentication).noneMatch(AuthoritiesConstants.ANONYMOUS::equals); + } + + /** + * If the current user has a specific authority (security role). + *

+ * The name of this method comes from the {@code isUserInRole()} method in the Servlet API. + * + * @param authority the authority to check. + * @return true if the current user has the authority, false otherwise. + */ + public static boolean isCurrentUserInRole(String authority) { + Authentication authentication = SecurityContextHolder.getContext().getAuthentication(); + return authentication != null && + getAuthorities(authentication).anyMatch(authority::equals); + } + + private static Stream getAuthorities(Authentication authentication) { + return authentication.getAuthorities().stream() + .map(GrantedAuthority::getAuthority); + } + +} diff --git a/src/main/java/it/cnr/isti/epasmed/security/SpringSecurityAuditorAware.java b/src/main/java/it/cnr/isti/epasmed/security/SpringSecurityAuditorAware.java new file mode 100644 index 0000000..dcdd788 --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/security/SpringSecurityAuditorAware.java @@ -0,0 +1,20 @@ +package it.cnr.isti.epasmed.security; + +import it.cnr.isti.epasmed.config.Constants; + +import java.util.Optional; + +import org.springframework.data.domain.AuditorAware; +import org.springframework.stereotype.Component; + +/** + * Implementation of {@link AuditorAware} based on Spring Security. + */ +@Component +public class SpringSecurityAuditorAware implements AuditorAware { + + @Override + public Optional getCurrentAuditor() { + return Optional.of(SecurityUtils.getCurrentUserLogin().orElse(Constants.SYSTEM_ACCOUNT)); + } +} diff --git a/src/main/java/it/cnr/isti/epasmed/security/UserNotActivatedException.java b/src/main/java/it/cnr/isti/epasmed/security/UserNotActivatedException.java new file mode 100644 index 0000000..b04b91a --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/security/UserNotActivatedException.java @@ -0,0 +1,19 @@ +package it.cnr.isti.epasmed.security; + +import org.springframework.security.core.AuthenticationException; + +/** + * This exception is thrown in case of a not activated user trying to authenticate. + */ +public class UserNotActivatedException extends AuthenticationException { + + private static final long serialVersionUID = 1L; + + public UserNotActivatedException(String message) { + super(message); + } + + public UserNotActivatedException(String message, Throwable t) { + super(message, t); + } +} diff --git a/src/main/java/it/cnr/isti/epasmed/security/package-info.java b/src/main/java/it/cnr/isti/epasmed/security/package-info.java new file mode 100644 index 0000000..b5c45cf --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/security/package-info.java @@ -0,0 +1,4 @@ +/** + * Spring Security configuration. + */ +package it.cnr.isti.epasmed.security; diff --git a/src/main/java/it/cnr/isti/epasmed/service/AuditEventService.java b/src/main/java/it/cnr/isti/epasmed/service/AuditEventService.java new file mode 100644 index 0000000..c04d64c --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/service/AuditEventService.java @@ -0,0 +1,77 @@ +package it.cnr.isti.epasmed.service; + +import io.github.jhipster.config.JHipsterProperties; +import it.cnr.isti.epasmed.config.audit.AuditEventConverter; +import it.cnr.isti.epasmed.repository.PersistenceAuditEventRepository; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; +import org.springframework.boot.actuate.audit.AuditEvent; +import org.springframework.data.domain.Page; +import org.springframework.data.domain.Pageable; +import org.springframework.scheduling.annotation.Scheduled; +import org.springframework.stereotype.Service; +import org.springframework.transaction.annotation.Transactional; + +import java.time.Instant; +import java.time.temporal.ChronoUnit; +import java.util.Optional; + +/** + * Service for managing audit events. + *

+ * This is the default implementation to support SpringBoot Actuator {@code AuditEventRepository}. + */ +@Service +@Transactional("epasMedTransactionManager") +public class AuditEventService { + + private final Logger log = LoggerFactory.getLogger(AuditEventService.class); + + private final JHipsterProperties jHipsterProperties; + + private final PersistenceAuditEventRepository persistenceAuditEventRepository; + + private final AuditEventConverter auditEventConverter; + + public AuditEventService( + PersistenceAuditEventRepository persistenceAuditEventRepository, + AuditEventConverter auditEventConverter, JHipsterProperties jhipsterProperties) { + + this.persistenceAuditEventRepository = persistenceAuditEventRepository; + this.auditEventConverter = auditEventConverter; + this.jHipsterProperties = jhipsterProperties; + } + + /** + * Old audit events should be automatically deleted after 30 days. + * + * This is scheduled to get fired at 12:00 (am). + */ + @Scheduled(cron = "0 0 12 * * ?") + public void removeOldAuditEvents() { + persistenceAuditEventRepository + .findByAuditEventDateBefore(Instant.now().minus(jHipsterProperties.getAuditEvents().getRetentionPeriod(), ChronoUnit.DAYS)) + .forEach(auditEvent -> { + log.debug("Deleting audit data {}", auditEvent); + persistenceAuditEventRepository.delete(auditEvent); + }); + } + + @Transactional(value="epasMedTransactionManager",readOnly = true) + public Page findAll(Pageable pageable) { + return persistenceAuditEventRepository.findAll(pageable) + .map(auditEventConverter::convertToAuditEvent); + } + + @Transactional(value="epasMedTransactionManager",readOnly = true) + public Page findByDates(Instant fromDate, Instant toDate, Pageable pageable) { + return persistenceAuditEventRepository.findAllByAuditEventDateBetween(fromDate, toDate, pageable) + .map(auditEventConverter::convertToAuditEvent); + } + + @Transactional(value="epasMedTransactionManager",readOnly = true) + public Optional find(Long id) { + return persistenceAuditEventRepository.findById(id) + .map(auditEventConverter::convertToAuditEvent); + } +} diff --git a/src/main/java/it/cnr/isti/epasmed/service/EmailAlreadyUsedException.java b/src/main/java/it/cnr/isti/epasmed/service/EmailAlreadyUsedException.java new file mode 100644 index 0000000..b652756 --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/service/EmailAlreadyUsedException.java @@ -0,0 +1,11 @@ +package it.cnr.isti.epasmed.service; + +public class EmailAlreadyUsedException extends RuntimeException { + + private static final long serialVersionUID = 1L; + + public EmailAlreadyUsedException() { + super("Email is already in use!"); + } + +} diff --git a/src/main/java/it/cnr/isti/epasmed/service/InvalidPasswordException.java b/src/main/java/it/cnr/isti/epasmed/service/InvalidPasswordException.java new file mode 100644 index 0000000..7ef3afc --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/service/InvalidPasswordException.java @@ -0,0 +1,11 @@ +package it.cnr.isti.epasmed.service; + +public class InvalidPasswordException extends RuntimeException { + + private static final long serialVersionUID = 1L; + + public InvalidPasswordException() { + super("Incorrect password"); + } + +} diff --git a/src/main/java/it/cnr/isti/epasmed/service/MailService.java b/src/main/java/it/cnr/isti/epasmed/service/MailService.java new file mode 100644 index 0000000..e439dc5 --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/service/MailService.java @@ -0,0 +1,106 @@ +package it.cnr.isti.epasmed.service; + +import it.cnr.isti.epasmed.domain.User; + +import io.github.jhipster.config.JHipsterProperties; + +import java.nio.charset.StandardCharsets; +import java.util.Locale; +import javax.mail.MessagingException; +import javax.mail.internet.MimeMessage; + +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; +import org.springframework.context.MessageSource; +import org.springframework.mail.MailException; +import org.springframework.mail.javamail.JavaMailSender; +import org.springframework.mail.javamail.MimeMessageHelper; +import org.springframework.scheduling.annotation.Async; +import org.springframework.stereotype.Service; +import org.thymeleaf.context.Context; +import org.thymeleaf.spring5.SpringTemplateEngine; + +/** + * Service for sending emails. + *

+ * We use the {@link Async} annotation to send emails asynchronously. + */ +@Service +public class MailService { + + private final Logger log = LoggerFactory.getLogger(MailService.class); + + private static final String USER = "user"; + + private static final String BASE_URL = "baseUrl"; + + private final JHipsterProperties jHipsterProperties; + + private final JavaMailSender javaMailSender; + + private final MessageSource messageSource; + + private final SpringTemplateEngine templateEngine; + + public MailService(JHipsterProperties jHipsterProperties, JavaMailSender javaMailSender, + MessageSource messageSource, SpringTemplateEngine templateEngine) { + + this.jHipsterProperties = jHipsterProperties; + this.javaMailSender = javaMailSender; + this.messageSource = messageSource; + this.templateEngine = templateEngine; + } + + @Async + public void sendEmail(String to, String subject, String content, boolean isMultipart, boolean isHtml) { + log.debug("Send email[multipart '{}' and html '{}'] to '{}' with subject '{}' and content={}", + isMultipart, isHtml, to, subject, content); + + // Prepare message using a Spring helper + MimeMessage mimeMessage = javaMailSender.createMimeMessage(); + try { + MimeMessageHelper message = new MimeMessageHelper(mimeMessage, isMultipart, StandardCharsets.UTF_8.name()); + message.setTo(to); + message.setFrom(jHipsterProperties.getMail().getFrom()); + message.setSubject(subject); + message.setText(content, isHtml); + javaMailSender.send(mimeMessage); + log.debug("Sent email to User '{}'", to); + } catch (MailException | MessagingException e) { + log.warn("Email could not be sent to user '{}'", to, e); + } + } + + @Async + public void sendEmailFromTemplate(User user, String templateName, String titleKey) { + if (user.getEmail() == null) { + log.debug("Email doesn't exist for user '{}'", user.getLogin()); + return; + } + Locale locale = Locale.forLanguageTag(user.getLangKey()); + Context context = new Context(locale); + context.setVariable(USER, user); + context.setVariable(BASE_URL, jHipsterProperties.getMail().getBaseUrl()); + String content = templateEngine.process(templateName, context); + String subject = messageSource.getMessage(titleKey, null, locale); + sendEmail(user.getEmail(), subject, content, false, true); + } + + @Async + public void sendActivationEmail(User user) { + log.debug("Sending activation email to '{}'", user.getEmail()); + sendEmailFromTemplate(user, "mail/activationEmail", "email.activation.title"); + } + + @Async + public void sendCreationEmail(User user) { + log.debug("Sending creation email to '{}'", user.getEmail()); + sendEmailFromTemplate(user, "mail/creationEmail", "email.activation.title"); + } + + @Async + public void sendPasswordResetMail(User user) { + log.debug("Sending password reset email to '{}'", user.getEmail()); + sendEmailFromTemplate(user, "mail/passwordResetEmail", "email.reset.title"); + } +} diff --git a/src/main/java/it/cnr/isti/epasmed/service/TabsSIService.java b/src/main/java/it/cnr/isti/epasmed/service/TabsSIService.java new file mode 100644 index 0000000..3cbf06a --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/service/TabsSIService.java @@ -0,0 +1,125 @@ +package it.cnr.isti.epasmed.service; + +import java.time.LocalDateTime; +import java.util.List; +import java.util.Optional; + +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.core.convert.ConversionService; +import org.springframework.data.domain.Page; +import org.springframework.data.domain.Pageable; +import org.springframework.stereotype.Service; +import org.springframework.transaction.annotation.Transactional; + +import it.cnr.isti.epasmed.domain.TabsSI; +import it.cnr.isti.epasmed.repository.TabsSIRepository; + +/** + * Service class for managing users. + */ +@Service +@Transactional("epasMedTransactionManager") +public class TabsSIService { + + private final Logger log = LoggerFactory.getLogger(TabsSIService.class); + + private final TabsSIRepository tabsSIRepository; + + @Autowired + ConversionService conversionService; + + // @Autowired + // FormattingConversionService conversionDateTimeService; + + public TabsSIService(TabsSIRepository tabsSIRepository) { + this.tabsSIRepository = tabsSIRepository; + } + + @Transactional(value = "epasMedTransactionManager", readOnly = true) + public Page getAllTabsSI(Pageable pageable) { + log.debug("TabsSIService getAllTabsSI(): {}", pageable); + tabsSIRepository.findAll(); + return tabsSIRepository.findAll(pageable); + } + + @Transactional(value = "epasMedTransactionManager", readOnly = true) + public List getAllTabsSI() { + log.debug("TabsSIService getAllTabsSI()"); + return tabsSIRepository.findAll(); + } + + @Transactional(value = "epasMedTransactionManager", readOnly = true) + public Optional getTabsSIById(Long id) { + return tabsSIRepository.findById(id); + } + + public TabsSI createTabsSI(TabsSI tabsSIDTO) { + log.debug("Creating new TabsSI: {}", tabsSIDTO); + TabsSI tabsSI = new TabsSI(); + tabsSI.setNome(tabsSIDTO.getNome()); + tabsSI.setOperazioni(tabsSIDTO.getOperazioni()); + tabsSI.setIdFlusso(tabsSIDTO.getIdFlusso()); + tabsSI.setLastUpdate(tabsSIDTO.getLastUpdate()); + tabsSIRepository.save(tabsSI); + + log.debug("Created TabsSI: {}", tabsSI); + return tabsSI; + } + + /** + * Update all information for a specific table, and return the modified table. + * + * @param tabsSI table to update. + * @return updated table. + */ + public Optional updateTabsSI(TabsSI tabsSIDTO) { + Optional tSI = tabsSIRepository.findById(tabsSIDTO.getId()); + if (tSI.isPresent()) { + TabsSI tabsSI = tSI.get(); + tabsSI.setNome(tabsSIDTO.getNome()); + tabsSI.setOperazioni(tabsSIDTO.getOperazioni()); + tabsSI.setIdFlusso(tabsSIDTO.getIdFlusso()); + tabsSI.setLastUpdate(tabsSIDTO.getLastUpdate()); + tabsSI = tabsSIRepository.save(tabsSI); + log.debug("Changed Information for TabsSI: {}", tabsSI); + return Optional.of(tabsSI); + } else { + return tSI; + } + } + + public void deleteTabsSI(Long id) { + tabsSIRepository.findById(id).ifPresent(tabsSI -> { + tabsSIRepository.delete(tabsSI); + log.debug("Deleted TabsSI: {}", tabsSI); + }); + } + + /** + * Update TabsSI information (id, nome, operazioni, id flusso, last update). + * + * @param id Identify + * @param nome Nome table + * @param operazioni Operazioni + * @param idFlusso Id Flusso + * @param lastUpdate Last Update + */ + public void updateTabSI(String id, String nome, String operazioni, String idFlusso, String lastUpdate) { + LocalDateTime lastU = conversionService.convert(lastUpdate, LocalDateTime.class); + // LocalDateTime lastU=conversionDateTimeService.convert(lastUpdate, + // LocalDateTime.class); + + Optional.of(tabsSIRepository.findById(Long.valueOf(id))).filter(Optional::isPresent).map(Optional::get) + .map(tabsSI -> { + tabsSI.setNome(nome); + tabsSI.setOperazioni(operazioni); + tabsSI.setIdFlusso(Long.valueOf(idFlusso)); + tabsSI.setLastUpdate(lastU); + log.debug("Changed Information for TabsSI: {}", tabsSI); + return tabsSI; + }); + } + +} diff --git a/src/main/java/it/cnr/isti/epasmed/service/UserService.java b/src/main/java/it/cnr/isti/epasmed/service/UserService.java new file mode 100644 index 0000000..3858e90 --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/service/UserService.java @@ -0,0 +1,303 @@ +package it.cnr.isti.epasmed.service; + +import it.cnr.isti.epasmed.config.Constants; +import it.cnr.isti.epasmed.domain.Authority; +import it.cnr.isti.epasmed.domain.User; +import it.cnr.isti.epasmed.repository.AuthorityRepository; +import it.cnr.isti.epasmed.repository.PersistentTokenRepository; +import it.cnr.isti.epasmed.repository.UserRepository; +import it.cnr.isti.epasmed.security.AuthoritiesConstants; +import it.cnr.isti.epasmed.security.SecurityUtils; +import it.cnr.isti.epasmed.service.dto.UserDTO; + +import io.github.jhipster.security.RandomUtil; + +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; +import org.springframework.data.domain.Page; +import org.springframework.data.domain.Pageable; +import org.springframework.scheduling.annotation.Scheduled; +import org.springframework.security.crypto.password.PasswordEncoder; +import org.springframework.stereotype.Service; +import org.springframework.transaction.annotation.Transactional; + +import java.time.LocalDate; +import java.time.Instant; +import java.time.temporal.ChronoUnit; +import java.util.*; +import java.util.stream.Collectors; + +/** + * Service class for managing users. + */ +@Service +@Transactional("epasMedTransactionManager") +public class UserService { + + private final Logger log = LoggerFactory.getLogger(UserService.class); + + private final UserRepository userRepository; + + private final PasswordEncoder passwordEncoder; + + private final PersistentTokenRepository persistentTokenRepository; + + private final AuthorityRepository authorityRepository; + + public UserService(UserRepository userRepository, PasswordEncoder passwordEncoder, PersistentTokenRepository persistentTokenRepository, AuthorityRepository authorityRepository) { + this.userRepository = userRepository; + this.passwordEncoder = passwordEncoder; + this.persistentTokenRepository = persistentTokenRepository; + this.authorityRepository = authorityRepository; + } + + public Optional activateRegistration(String key) { + log.debug("Activating user for activation key {}", key); + return userRepository.findOneByActivationKey(key) + .map(user -> { + // activate given user for the registration key. + user.setActivated(true); + user.setActivationKey(null); + log.debug("Activated user: {}", user); + return user; + }); + } + + public Optional completePasswordReset(String newPassword, String key) { + log.debug("Reset user password for reset key {}", key); + return userRepository.findOneByResetKey(key) + .filter(user -> user.getResetDate().isAfter(Instant.now().minusSeconds(86400))) + .map(user -> { + user.setPassword(passwordEncoder.encode(newPassword)); + user.setResetKey(null); + user.setResetDate(null); + return user; + }); + } + + public Optional requestPasswordReset(String mail) { + return userRepository.findOneByEmailIgnoreCase(mail) + .filter(User::getActivated) + .map(user -> { + user.setResetKey(RandomUtil.generateResetKey()); + user.setResetDate(Instant.now()); + return user; + }); + } + + public User registerUser(UserDTO userDTO, String password) { + userRepository.findOneByLogin(userDTO.getLogin().toLowerCase()).ifPresent(existingUser -> { + boolean removed = removeNonActivatedUser(existingUser); + if (!removed) { + throw new UsernameAlreadyUsedException(); + } + }); + userRepository.findOneByEmailIgnoreCase(userDTO.getEmail()).ifPresent(existingUser -> { + boolean removed = removeNonActivatedUser(existingUser); + if (!removed) { + throw new EmailAlreadyUsedException(); + } + }); + User newUser = new User(); + String encryptedPassword = passwordEncoder.encode(password); + newUser.setLogin(userDTO.getLogin().toLowerCase()); + // new user gets initially a generated password + newUser.setPassword(encryptedPassword); + newUser.setFirstName(userDTO.getFirstName()); + newUser.setLastName(userDTO.getLastName()); + if (userDTO.getEmail() != null) { + newUser.setEmail(userDTO.getEmail().toLowerCase()); + } + newUser.setImageUrl(userDTO.getImageUrl()); + newUser.setLangKey(userDTO.getLangKey()); + // new user is not active + newUser.setActivated(false); + // new user gets registration key + newUser.setActivationKey(RandomUtil.generateActivationKey()); + Set authorities = new HashSet<>(); + authorityRepository.findById(AuthoritiesConstants.USER).ifPresent(authorities::add); + newUser.setAuthorities(authorities); + userRepository.save(newUser); + log.debug("Created Information for User: {}", newUser); + return newUser; + } + + private boolean removeNonActivatedUser(User existingUser) { + if (existingUser.getActivated()) { + return false; + } + userRepository.delete(existingUser); + userRepository.flush(); + return true; + } + + public User createUser(UserDTO userDTO) { + User user = new User(); + user.setLogin(userDTO.getLogin().toLowerCase()); + user.setFirstName(userDTO.getFirstName()); + user.setLastName(userDTO.getLastName()); + if (userDTO.getEmail() != null) { + user.setEmail(userDTO.getEmail().toLowerCase()); + } + user.setImageUrl(userDTO.getImageUrl()); + if (userDTO.getLangKey() == null) { + user.setLangKey(Constants.DEFAULT_LANGUAGE); // default language + } else { + user.setLangKey(userDTO.getLangKey()); + } + String encryptedPassword = passwordEncoder.encode(RandomUtil.generatePassword()); + user.setPassword(encryptedPassword); + user.setResetKey(RandomUtil.generateResetKey()); + user.setResetDate(Instant.now()); + user.setActivated(true); + if (userDTO.getAuthorities() != null) { + Set authorities = userDTO.getAuthorities().stream() + .map(authorityRepository::findById) + .filter(Optional::isPresent) + .map(Optional::get) + .collect(Collectors.toSet()); + user.setAuthorities(authorities); + } + userRepository.save(user); + log.debug("Created Information for User: {}", user); + return user; + } + + /** + * Update all information for a specific user, and return the modified user. + * + * @param userDTO user to update. + * @return updated user. + */ + public Optional updateUser(UserDTO userDTO) { + return Optional.of(userRepository + .findById(userDTO.getId())) + .filter(Optional::isPresent) + .map(Optional::get) + .map(user -> { + user.setLogin(userDTO.getLogin().toLowerCase()); + user.setFirstName(userDTO.getFirstName()); + user.setLastName(userDTO.getLastName()); + if (userDTO.getEmail() != null) { + user.setEmail(userDTO.getEmail().toLowerCase()); + } + user.setImageUrl(userDTO.getImageUrl()); + user.setActivated(userDTO.isActivated()); + user.setLangKey(userDTO.getLangKey()); + Set managedAuthorities = user.getAuthorities(); + managedAuthorities.clear(); + userDTO.getAuthorities().stream() + .map(authorityRepository::findById) + .filter(Optional::isPresent) + .map(Optional::get) + .forEach(managedAuthorities::add); + log.debug("Changed Information for User: {}", user); + return user; + }) + .map(UserDTO::new); + } + + public void deleteUser(String login) { + userRepository.findOneByLogin(login).ifPresent(user -> { + userRepository.delete(user); + log.debug("Deleted User: {}", user); + }); + } + + /** + * Update basic information (first name, last name, email, language) for the current user. + * + * @param firstName first name of user. + * @param lastName last name of user. + * @param email email id of user. + * @param langKey language key. + * @param imageUrl image URL of user. + */ + public void updateUser(String firstName, String lastName, String email, String langKey, String imageUrl) { + SecurityUtils.getCurrentUserLogin() + .flatMap(userRepository::findOneByLogin) + .ifPresent(user -> { + user.setFirstName(firstName); + user.setLastName(lastName); + if (email != null) { + user.setEmail(email.toLowerCase()); + } + user.setLangKey(langKey); + user.setImageUrl(imageUrl); + log.debug("Changed Information for User: {}", user); + }); + } + + + @Transactional(value="epasMedTransactionManager") + public void changePassword(String currentClearTextPassword, String newPassword) { + SecurityUtils.getCurrentUserLogin() + .flatMap(userRepository::findOneByLogin) + .ifPresent(user -> { + String currentEncryptedPassword = user.getPassword(); + if (!passwordEncoder.matches(currentClearTextPassword, currentEncryptedPassword)) { + throw new InvalidPasswordException(); + } + String encryptedPassword = passwordEncoder.encode(newPassword); + user.setPassword(encryptedPassword); + log.debug("Changed password for User: {}", user); + }); + } + + @Transactional(value="epasMedTransactionManager",readOnly = true) + public Page getAllManagedUsers(Pageable pageable) { + return userRepository.findAllByLoginNot(pageable, Constants.ANONYMOUS_USER).map(UserDTO::new); + } + + @Transactional(value="epasMedTransactionManager",readOnly = true) + public Optional getUserWithAuthoritiesByLogin(String login) { + return userRepository.findOneWithAuthoritiesByLogin(login); + } + + @Transactional(value="epasMedTransactionManager",readOnly = true) + public Optional getUserWithAuthorities() { + return SecurityUtils.getCurrentUserLogin().flatMap(userRepository::findOneWithAuthoritiesByLogin); + } + + /** + * Persistent Token are used for providing automatic authentication, they should be automatically deleted after + * 30 days. + *

+ * This is scheduled to get fired everyday, at midnight. + */ + @Scheduled(cron = "0 0 0 * * ?") + public void removeOldPersistentTokens() { + LocalDate now = LocalDate.now(); + persistentTokenRepository.findByTokenDateBefore(now.minusMonths(1)).forEach(token -> { + log.debug("Deleting token {}", token.getSeries()); + User user = token.getUser(); + user.getPersistentTokens().remove(token); + persistentTokenRepository.delete(token); + }); + } + + /** + * Not activated users should be automatically deleted after 3 days. + *

+ * This is scheduled to get fired everyday, at 01:00 (am). + */ + @Scheduled(cron = "0 0 1 * * ?") + public void removeNotActivatedUsers() { + userRepository + .findAllByActivatedIsFalseAndActivationKeyIsNotNullAndCreatedDateBefore(Instant.now().minus(3, ChronoUnit.DAYS)) + .forEach(user -> { + log.debug("Deleting not activated user {}", user.getLogin()); + userRepository.delete(user); + }); + } + + /** + * Gets a list of all the authorities. + * @return a list of all the authorities. + */ + @Transactional(value="epasMedTransactionManager",readOnly = true) + public List getAuthorities() { + return authorityRepository.findAll().stream().map(Authority::getName).collect(Collectors.toList()); + } + +} diff --git a/src/main/java/it/cnr/isti/epasmed/service/UsernameAlreadyUsedException.java b/src/main/java/it/cnr/isti/epasmed/service/UsernameAlreadyUsedException.java new file mode 100644 index 0000000..a2482c8 --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/service/UsernameAlreadyUsedException.java @@ -0,0 +1,11 @@ +package it.cnr.isti.epasmed.service; + +public class UsernameAlreadyUsedException extends RuntimeException { + + private static final long serialVersionUID = 1L; + + public UsernameAlreadyUsedException() { + super("Login name already used!"); + } + +} diff --git a/src/main/java/it/cnr/isti/epasmed/service/dto/PasswordChangeDTO.java b/src/main/java/it/cnr/isti/epasmed/service/dto/PasswordChangeDTO.java new file mode 100644 index 0000000..b21522b --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/service/dto/PasswordChangeDTO.java @@ -0,0 +1,35 @@ +package it.cnr.isti.epasmed.service.dto; + +/** + * A DTO representing a password change required data - current and new password. + */ +public class PasswordChangeDTO { + private String currentPassword; + private String newPassword; + + public PasswordChangeDTO() { + // Empty constructor needed for Jackson. + } + + public PasswordChangeDTO(String currentPassword, String newPassword) { + this.currentPassword = currentPassword; + this.newPassword = newPassword; + } + + public String getCurrentPassword() { + + return currentPassword; + } + + public void setCurrentPassword(String currentPassword) { + this.currentPassword = currentPassword; + } + + public String getNewPassword() { + return newPassword; + } + + public void setNewPassword(String newPassword) { + this.newPassword = newPassword; + } +} diff --git a/src/main/java/it/cnr/isti/epasmed/service/dto/UserDTO.java b/src/main/java/it/cnr/isti/epasmed/service/dto/UserDTO.java new file mode 100644 index 0000000..22ced16 --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/service/dto/UserDTO.java @@ -0,0 +1,197 @@ +package it.cnr.isti.epasmed.service.dto; + +import it.cnr.isti.epasmed.config.Constants; + +import it.cnr.isti.epasmed.domain.Authority; +import it.cnr.isti.epasmed.domain.User; + +import javax.validation.constraints.*; +import java.time.Instant; +import java.util.Set; +import java.util.stream.Collectors; + +/** + * A DTO representing a user, with his authorities. + */ +public class UserDTO { + + private Long id; + + @NotBlank + @Pattern(regexp = Constants.LOGIN_REGEX) + @Size(min = 1, max = 50) + private String login; + + @Size(max = 50) + private String firstName; + + @Size(max = 50) + private String lastName; + + @Email + @Size(min = 5, max = 254) + private String email; + + @Size(max = 256) + private String imageUrl; + + private boolean activated = false; + + @Size(min = 2, max = 10) + private String langKey; + + private String createdBy; + + private Instant createdDate; + + private String lastModifiedBy; + + private Instant lastModifiedDate; + + private Set authorities; + + public UserDTO() { + // Empty constructor needed for Jackson. + } + + public UserDTO(User user) { + this.id = user.getId(); + this.login = user.getLogin(); + this.firstName = user.getFirstName(); + this.lastName = user.getLastName(); + this.email = user.getEmail(); + this.activated = user.getActivated(); + this.imageUrl = user.getImageUrl(); + this.langKey = user.getLangKey(); + this.createdBy = user.getCreatedBy(); + this.createdDate = user.getCreatedDate(); + this.lastModifiedBy = user.getLastModifiedBy(); + this.lastModifiedDate = user.getLastModifiedDate(); + this.authorities = user.getAuthorities().stream() + .map(Authority::getName) + .collect(Collectors.toSet()); + } + + public Long getId() { + return id; + } + + public void setId(Long id) { + this.id = id; + } + + public String getLogin() { + return login; + } + + public void setLogin(String login) { + this.login = login; + } + + public String getFirstName() { + return firstName; + } + + public void setFirstName(String firstName) { + this.firstName = firstName; + } + + public String getLastName() { + return lastName; + } + + public void setLastName(String lastName) { + this.lastName = lastName; + } + + public String getEmail() { + return email; + } + + public void setEmail(String email) { + this.email = email; + } + + public String getImageUrl() { + return imageUrl; + } + + public void setImageUrl(String imageUrl) { + this.imageUrl = imageUrl; + } + + public boolean isActivated() { + return activated; + } + + public void setActivated(boolean activated) { + this.activated = activated; + } + + public String getLangKey() { + return langKey; + } + + public void setLangKey(String langKey) { + this.langKey = langKey; + } + + public String getCreatedBy() { + return createdBy; + } + + public void setCreatedBy(String createdBy) { + this.createdBy = createdBy; + } + + public Instant getCreatedDate() { + return createdDate; + } + + public void setCreatedDate(Instant createdDate) { + this.createdDate = createdDate; + } + + public String getLastModifiedBy() { + return lastModifiedBy; + } + + public void setLastModifiedBy(String lastModifiedBy) { + this.lastModifiedBy = lastModifiedBy; + } + + public Instant getLastModifiedDate() { + return lastModifiedDate; + } + + public void setLastModifiedDate(Instant lastModifiedDate) { + this.lastModifiedDate = lastModifiedDate; + } + + public Set getAuthorities() { + return authorities; + } + + public void setAuthorities(Set authorities) { + this.authorities = authorities; + } + + // prettier-ignore + @Override + public String toString() { + return "UserDTO{" + + "login='" + login + '\'' + + ", firstName='" + firstName + '\'' + + ", lastName='" + lastName + '\'' + + ", email='" + email + '\'' + + ", imageUrl='" + imageUrl + '\'' + + ", activated=" + activated + + ", langKey='" + langKey + '\'' + + ", createdBy=" + createdBy + + ", createdDate=" + createdDate + + ", lastModifiedBy='" + lastModifiedBy + '\'' + + ", lastModifiedDate=" + lastModifiedDate + + ", authorities=" + authorities + + "}"; + } +} diff --git a/src/main/java/it/cnr/isti/epasmed/service/dto/package-info.java b/src/main/java/it/cnr/isti/epasmed/service/dto/package-info.java new file mode 100644 index 0000000..48c59af --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/service/dto/package-info.java @@ -0,0 +1,4 @@ +/** + * Data Transfer Objects. + */ +package it.cnr.isti.epasmed.service.dto; diff --git a/src/main/java/it/cnr/isti/epasmed/service/mapper/UserMapper.java b/src/main/java/it/cnr/isti/epasmed/service/mapper/UserMapper.java new file mode 100644 index 0000000..f460f6f --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/service/mapper/UserMapper.java @@ -0,0 +1,81 @@ +package it.cnr.isti.epasmed.service.mapper; + +import it.cnr.isti.epasmed.domain.Authority; +import it.cnr.isti.epasmed.domain.User; +import it.cnr.isti.epasmed.service.dto.UserDTO; + +import org.springframework.stereotype.Service; + +import java.util.*; +import java.util.stream.Collectors; + +/** + * Mapper for the entity {@link User} and its DTO called {@link UserDTO}. + * + * Normal mappers are generated using MapStruct, this one is hand-coded as MapStruct + * support is still in beta, and requires a manual step with an IDE. + */ +@Service +public class UserMapper { + + public List usersToUserDTOs(List users) { + return users.stream() + .filter(Objects::nonNull) + .map(this::userToUserDTO) + .collect(Collectors.toList()); + } + + public UserDTO userToUserDTO(User user) { + return new UserDTO(user); + } + + public List userDTOsToUsers(List userDTOs) { + return userDTOs.stream() + .filter(Objects::nonNull) + .map(this::userDTOToUser) + .collect(Collectors.toList()); + } + + public User userDTOToUser(UserDTO userDTO) { + if (userDTO == null) { + return null; + } else { + User user = new User(); + user.setId(userDTO.getId()); + user.setLogin(userDTO.getLogin()); + user.setFirstName(userDTO.getFirstName()); + user.setLastName(userDTO.getLastName()); + user.setEmail(userDTO.getEmail()); + user.setImageUrl(userDTO.getImageUrl()); + user.setActivated(userDTO.isActivated()); + user.setLangKey(userDTO.getLangKey()); + Set authorities = this.authoritiesFromStrings(userDTO.getAuthorities()); + user.setAuthorities(authorities); + return user; + } + } + + + private Set authoritiesFromStrings(Set authoritiesAsString) { + Set authorities = new HashSet<>(); + + if (authoritiesAsString != null) { + authorities = authoritiesAsString.stream().map(string -> { + Authority auth = new Authority(); + auth.setName(string); + return auth; + }).collect(Collectors.toSet()); + } + + return authorities; + } + + public User userFromId(Long id) { + if (id == null) { + return null; + } + User user = new User(); + user.setId(id); + return user; + } +} diff --git a/src/main/java/it/cnr/isti/epasmed/service/mapper/package-info.java b/src/main/java/it/cnr/isti/epasmed/service/mapper/package-info.java new file mode 100644 index 0000000..1a88883 --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/service/mapper/package-info.java @@ -0,0 +1,4 @@ +/** + * MapStruct mappers for mapping domain objects and Data Transfer Objects. + */ +package it.cnr.isti.epasmed.service.mapper; diff --git a/src/main/java/it/cnr/isti/epasmed/service/package-info.java b/src/main/java/it/cnr/isti/epasmed/service/package-info.java new file mode 100644 index 0000000..efe60f2 --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/service/package-info.java @@ -0,0 +1,4 @@ +/** + * Service layer beans. + */ +package it.cnr.isti.epasmed.service; diff --git a/src/main/java/it/cnr/isti/epasmed/sistemainformativo/model/SIAnagrafico.java b/src/main/java/it/cnr/isti/epasmed/sistemainformativo/model/SIAnagrafico.java new file mode 100644 index 0000000..3b9f722 --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/sistemainformativo/model/SIAnagrafico.java @@ -0,0 +1,30 @@ +package it.cnr.isti.epasmed.sistemainformativo.model; + +import java.io.Serializable; +import java.sql.Timestamp; +import java.util.Date; + +import lombok.AllArgsConstructor; +import lombok.Data; +import lombok.NoArgsConstructor; + +@Data +@AllArgsConstructor +@NoArgsConstructor +public class SIAnagrafico implements Serializable { + + private static final long serialVersionUID = 1L; + private String codicefiscale; + private String cognome; + private String nome; + private Date datanascita; + private Integer numbadge; + private Timestamp data_mod; + private String flag_del; + private Long id_flusso; + private Integer id; + private Integer matricola; +} + + + diff --git a/src/main/java/it/cnr/isti/epasmed/sistemainformativo/model/SIEmail.java b/src/main/java/it/cnr/isti/epasmed/sistemainformativo/model/SIEmail.java new file mode 100644 index 0000000..0992267 --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/sistemainformativo/model/SIEmail.java @@ -0,0 +1,25 @@ +package it.cnr.isti.epasmed.sistemainformativo.model; + +import java.io.Serializable; +import java.sql.Timestamp; + +import lombok.AllArgsConstructor; +import lombok.Data; +import lombok.NoArgsConstructor; + +@Data +@AllArgsConstructor +@NoArgsConstructor +public class SIEmail implements Serializable { + + private static final long serialVersionUID = 1L; + private Integer id; + private Integer idpersona; + private String cf; + private String email; + private String tipo; + private Timestamp data_mod; + private String flag_del; + private Long id_flusso; + +} diff --git a/src/main/java/it/cnr/isti/epasmed/sistemainformativo/model/SIGruppi.java b/src/main/java/it/cnr/isti/epasmed/sistemainformativo/model/SIGruppi.java new file mode 100644 index 0000000..9b30514 --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/sistemainformativo/model/SIGruppi.java @@ -0,0 +1,23 @@ +package it.cnr.isti.epasmed.sistemainformativo.model; + +import java.io.Serializable; +import java.sql.Timestamp; + +import lombok.AllArgsConstructor; +import lombok.Data; +import lombok.NoArgsConstructor; + +@Data +@AllArgsConstructor +@NoArgsConstructor +public class SIGruppi implements Serializable { + + private static final long serialVersionUID = 1L; + private Integer id; + private String sigla; + private String descrizione; + private Timestamp data_mod; + private String flag_del; + private Long id_flusso; + +} diff --git a/src/main/java/it/cnr/isti/epasmed/sistemainformativo/model/SIGruppiPers.java b/src/main/java/it/cnr/isti/epasmed/sistemainformativo/model/SIGruppiPers.java new file mode 100644 index 0000000..e73df0e --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/sistemainformativo/model/SIGruppiPers.java @@ -0,0 +1,30 @@ +package it.cnr.isti.epasmed.sistemainformativo.model; + +import java.io.Serializable; +import java.sql.Timestamp; +import java.util.Date; + +import lombok.AllArgsConstructor; +import lombok.Data; +import lombok.NoArgsConstructor; + +@Data +@AllArgsConstructor +@NoArgsConstructor +public class SIGruppiPers implements Serializable { + + private static final long serialVersionUID = 1L; + private Integer id; + private Integer idgruppo; + private Integer idpersona; + private String cf; + private String gruppo; + private Date dal; + private Date al; + private Integer percentuale; + private Timestamp data_mod; + private String flag_del; + private Long id_flusso; + + +} diff --git a/src/main/java/it/cnr/isti/epasmed/sistemainformativo/model/SIPosizioni.java b/src/main/java/it/cnr/isti/epasmed/sistemainformativo/model/SIPosizioni.java new file mode 100644 index 0000000..f5f0b1a --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/sistemainformativo/model/SIPosizioni.java @@ -0,0 +1,35 @@ +package it.cnr.isti.epasmed.sistemainformativo.model; + +import java.io.Serializable; +import java.sql.Timestamp; +import java.util.Date; + +import lombok.AllArgsConstructor; +import lombok.Data; +import lombok.NoArgsConstructor; + +@Data +@AllArgsConstructor +@NoArgsConstructor +public class SIPosizioni implements Serializable { + + private static final long serialVersionUID = 1L; + private Integer id; + private Integer idpersona; + private String cf; + private Date dal; + private Date al; + private Integer matricola; + private String profilo; + private String subprofilo; + private Integer livello; + private String tipocontratto; + private String motivofinerapportoti; + private Integer fasciaimportoassegno; + private Timestamp data_mod; + private String flag_del; + private Long id_flusso; + private Integer finerapportoti; + + +} diff --git a/src/main/java/it/cnr/isti/epasmed/sistemainformativo/model/SIProroghe.java b/src/main/java/it/cnr/isti/epasmed/sistemainformativo/model/SIProroghe.java new file mode 100644 index 0000000..6fc9cd4 --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/sistemainformativo/model/SIProroghe.java @@ -0,0 +1,29 @@ +package it.cnr.isti.epasmed.sistemainformativo.model; + +import java.io.Serializable; +import java.sql.Timestamp; +import java.util.Date; + +import lombok.AllArgsConstructor; +import lombok.Data; +import lombok.NoArgsConstructor; + +@Data +@AllArgsConstructor +@NoArgsConstructor +public class SIProroghe implements Serializable { + + private static final long serialVersionUID = 1L; + private Integer id; + private Integer idposizione; + private String cf; + private Date posizione_dal; + private Date proroga_al; + private Timestamp data_mod; + private String flag_del; + private Long id_flusso; + private Date posizione_al; + private Date proroga_dal; + + +} diff --git a/src/main/java/it/cnr/isti/epasmed/sistemainformativo/model/SITelefoni.java b/src/main/java/it/cnr/isti/epasmed/sistemainformativo/model/SITelefoni.java new file mode 100644 index 0000000..d7e6af8 --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/sistemainformativo/model/SITelefoni.java @@ -0,0 +1,25 @@ +package it.cnr.isti.epasmed.sistemainformativo.model; + +import java.io.Serializable; +import java.sql.Timestamp; + +import lombok.AllArgsConstructor; +import lombok.Data; +import lombok.NoArgsConstructor; + +@Data +@AllArgsConstructor +@NoArgsConstructor +public class SITelefoni implements Serializable { + + private static final long serialVersionUID = 1L; + private Integer id; + private Integer idpersona; + private String cf; + private String tiporecapitotelefonico; + private String recapitotelefonico; + private Timestamp data_mod; + private String flag_del; + private Long id_flusso; + +} diff --git a/src/main/java/it/cnr/isti/epasmed/sistemainformativo/repository/SIAnagraficoRepository.java b/src/main/java/it/cnr/isti/epasmed/sistemainformativo/repository/SIAnagraficoRepository.java new file mode 100644 index 0000000..ebb2a3b --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/sistemainformativo/repository/SIAnagraficoRepository.java @@ -0,0 +1,71 @@ +package it.cnr.isti.epasmed.sistemainformativo.repository; + +import java.sql.ResultSet; +import java.sql.SQLException; +import java.util.List; + +import javax.sql.DataSource; + +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; +import org.springframework.beans.factory.annotation.Qualifier; +import org.springframework.jdbc.core.JdbcTemplate; +import org.springframework.jdbc.core.RowMapper; +import org.springframework.stereotype.Repository; + +import it.cnr.isti.epasmed.sistemainformativo.model.SIAnagrafico; + +@Repository +public class SIAnagraficoRepository { + + private final Logger logger = LoggerFactory.getLogger(SIAnagraficoRepository.class); + + private JdbcTemplate jdbcTemplate; + + // + public SIAnagraficoRepository(final @Qualifier("sistemaInformativoDataSource") DataSource dataSource) { + super(); + this.jdbcTemplate = new JdbcTemplate(dataSource); + } + + public List readNewFlux(Long flusso) { + logger.info("Read Flux > {}",flusso); + String query="select codicefiscale, cognome, nome, datanascita," + + " numbadge, data_mod, flag_del, id_flusso, id " + + " from anagrafico " + + " where id_flusso > " + + flusso + + " and ( " + + " exists( " + + " select 1 " + + " from posizioni p " + + " where p.cf::text=codicefiscale and p.tipocontratto='Dipendente') " + + " or " + + " exists( " + + " select 1 " + + " from posizioni ps " + + " where ps.cf::text=codicefiscale and ps.profilo='Volontario Servizio Civile') " + + " )" + " order by codicefiscale, id_flusso, data_mod"; + + List anagrafica= jdbcTemplate.query(query, new RowMapper() { + @Override + public SIAnagrafico mapRow(ResultSet rs, int rowNum) throws SQLException { + SIAnagrafico a=new SIAnagrafico(); + a.setCodicefiscale(rs.getString(1)); + a.setCognome(rs.getString(2)); + a.setNome(rs.getString(3)); + a.setDatanascita(rs.getDate(4)); + a.setNumbadge(rs.getInt(5)); + a.setData_mod(rs.getTimestamp(6)); + a.setFlag_del(rs.getString(7)); + a.setId_flusso(rs.getLong(8)); + a.setId(rs.getInt(9)); + return a; + }}); + return anagrafica; + + } + + + +} diff --git a/src/main/java/it/cnr/isti/epasmed/sistemainformativo/repository/SIEmailRepository.java b/src/main/java/it/cnr/isti/epasmed/sistemainformativo/repository/SIEmailRepository.java new file mode 100644 index 0000000..6b1fe67 --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/sistemainformativo/repository/SIEmailRepository.java @@ -0,0 +1,59 @@ +package it.cnr.isti.epasmed.sistemainformativo.repository; + +import java.sql.ResultSet; +import java.sql.SQLException; +import java.util.List; + +import javax.sql.DataSource; + +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; +import org.springframework.beans.factory.annotation.Qualifier; +import org.springframework.jdbc.core.JdbcTemplate; +import org.springframework.jdbc.core.RowMapper; +import org.springframework.stereotype.Repository; + +import it.cnr.isti.epasmed.sistemainformativo.model.SIEmail; + +@Repository +public class SIEmailRepository { + + private final Logger logger = LoggerFactory.getLogger(SIEmailRepository.class); + + private JdbcTemplate jdbcTemplate; + + // + public SIEmailRepository(final @Qualifier("sistemaInformativoDataSource") DataSource dataSource) { + super(); + this.jdbcTemplate = new JdbcTemplate(dataSource); + } + + public List readNewFlux(Long flusso) { + logger.info("Read Flux > {}",flusso); + String query="select cf, email, tipoemail, data_mod, flag_del, id_flusso, id, idpersona " + + " from mail " + + " WHERE id_flusso > " + + flusso + " order by cf, id_flusso, data_mod"; + + List emails= jdbcTemplate.query(query, new RowMapper() { + @Override + public SIEmail mapRow(ResultSet rs, int rowNum) throws SQLException { + SIEmail m = new SIEmail(); + m.setCf(rs.getString(1)); + m.setEmail(rs.getString(2)); + m.setTipo(rs.getString(3)); + m.setData_mod(rs.getTimestamp(4)); + m.setFlag_del(rs.getString(5)); + m.setId_flusso(rs.getLong(6)); + m.setId(rs.getInt(7)); + m.setIdpersona(rs.getInt(8)); + + return m; + }}); + return emails; + + } + + + +} diff --git a/src/main/java/it/cnr/isti/epasmed/sistemainformativo/repository/SIGruppiPersRepository.java b/src/main/java/it/cnr/isti/epasmed/sistemainformativo/repository/SIGruppiPersRepository.java new file mode 100644 index 0000000..88dc95d --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/sistemainformativo/repository/SIGruppiPersRepository.java @@ -0,0 +1,63 @@ +package it.cnr.isti.epasmed.sistemainformativo.repository; + +import java.sql.ResultSet; +import java.sql.SQLException; +import java.util.List; + +import javax.sql.DataSource; + +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; +import org.springframework.beans.factory.annotation.Qualifier; +import org.springframework.jdbc.core.JdbcTemplate; +import org.springframework.jdbc.core.RowMapper; +import org.springframework.stereotype.Repository; + +import it.cnr.isti.epasmed.sistemainformativo.model.SIGruppiPers; + +@Repository +public class SIGruppiPersRepository { + + private final Logger logger = LoggerFactory.getLogger(SIGruppiPersRepository.class); + + private JdbcTemplate jdbcTemplate; + + // + public SIGruppiPersRepository(final @Qualifier("sistemaInformativoDataSource") DataSource dataSource) { + super(); + this.jdbcTemplate = new JdbcTemplate(dataSource); + } + + public List readNewFlux(Long flusso) { + logger.info("Read Flux > {}",flusso); + String query="select cf, gruppo, dal, al, percentuale, data_mod, flag_del, id_flusso, id, idpersona, idgruppo " + + " from gruppo_pers " + + " WHERE id_flusso > " + + flusso + + " order by cf, gruppo, dal, al, id_flusso, data_mod"; + + List gruppi= jdbcTemplate.query(query, new RowMapper() { + @Override + public SIGruppiPers mapRow(ResultSet rs, int rowNum) throws SQLException { + SIGruppiPers gp = new SIGruppiPers(); + gp.setCf(rs.getString(1)); + gp.setGruppo(rs.getString(2)); + gp.setDal(rs.getDate(3)); + gp.setAl(rs.getDate(4)); + gp.setPercentuale(rs.getInt(5)); + gp.setData_mod(rs.getTimestamp(6)); + gp.setFlag_del(rs.getString(7)); + gp.setId_flusso(rs.getLong(8)); + gp.setId(rs.getInt(9)); + gp.setIdpersona(rs.getInt(10)); + gp.setIdgruppo(rs.getInt(11)); + + return gp; + }}); + return gruppi; + + } + + + +} diff --git a/src/main/java/it/cnr/isti/epasmed/sistemainformativo/repository/SIGruppiRepository.java b/src/main/java/it/cnr/isti/epasmed/sistemainformativo/repository/SIGruppiRepository.java new file mode 100644 index 0000000..fb7078c --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/sistemainformativo/repository/SIGruppiRepository.java @@ -0,0 +1,57 @@ +package it.cnr.isti.epasmed.sistemainformativo.repository; + +import java.sql.ResultSet; +import java.sql.SQLException; +import java.util.List; + +import javax.sql.DataSource; + +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; +import org.springframework.beans.factory.annotation.Qualifier; +import org.springframework.jdbc.core.JdbcTemplate; +import org.springframework.jdbc.core.RowMapper; +import org.springframework.stereotype.Repository; + +import it.cnr.isti.epasmed.sistemainformativo.model.SIGruppi; + +@Repository +public class SIGruppiRepository { + + private final Logger logger = LoggerFactory.getLogger(SIGruppiRepository.class); + + private JdbcTemplate jdbcTemplate; + + // + public SIGruppiRepository(final @Qualifier("sistemaInformativoDataSource") DataSource dataSource) { + super(); + this.jdbcTemplate = new JdbcTemplate(dataSource); + } + + public List readNewFlux(Long flusso) { + logger.info("Read Flux > {}",flusso); + String query="select sigla, descrizione, data_mod, flag_del, id_flusso, id " + + " from gruppi " + + " WHERE id_flusso > " + + flusso + + " order by sigla, id_flusso, data_mod"; + + List gruppi= jdbcTemplate.query(query, new RowMapper() { + @Override + public SIGruppi mapRow(ResultSet rs, int rowNum) throws SQLException { + SIGruppi g = new SIGruppi(); + g.setSigla(rs.getString(1)); + g.setDescrizione(rs.getString(2)); + g.setData_mod(rs.getTimestamp(3)); + g.setFlag_del(rs.getString(4)); + g.setId_flusso(rs.getLong(5)); + g.setId(rs.getInt(6)); + return g; + }}); + return gruppi; + + } + + + +} diff --git a/src/main/java/it/cnr/isti/epasmed/sistemainformativo/repository/SIMasterLogRepository.java b/src/main/java/it/cnr/isti/epasmed/sistemainformativo/repository/SIMasterLogRepository.java new file mode 100644 index 0000000..f32b1a6 --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/sistemainformativo/repository/SIMasterLogRepository.java @@ -0,0 +1,124 @@ +package it.cnr.isti.epasmed.sistemainformativo.repository; + +import java.sql.ResultSet; +import java.sql.SQLException; +import java.util.List; + +import javax.sql.DataSource; + +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; +import org.springframework.beans.factory.annotation.Qualifier; +import org.springframework.jdbc.core.JdbcTemplate; +import org.springframework.jdbc.core.RowMapper; +import org.springframework.stereotype.Repository; + +@Repository +public class SIMasterLogRepository { + + private final Logger log = LoggerFactory.getLogger(SIMasterLogRepository.class); + + private JdbcTemplate jdbcTemplate; + + // + public SIMasterLogRepository(final @Qualifier("sistemaInformativoDataSource") DataSource dataSource) { + super(); + this.jdbcTemplate = new JdbcTemplate(dataSource); + // this.dataSource = dataSource; + // logger.debug("MasterLogRepository DataSource set: "+dataSource.toString()); + } + + public int count() { + log.debug("Call MasterLogRepository Count()"); + // return 1000; + return jdbcTemplate.queryForObject("select count(*) from master_log", Integer.class); + } + + public void startFluxReads() { + log.info("Iniziato Flusso in lettura"); + jdbcTemplate.update("INSERT INTO master_log (codice_flusso, tabelle, operazione, data_inizio_oper) " + + "VALUES ('epas','anagrafico,mail,telefoni,gruppi','L',current_timestamp)"); + log.info("Flusso in lettura aperto"); + + } + + public void closeFluxReads(String tabelleLette) throws Exception { + log.info("Chiusura Flusso in lettura"); + log.info("Tabelle Lette: {}", tabelleLette); + if (tabelleLette.isEmpty()) { + tabelleLette = " "; + } + + String query = "SELECT id_flusso FROM master_log" + + " WHERE codice_flusso='epas' AND operazione='L' AND data_fine_oper IS NULL" + + " ORDER BY id_flusso DESC LIMIT 1"; + + List idsFlux = jdbcTemplate.query(query, new RowMapper() { + @Override + public Long mapRow(ResultSet rs, int rowNum) throws SQLException { + Long idFlusso = rs.getLong("id_flusso"); + return idFlusso; + } + }); + log.info("Flusso da chiudere: {}", idsFlux); + + if (idsFlux != null && !idsFlux.isEmpty()) { + + jdbcTemplate.update("UPDATE master_log SET data_fine_oper=current_timestamp, tabelle='" + tabelleLette + + "', esito_oper='OK' WHERE codice_flusso='epas' AND id_flusso=" + idsFlux.get(0) + " "); + + log.info("Flusso in lettura chiuso"); + } else { + log.error("Flusso da chiudere non trovato: {}", idsFlux); + throw new Exception("Errore chiudendo il flusso di lettura, id: "+idsFlux); + + } + } + + public Long startFluxWrites() throws Exception { + log.info("Iniziato Flusso in scrittura"); + jdbcTemplate.update("INSERT INTO master_log (codice_flusso, tabelle, operazione, data_inizio_oper) \"\n" + + " + \"VALUES ('epas','cartellini, cartellini_rendicontazioni, festivita, pers_orario, lavoro_fuori_sede','S',current_timestamp)"); + log.info("Flusso in scrittura aperto"); + + String query = "SELECT id_flusso FROM master_log" + " WHERE codice_flusso='epas' AND operazione='S'" + + " AND data_fine_oper IS NULL" + " ORDER BY id_flusso DESC"; + + List idsFlux = jdbcTemplate.query(query, new RowMapper() { + @Override + public Long mapRow(ResultSet rs, int rowNum) throws SQLException { + Long idFlusso = rs.getLong("id_flusso"); + return idFlusso; + } + }); + Long idFlux = 0l; + if (idsFlux != null && !idsFlux.isEmpty()) { + idFlux = idsFlux.get(0); + log.info("Flusso in scrittura id: {}", idFlux); + } else { + log.error("Errore inizializando il flusso di scrittura, id: {}",idsFlux); + throw new Exception("Errore inizializando il flusso di scrittura, id: "+idsFlux); + } + + return idFlux; + } + + public void closeFluxWrites(Long idFlux, String tabelleScritte) { + log.info("Chiusura Flusso in scrittura"); + log.info("Tabelle Scritte: {}", tabelleScritte); + if (tabelleScritte.isEmpty()) { + tabelleScritte = " "; + } + + if (idFlux != null && idFlux.longValue() != 0) { + + jdbcTemplate.update("UPDATE master_log SET data_fine_oper=current_timestamp, tabelle='" + tabelleScritte + + "', esito_oper='OK' WHERE codice_flusso='epas' AND id_flusso=" + idFlux + " "); + + log.info("Flusso in scrittura chiuso"); + } else { + log.info("Flusso in scrittura da chiudere non trovato: {}", idFlux); + } + } + +} diff --git a/src/main/java/it/cnr/isti/epasmed/sistemainformativo/repository/SIPosizioniRepository.java b/src/main/java/it/cnr/isti/epasmed/sistemainformativo/repository/SIPosizioniRepository.java new file mode 100644 index 0000000..c6bcbd7 --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/sistemainformativo/repository/SIPosizioniRepository.java @@ -0,0 +1,69 @@ +package it.cnr.isti.epasmed.sistemainformativo.repository; + +import java.sql.ResultSet; +import java.sql.SQLException; +import java.util.List; + +import javax.sql.DataSource; + +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; +import org.springframework.beans.factory.annotation.Qualifier; +import org.springframework.jdbc.core.JdbcTemplate; +import org.springframework.jdbc.core.RowMapper; +import org.springframework.stereotype.Repository; + +import it.cnr.isti.epasmed.sistemainformativo.model.SIPosizioni; + +@Repository +public class SIPosizioniRepository { + + private final Logger logger = LoggerFactory.getLogger(SIPosizioniRepository.class); + + private JdbcTemplate jdbcTemplate; + + // + public SIPosizioniRepository(final @Qualifier("sistemaInformativoDataSource") DataSource dataSource) { + super(); + this.jdbcTemplate = new JdbcTemplate(dataSource); + } + + public List readNewFlux(Long flusso) { + logger.info("Read Flux > {}",flusso); + String query="select cf, dal, al, matricola, profilo, subprofilo, livello, tipocontratto, " + + " motivofinerapportoti, fasciaimportoassegno, data_mod, flag_del,id_flusso, finerapportoti, idpersona, id " + + " from posizioni " + + " where tipocontratto='Dipendente' and id_flusso > " + + flusso + + " order by id_flusso, data_mod, id, cf, dal, al"; + + List posizioni= jdbcTemplate.query(query, new RowMapper() { + @Override + public SIPosizioni mapRow(ResultSet rs, int rowNum) throws SQLException { + SIPosizioni pos = new SIPosizioni(); + pos.setCf(rs.getString(1)); + pos.setDal(rs.getDate(2)); + pos.setAl(rs.getDate(3)); + pos.setMatricola(rs.getInt(4)); + pos.setProfilo(rs.getString(5)); + pos.setSubprofilo(rs.getString(6)); + pos.setLivello(rs.getInt(7)); + pos.setTipocontratto(rs.getString(8)); + pos.setMotivofinerapportoti(rs.getString(9)); + pos.setFasciaimportoassegno(rs.getInt(10)); + pos.setData_mod(rs.getTimestamp(11)); + pos.setFlag_del(rs.getString(12)); + pos.setId_flusso(rs.getLong(13)); + pos.setFinerapportoti(rs.getInt(14)); + pos.setIdpersona(rs.getInt(15)); + pos.setId(rs.getInt(16)); + + return pos; + }}); + return posizioni; + + } + + + +} diff --git a/src/main/java/it/cnr/isti/epasmed/sistemainformativo/repository/SIProrogheRepository.java b/src/main/java/it/cnr/isti/epasmed/sistemainformativo/repository/SIProrogheRepository.java new file mode 100644 index 0000000..7642345 --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/sistemainformativo/repository/SIProrogheRepository.java @@ -0,0 +1,63 @@ +package it.cnr.isti.epasmed.sistemainformativo.repository; + +import java.sql.ResultSet; +import java.sql.SQLException; +import java.util.List; + +import javax.sql.DataSource; + +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; +import org.springframework.beans.factory.annotation.Qualifier; +import org.springframework.jdbc.core.JdbcTemplate; +import org.springframework.jdbc.core.RowMapper; +import org.springframework.stereotype.Repository; + +import it.cnr.isti.epasmed.sistemainformativo.model.SIProroghe; + +@Repository +public class SIProrogheRepository { + + private final Logger logger = LoggerFactory.getLogger(SIProrogheRepository.class); + + private JdbcTemplate jdbcTemplate; + + // + public SIProrogheRepository(final @Qualifier("sistemaInformativoDataSource") DataSource dataSource) { + super(); + this.jdbcTemplate = new JdbcTemplate(dataSource); + } + + public List readNewFlux(Long flusso) { + logger.info("Read Flux > {}",flusso); + String query="select cf,posizione_dal, proroga_al, data_mod,flag_del,id_flusso, " + + " posizione_al, proroga_dal, id, idposizione" + + " from proroghe " + + " where id_flusso > " + + flusso + + " order by id_flusso, data_mod, id, cf, posizione_dal, posizione_al, proroga_dal, proroga_al"; + + List proroghe= jdbcTemplate.query(query, new RowMapper() { + @Override + public SIProroghe mapRow(ResultSet rs, int rowNum) throws SQLException { + SIProroghe pro = new SIProroghe(); + pro.setCf(rs.getString(1)); + pro.setPosizione_dal(rs.getDate(2)); + pro.setProroga_al(rs.getDate(3)); + pro.setData_mod(rs.getTimestamp(4)); + pro.setFlag_del(rs.getString(5)); + pro.setId_flusso(rs.getLong(6)); + pro.setPosizione_al(rs.getDate(7)); + pro.setProroga_dal(rs.getDate(8)); + pro.setId(rs.getInt(9)); + pro.setIdposizione(rs.getInt(10)); + + return pro; + }}); + return proroghe; + + } + + + +} diff --git a/src/main/java/it/cnr/isti/epasmed/sistemainformativo/repository/SITelefoniRepository.java b/src/main/java/it/cnr/isti/epasmed/sistemainformativo/repository/SITelefoniRepository.java new file mode 100644 index 0000000..940b994 --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/sistemainformativo/repository/SITelefoniRepository.java @@ -0,0 +1,59 @@ +package it.cnr.isti.epasmed.sistemainformativo.repository; + +import java.sql.ResultSet; +import java.sql.SQLException; +import java.util.List; + +import javax.sql.DataSource; + +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; +import org.springframework.beans.factory.annotation.Qualifier; +import org.springframework.jdbc.core.JdbcTemplate; +import org.springframework.jdbc.core.RowMapper; +import org.springframework.stereotype.Repository; + +import it.cnr.isti.epasmed.sistemainformativo.model.SITelefoni; + +@Repository +public class SITelefoniRepository { + + private final Logger logger = LoggerFactory.getLogger(SITelefoniRepository.class); + + private JdbcTemplate jdbcTemplate; + + // + public SITelefoniRepository(final @Qualifier("sistemaInformativoDataSource") DataSource dataSource) { + super(); + this.jdbcTemplate = new JdbcTemplate(dataSource); + } + + public List readNewFlux(Long flusso) { + logger.info("Read Flux > {}",flusso); + String query="select cf, tiporecapitotelefonico, recapitotelefonico, data_mod, flag_del, id_flusso, id, idpersona " + + " from telefoni " + + " WHERE id_flusso > " + + flusso + " order by cf, id_flusso, data_mod"; + + List telefoni= jdbcTemplate.query(query, new RowMapper() { + @Override + public SITelefoni mapRow(ResultSet rs, int rowNum) throws SQLException { + SITelefoni t = new SITelefoni(); + t.setCf(rs.getString(1)); + t.setTiporecapitotelefonico(rs.getString(2)); + t.setRecapitotelefonico(rs.getString(3)); + t.setData_mod(rs.getTimestamp(4)); + t.setFlag_del(rs.getString(5)); + t.setId_flusso(rs.getLong(6)); + t.setId(rs.getInt(7)); + t.setIdpersona(rs.getInt(8)); + + return t; + }}); + return telefoni; + + } + + + +} diff --git a/src/main/java/it/cnr/isti/epasmed/sistemainformativo/service/SIAnagraficoService.java b/src/main/java/it/cnr/isti/epasmed/sistemainformativo/service/SIAnagraficoService.java new file mode 100644 index 0000000..498f1a0 --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/sistemainformativo/service/SIAnagraficoService.java @@ -0,0 +1,37 @@ +package it.cnr.isti.epasmed.sistemainformativo.service; + +import java.util.List; + +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; +import org.springframework.stereotype.Service; +import org.springframework.transaction.annotation.Transactional; + +import it.cnr.isti.epasmed.sistemainformativo.model.SIAnagrafico; +import it.cnr.isti.epasmed.sistemainformativo.repository.SIAnagraficoRepository; + + + +/** + * Service class for managing Anagrafico. + */ +@Service +@Transactional("sistemaInformativoTransactionManager") +public class SIAnagraficoService { + + private final Logger log = LoggerFactory.getLogger(SIAnagraficoService.class); + + private final SIAnagraficoRepository anagraficoRepository; + + public SIAnagraficoService(SIAnagraficoRepository anagraficoRepository) { + super(); + this.anagraficoRepository = anagraficoRepository; + } + + public List readNewFlux(Long flusso) { + log.debug("Request Anagrafico Flux > {}",flusso); + return anagraficoRepository.readNewFlux(flusso); + } + + +} \ No newline at end of file diff --git a/src/main/java/it/cnr/isti/epasmed/sistemainformativo/service/SIEmailService.java b/src/main/java/it/cnr/isti/epasmed/sistemainformativo/service/SIEmailService.java new file mode 100644 index 0000000..5efac2b --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/sistemainformativo/service/SIEmailService.java @@ -0,0 +1,37 @@ +package it.cnr.isti.epasmed.sistemainformativo.service; + +import java.util.List; + +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; +import org.springframework.stereotype.Service; +import org.springframework.transaction.annotation.Transactional; + +import it.cnr.isti.epasmed.sistemainformativo.model.SIEmail; +import it.cnr.isti.epasmed.sistemainformativo.repository.SIEmailRepository; + + + +/** + * Service class for managing Email. + */ +@Service +@Transactional("sistemaInformativoTransactionManager") +public class SIEmailService { + + private final Logger log = LoggerFactory.getLogger(SIEmailService.class); + + private final SIEmailRepository emailRepository; + + public SIEmailService(SIEmailRepository emailRepository) { + super(); + this.emailRepository = emailRepository; + } + + public List readNewFlux(Long flusso) { + log.debug("Request SIEmail Flux > {}",flusso); + return emailRepository.readNewFlux(flusso); + } + + +} \ No newline at end of file diff --git a/src/main/java/it/cnr/isti/epasmed/sistemainformativo/service/SIGruppiPersService.java b/src/main/java/it/cnr/isti/epasmed/sistemainformativo/service/SIGruppiPersService.java new file mode 100644 index 0000000..969598a --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/sistemainformativo/service/SIGruppiPersService.java @@ -0,0 +1,37 @@ +package it.cnr.isti.epasmed.sistemainformativo.service; + +import java.util.List; + +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; +import org.springframework.stereotype.Service; +import org.springframework.transaction.annotation.Transactional; + +import it.cnr.isti.epasmed.sistemainformativo.model.SIGruppiPers; +import it.cnr.isti.epasmed.sistemainformativo.repository.SIGruppiPersRepository; + + + +/** + * Service class for managing GruppiPers. + */ +@Service +@Transactional("sistemaInformativoTransactionManager") +public class SIGruppiPersService { + + private final Logger log = LoggerFactory.getLogger(SIGruppiPersService.class); + + private final SIGruppiPersRepository gruppiPersRepository; + + public SIGruppiPersService(SIGruppiPersRepository gruppiPersRepository) { + super(); + this.gruppiPersRepository = gruppiPersRepository; + } + + public List readNewFlux(Long flusso) { + log.debug("Request GruppiPers Flux > {}",flusso); + return gruppiPersRepository.readNewFlux(flusso); + } + + +} \ No newline at end of file diff --git a/src/main/java/it/cnr/isti/epasmed/sistemainformativo/service/SIGruppiService.java b/src/main/java/it/cnr/isti/epasmed/sistemainformativo/service/SIGruppiService.java new file mode 100644 index 0000000..5a7855d --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/sistemainformativo/service/SIGruppiService.java @@ -0,0 +1,37 @@ +package it.cnr.isti.epasmed.sistemainformativo.service; + +import java.util.List; + +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; +import org.springframework.stereotype.Service; +import org.springframework.transaction.annotation.Transactional; + +import it.cnr.isti.epasmed.sistemainformativo.model.SIGruppi; +import it.cnr.isti.epasmed.sistemainformativo.repository.SIGruppiRepository; + + + +/** + * Service class for managing Gruppi. + */ +@Service +@Transactional("sistemaInformativoTransactionManager") +public class SIGruppiService { + + private final Logger log = LoggerFactory.getLogger(SIGruppiService.class); + + private final SIGruppiRepository gruppiRepository; + + public SIGruppiService(SIGruppiRepository gruppiRepository) { + super(); + this.gruppiRepository = gruppiRepository; + } + + public List readNewFlux(Long flusso) { + log.debug("Request Gruppi Flux > {}",flusso); + return gruppiRepository.readNewFlux(flusso); + } + + +} \ No newline at end of file diff --git a/src/main/java/it/cnr/isti/epasmed/sistemainformativo/service/SIMasterLogService.java b/src/main/java/it/cnr/isti/epasmed/sistemainformativo/service/SIMasterLogService.java new file mode 100644 index 0000000..2f599fb --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/sistemainformativo/service/SIMasterLogService.java @@ -0,0 +1,52 @@ +package it.cnr.isti.epasmed.sistemainformativo.service; + +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; +import org.springframework.stereotype.Service; +import org.springframework.transaction.annotation.Transactional; + +import it.cnr.isti.epasmed.sistemainformativo.repository.SIMasterLogRepository; + + +/** + * Service class for managing MasterLog. + */ +@Service +@Transactional("sistemaInformativoTransactionManager") +public class SIMasterLogService { + + private final Logger log = LoggerFactory.getLogger(SIMasterLogService.class); + + private final SIMasterLogRepository masterLogRepository; + + public SIMasterLogService(SIMasterLogRepository masterLogRepository) { + super(); + this.masterLogRepository = masterLogRepository; + } + + + public int count() { + log.debug("Request MasterLog count()"); + return masterLogRepository.count(); + } + + public void startFluxReads() { + log.debug("Request Start Flux Reads"); + masterLogRepository.startFluxReads(); + } + + public void closeFluxReads(String tabelleLette) throws Exception { + log.debug("Request Close Flux Reads"); + masterLogRepository.closeFluxReads(tabelleLette); + } + + public Long startFluxWrites() throws Exception { + log.debug("Request Start Flux Writes"); + return masterLogRepository.startFluxWrites(); + } + + public void closeFluxWrites(Long idFlusso,String tabelleScritte) { + log.debug("Request Close Flux Writes"); + masterLogRepository.closeFluxWrites(idFlusso,tabelleScritte); + } +} \ No newline at end of file diff --git a/src/main/java/it/cnr/isti/epasmed/sistemainformativo/service/SIPosizioniService.java b/src/main/java/it/cnr/isti/epasmed/sistemainformativo/service/SIPosizioniService.java new file mode 100644 index 0000000..c999a5a --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/sistemainformativo/service/SIPosizioniService.java @@ -0,0 +1,37 @@ +package it.cnr.isti.epasmed.sistemainformativo.service; + +import java.util.List; + +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; +import org.springframework.stereotype.Service; +import org.springframework.transaction.annotation.Transactional; + +import it.cnr.isti.epasmed.sistemainformativo.model.SIPosizioni; +import it.cnr.isti.epasmed.sistemainformativo.repository.SIPosizioniRepository; + + + +/** + * Service class for managing Posizioni. + */ +@Service +@Transactional("sistemaInformativoTransactionManager") +public class SIPosizioniService { + + private final Logger log = LoggerFactory.getLogger(SIPosizioniService.class); + + private final SIPosizioniRepository posizioniRepository; + + public SIPosizioniService(SIPosizioniRepository posizioniRepository) { + super(); + this.posizioniRepository = posizioniRepository; + } + + public List readNewFlux(Long flusso) { + log.debug("Request Posizioni Flux > {}",flusso); + return posizioniRepository.readNewFlux(flusso); + } + + +} \ No newline at end of file diff --git a/src/main/java/it/cnr/isti/epasmed/sistemainformativo/service/SIProrogheService.java b/src/main/java/it/cnr/isti/epasmed/sistemainformativo/service/SIProrogheService.java new file mode 100644 index 0000000..7a4b5e6 --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/sistemainformativo/service/SIProrogheService.java @@ -0,0 +1,37 @@ +package it.cnr.isti.epasmed.sistemainformativo.service; + +import java.util.List; + +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; +import org.springframework.stereotype.Service; +import org.springframework.transaction.annotation.Transactional; + +import it.cnr.isti.epasmed.sistemainformativo.model.SIProroghe; +import it.cnr.isti.epasmed.sistemainformativo.repository.SIProrogheRepository; + + + +/** + * Service class for managing Proroghe. + */ +@Service +@Transactional("sistemaInformativoTransactionManager") +public class SIProrogheService { + + private final Logger log = LoggerFactory.getLogger(SIProrogheService.class); + + private final SIProrogheRepository prorogheRepository; + + public SIProrogheService(SIProrogheRepository prorogheRepository) { + super(); + this.prorogheRepository = prorogheRepository; + } + + public List readNewFlux(Long flusso) { + log.debug("Request Proroghe Flux > {}",flusso); + return prorogheRepository.readNewFlux(flusso); + } + + +} \ No newline at end of file diff --git a/src/main/java/it/cnr/isti/epasmed/sistemainformativo/service/SITelefoniService.java b/src/main/java/it/cnr/isti/epasmed/sistemainformativo/service/SITelefoniService.java new file mode 100644 index 0000000..ac4be44 --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/sistemainformativo/service/SITelefoniService.java @@ -0,0 +1,37 @@ +package it.cnr.isti.epasmed.sistemainformativo.service; + +import java.util.List; + +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; +import org.springframework.stereotype.Service; +import org.springframework.transaction.annotation.Transactional; + +import it.cnr.isti.epasmed.sistemainformativo.model.SITelefoni; +import it.cnr.isti.epasmed.sistemainformativo.repository.SITelefoniRepository; + + + +/** + * Service class for managing Telefoni. + */ +@Service +@Transactional("sistemaInformativoTransactionManager") +public class SITelefoniService { + + private final Logger log = LoggerFactory.getLogger(SITelefoniService.class); + + private final SITelefoniRepository telefoniRepository; + + public SITelefoniService(SITelefoniRepository telefoniRepository) { + super(); + this.telefoniRepository = telefoniRepository; + } + + public List readNewFlux(Long flusso) { + log.debug("Request Telefoni Flux > {}",flusso); + return telefoniRepository.readNewFlux(flusso); + } + + +} \ No newline at end of file diff --git a/src/main/java/it/cnr/isti/epasmed/sync/SyncService.java b/src/main/java/it/cnr/isti/epasmed/sync/SyncService.java new file mode 100644 index 0000000..55f7454 --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/sync/SyncService.java @@ -0,0 +1,405 @@ +package it.cnr.isti.epasmed.sync; + +import java.time.LocalDateTime; +import java.util.List; + +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.stereotype.Service; + +import it.cnr.isti.epasmed.domain.TabsSI; +import it.cnr.isti.epasmed.epas.dto.EPASPersonsDTO; +import it.cnr.isti.epasmed.epas.mapper.EPASPersonsMapper; +import it.cnr.isti.epasmed.epas.model.EPASPersons; +import it.cnr.isti.epasmed.epas.service.EPASPersonsService; +import it.cnr.isti.epasmed.service.TabsSIService; +import it.cnr.isti.epasmed.sistemainformativo.model.SIAnagrafico; +import it.cnr.isti.epasmed.sistemainformativo.model.SIEmail; +import it.cnr.isti.epasmed.sistemainformativo.model.SITelefoni; +import it.cnr.isti.epasmed.sistemainformativo.service.SIAnagraficoService; +import it.cnr.isti.epasmed.sistemainformativo.service.SIEmailService; +import it.cnr.isti.epasmed.sistemainformativo.service.SIMasterLogService; +import it.cnr.isti.epasmed.sistemainformativo.service.SITelefoniService; + +@Service +public class SyncService { + private static final String ISTI_OFFICE_ID = "1"; + private static final String SI_FLAG_DEL_TRUE = "1"; + private static final String SI_RECAPITO_TELEFONICO_UFFICIO = "Ufficio"; + private static final String SI_TIPO_EMAIL_ISTITUZIONALE = "Istituzionale"; + private static final String SI_TIPO_EMAIL_CNR = "C.N.R."; + + private final Logger log = LoggerFactory.getLogger(SyncService.class); + + @Autowired + TabsSIService tabsSIService; + + @Autowired + SIMasterLogService siMasterLogService; + @Autowired + SIAnagraficoService siAnagraficoService; + @Autowired + SITelefoniService siTelefoniService; + @Autowired + SIEmailService siEmailService; + + @Autowired + EPASPersonsService epasPersonsService; + + @Autowired + EPASPersonsMapper epasPersonsMapper; + + private boolean banagrafico; + private boolean bemail; + private boolean btelefoni; + private boolean bgruppi; + private boolean bgruppo_pers; + private boolean bposizioni; + private boolean bproroghe; + private boolean borario; + private boolean bcartellini; + private boolean bcartellini_rendicontazioni; + private boolean bpers_orario; + private boolean blavoro_fuori_sede; + private boolean baspettative; + + public void executeReads() throws Exception { + banagrafico = false; + bemail = false; + btelefoni = false; + bgruppi = false; + bgruppo_pers = false; + bposizioni = false; + bproroghe = false; + + List tabsSI = tabsSIService.getAllTabsSI(); + siMasterLogService.startFluxReads(); + readData(tabsSI); + siMasterLogService.closeFluxReads(tabellelette()); + + } + + public void executeWrites() throws Exception { + borario = false; + bcartellini = false; + bcartellini_rendicontazioni = false; + bpers_orario = false; + blavoro_fuori_sede = false; + baspettative = false; + + List tabsSI = tabsSIService.getAllTabsSI(); + Long idFlux=siMasterLogService.startFluxWrites(); + writeData(idFlux,tabsSI); + siMasterLogService.closeFluxWrites(idFlux,tabellelette()); + + } + + private void readData(List tabsSI) { + TabsSI posizioniTab = null; + TabsSI prorogheTab = null; + for (TabsSI tab : tabsSI) { + log.info("TabSI: {}", tab); + if (tab.getOperazioni() != null && !tab.getOperazioni().isEmpty() + && tab.getOperazioni().compareTo("R") == 0) { + if (tab.getNome() == null || tab.getNome().isEmpty()) { + continue; + } + switch (tab.getNome()) { + case "anagrafico": + syncAnagrafico(tab); + break; + case "mail": + syncMail(tab); + break; + case "telefoni": + syncTelefoni(tab); + break; + case "gruppi": + syncGruppi(tab); + break; + case "gruppo_pers": + syncGruppoPers(tab); + break; + case "posizioni": + posizioniTab = tab; + break; + case "proroghe": + prorogheTab = tab; + break; + default: + break; + } + + } + } + if (posizioniTab != null || prorogheTab != null) { + syncPosizioniAndProroghe(posizioniTab, prorogheTab); + } + } + + private void syncAnagrafico(TabsSI tab) { + if (tab.getIdFlusso() == null || tab.getIdFlusso().longValue() == 0) { + log.error("Invalid Id Flusso for tab: {}", tab); + return; + } + long maxIdFlusso = 0; + + List sianList = siAnagraficoService.readNewFlux(tab.getIdFlusso()); + for (SIAnagrafico sia : sianList) { + if (sia.getId_flusso() != null && sia.getId_flusso().longValue() > maxIdFlusso) { + maxIdFlusso = sia.getId_flusso(); + } + if (sia.getCodicefiscale() != null && !sia.getCodicefiscale().isEmpty()) { + if (sia.getFlag_del() != null && !sia.getFlag_del().isEmpty()) { + if (sia.getFlag_del().compareTo(SI_FLAG_DEL_TRUE) != 0) { + EPASPersons epasPerson = epasPersonsService.getByFiscalCode(sia.getCodicefiscale()); + if (epasPerson == null) { + EPASPersonsDTO epasPersonsDTO = new EPASPersonsDTO(); + epasPersonsDTO.setFiscalCode(sia.getCodicefiscale()); + epasPersonsDTO.setName(sia.getNome()); + epasPersonsDTO.setSurname(sia.getCognome()); + epasPersonsDTO.setOfficeId(ISTI_OFFICE_ID); + epasPerson = epasPersonsService.create(epasPersonsDTO); + log.info("EPAS Created Person: {}", epasPerson); + } else { + EPASPersonsDTO epasPersonsDTO = epasPersonsMapper.epasPersonsToEPASPersonsDTO(epasPerson); + epasPersonsDTO.setName(sia.getNome()); + epasPersonsDTO.setSurname(sia.getCognome()); + epasPersonsService.updateByFiscalCode(epasPersonsDTO.getFiscalCode(), epasPersonsDTO); + log.info("EPAS Updated Person: {}", epasPersonsDTO); + } + } + } + } + + } + + if (maxIdFlusso > 0) { + banagrafico = true; + tab.setIdFlusso(maxIdFlusso); + tab.setLastUpdate(LocalDateTime.now()); + tabsSIService.updateTabsSI(tab); + } + + } + + private void syncMail(TabsSI tab) { + if (tab.getIdFlusso() == null || tab.getIdFlusso().longValue() == 0) { + log.error("Invalid Id Flusso for tab: {}", tab); + return; + } + long maxIdFlusso = 0; + + List seList = siEmailService.readNewFlux(tab.getIdFlusso()); + for (SIEmail se : seList) { + if (se.getId_flusso() != null && se.getId_flusso().longValue() > maxIdFlusso) { + maxIdFlusso = se.getId_flusso(); + } + if (se.getCf() != null && !se.getCf().isEmpty()) { + if (se.getFlag_del() != null && !se.getFlag_del().isEmpty()) { + if (se.getFlag_del().compareTo(SI_FLAG_DEL_TRUE) != 0) { + if (se.getTipo() != null && !se.getTipo().isEmpty()) { + String emailTipo = se.getTipo().trim(); + if (!emailTipo.isEmpty()) { + EPASPersons epasPerson = epasPersonsService.getByFiscalCode(se.getCf()); + if (epasPerson != null) { + EPASPersonsDTO epasPersonsDTO = epasPersonsMapper + .epasPersonsToEPASPersonsDTO(epasPerson); + switch (emailTipo) { + case SI_TIPO_EMAIL_ISTITUZIONALE: + epasPersonsDTO.setEmail(se.getEmail()); + epasPersonsService.updateByFiscalCode(epasPersonsDTO.getFiscalCode(), + epasPersonsDTO); + log.info("EPAS Updated Person: {}", epasPersonsDTO); + + break; + case SI_TIPO_EMAIL_CNR: + epasPersonsDTO.setEppn(se.getEmail()); + epasPersonsService.updateByFiscalCode(epasPersonsDTO.getFiscalCode(), + epasPersonsDTO); + log.info("EPAS Updated Person: {}", epasPersonsDTO); + break; + default: + break; + } + } + } + } + } + } + } + + } + + if (maxIdFlusso > 0) { + bemail = true; + tab.setIdFlusso(maxIdFlusso); + tab.setLastUpdate(LocalDateTime.now()); + tabsSIService.updateTabsSI(tab); + } + + } + + private void syncTelefoni(TabsSI tab) { + if (tab.getIdFlusso() == null || tab.getIdFlusso().longValue() == 0) { + log.info("Invalid Id Flusso for tab: {}", tab); + return; + } + long maxIdFlusso = 0; + + List sitList = siTelefoniService.readNewFlux(tab.getIdFlusso()); + for (SITelefoni sit : sitList) { + if (sit.getId_flusso() != null && sit.getId_flusso().longValue() > maxIdFlusso) { + maxIdFlusso = sit.getId_flusso(); + } + if (sit.getCf() != null && !sit.getCf().isEmpty()) { + if (sit.getFlag_del() != null && !sit.getFlag_del().isEmpty()) { + if (sit.getFlag_del().compareTo(SI_FLAG_DEL_TRUE) != 0) { + if (sit.getTiporecapitotelefonico() != null && !sit.getTiporecapitotelefonico().isEmpty()) { + String tipoRecapitoTelefonico = sit.getTiporecapitotelefonico().trim(); + if (!tipoRecapitoTelefonico.isEmpty()) { + if (tipoRecapitoTelefonico.compareTo(SI_RECAPITO_TELEFONICO_UFFICIO) == 0) { + EPASPersons epasPerson = epasPersonsService.getByFiscalCode(sit.getCf()); + if (epasPerson != null) { + EPASPersonsDTO epasPersonsDTO = epasPersonsMapper + .epasPersonsToEPASPersonsDTO(epasPerson); + epasPersonsDTO.setTelephone(sit.getRecapitotelefonico()); + epasPersonsService.updateByFiscalCode(epasPersonsDTO.getFiscalCode(), + epasPersonsDTO); + log.info("EPAS Updated Person: {}", epasPersonsDTO); + } + } + } + } + } + } + } + + } + + if (maxIdFlusso > 0) { + btelefoni = true; + tab.setIdFlusso(maxIdFlusso); + tab.setLastUpdate(LocalDateTime.now()); + tabsSIService.updateTabsSI(tab); + } + + } + + private void syncGruppi(TabsSI tab) { + // TODO Auto-generated method stub + + } + + private void syncGruppoPers(TabsSI tab) { + // TODO Auto-generated method stub + + } + + private void syncPosizioniAndProroghe(TabsSI posizioniTab, TabsSI prorogheTab) { + + } + + private String tabellelette() { + String tabelle_lette = ""; + if (banagrafico) { + tabelle_lette = "anagrafico"; + } + if (bemail) { + if (tabelle_lette == null || tabelle_lette.isEmpty()) { + tabelle_lette = "mail"; + } else { + tabelle_lette = tabelle_lette + ",mail"; + } + } + if (btelefoni) { + if (tabelle_lette == null || tabelle_lette.isEmpty()) { + tabelle_lette = "telefoni"; + } else { + tabelle_lette = tabelle_lette + ",telefoni"; + } + } + + if (bgruppi) { + if (tabelle_lette == null || tabelle_lette.isEmpty()) { + tabelle_lette = "gruppi"; + } else { + tabelle_lette = tabelle_lette + ",gruppi"; + } + } + + if (bgruppo_pers) { + if (tabelle_lette == null || tabelle_lette.isEmpty()) { + tabelle_lette = "gruppo_pers"; + } else { + tabelle_lette = tabelle_lette + ",gruppo_pers"; + } + } + + if (bposizioni) { + if (tabelle_lette == null || tabelle_lette.isEmpty()) { + tabelle_lette = "posizioni"; + } else { + tabelle_lette = tabelle_lette + ",posizioni"; + } + } + + if (bproroghe) { + if (tabelle_lette == null || tabelle_lette.isEmpty()) { + tabelle_lette = "proroghe"; + } else { + tabelle_lette = tabelle_lette + ",proroghe"; + } + } + + return tabelle_lette; + } + + + private void writeData(Long idFlux, List tabsSI) { + TabsSI caretelliniRendicontazioniTab = null; + TabsSI cartelliniTab = null; + + + + for (TabsSI tab : tabsSI) { + log.info("TabSI: {}", tab); + if (tab.getOperazioni() != null && !tab.getOperazioni().isEmpty() + && tab.getOperazioni().compareTo("W") == 0) { + if (tab.getNome() == null || tab.getNome().isEmpty()) { + continue; + } + + + + /* + switch (tab.getNome()) { + case "orario": + syncOrario(idFlux,tab); + break; + case "pers_orario": + syncOrario(idFlux,tab); + break; + case "cartellini": + syncCartellini(idFlux,tab); + break; + case "cartellini_rendicontazioni": + syncCartelliniRendicontazioni(idFlux,tab); + break; + + case "lavoro_fuori_sede": + syncLavoroFuoriSede(idFlux,tab); + break; + case "aspettative": + syncAspettative(idFlux,tab); + break; + default: + break; + } + */ + } + } + } + +} diff --git a/src/main/java/it/cnr/isti/epasmed/web/rest/AccountResource.java b/src/main/java/it/cnr/isti/epasmed/web/rest/AccountResource.java new file mode 100644 index 0000000..5641db1 --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/web/rest/AccountResource.java @@ -0,0 +1,240 @@ +package it.cnr.isti.epasmed.web.rest; + +import it.cnr.isti.epasmed.domain.PersistentToken; +import it.cnr.isti.epasmed.repository.PersistentTokenRepository; +import it.cnr.isti.epasmed.domain.User; +import it.cnr.isti.epasmed.repository.UserRepository; +import it.cnr.isti.epasmed.security.AuthoritiesConstants; +import it.cnr.isti.epasmed.security.SecurityUtils; +import it.cnr.isti.epasmed.service.MailService; +import it.cnr.isti.epasmed.service.UserService; +import it.cnr.isti.epasmed.service.dto.PasswordChangeDTO; +import it.cnr.isti.epasmed.service.dto.UserDTO; +import it.cnr.isti.epasmed.web.rest.errors.*; +import it.cnr.isti.epasmed.web.rest.vm.KeyAndPasswordVM; +import it.cnr.isti.epasmed.web.rest.vm.ManagedUserVM; + +import org.apache.commons.lang3.StringUtils; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; +import org.springframework.http.HttpStatus; +import org.springframework.security.access.annotation.Secured; +import org.springframework.security.access.prepost.PreAuthorize; +import org.springframework.web.bind.annotation.*; + +import javax.servlet.http.HttpServletRequest; +import javax.validation.Valid; +import java.io.UnsupportedEncodingException; +import java.net.URLDecoder; +import java.util.*; + +/** + * REST controller for managing the current user's account. + */ +@RestController +@RequestMapping("/api") +public class AccountResource { + + private static class AccountResourceException extends RuntimeException { + private AccountResourceException(String message) { + super(message); + } + } + + private final Logger log = LoggerFactory.getLogger(AccountResource.class); + + private final UserRepository userRepository; + + private final UserService userService; + + private final MailService mailService; + + private final PersistentTokenRepository persistentTokenRepository; + + public AccountResource(UserRepository userRepository, UserService userService, MailService mailService, PersistentTokenRepository persistentTokenRepository) { + + this.userRepository = userRepository; + this.userService = userService; + this.mailService = mailService; + this.persistentTokenRepository = persistentTokenRepository; + } + + /** + * {@code POST /register} : register the user. + * + * @param managedUserVM the managed user View Model. + * @throws InvalidPasswordException {@code 400 (Bad Request)} if the password is incorrect. + * @throws EmailAlreadyUsedException {@code 400 (Bad Request)} if the email is already used. + * @throws LoginAlreadyUsedException {@code 400 (Bad Request)} if the login is already used. + */ + @PostMapping("/register") + @PreAuthorize("hasAuthority(\"" + AuthoritiesConstants.ADMIN + "\")") + @ResponseStatus(HttpStatus.CREATED) + public void registerAccount(@Valid @RequestBody ManagedUserVM managedUserVM) { + if (!checkPasswordLength(managedUserVM.getPassword())) { + throw new InvalidPasswordException(); + } + User user = userService.registerUser(managedUserVM, managedUserVM.getPassword()); + mailService.sendActivationEmail(user); + } + + /** + * {@code GET /activate} : activate the registered user. + * + * @param key the activation key. + * @throws RuntimeException {@code 500 (Internal Server Error)} if the user couldn't be activated. + */ + @GetMapping("/activate") + public void activateAccount(@RequestParam(value = "key") String key) { + Optional user = userService.activateRegistration(key); + if (!user.isPresent()) { + throw new AccountResourceException("No user was found for this activation key"); + } + } + + /** + * {@code GET /authenticate} : check if the user is authenticated, and return its login. + * + * @param request the HTTP request. + * @return the login if the user is authenticated. + */ + @GetMapping("/authenticate") + public String isAuthenticated(HttpServletRequest request) { + log.debug("REST request to check if the current user is authenticated"); + return request.getRemoteUser(); + } + + /** + * {@code GET /account} : get the current user. + * + * @return the current user. + * @throws RuntimeException {@code 500 (Internal Server Error)} if the user couldn't be returned. + */ + @GetMapping("/account") + public UserDTO getAccount() { + return userService.getUserWithAuthorities() + .map(UserDTO::new) + .orElseThrow(() -> new AccountResourceException("User could not be found")); + } + + /** + * {@code POST /account} : update the current user information. + * + * @param userDTO the current user information. + * @throws EmailAlreadyUsedException {@code 400 (Bad Request)} if the email is already used. + * @throws RuntimeException {@code 500 (Internal Server Error)} if the user login wasn't found. + */ + @PostMapping("/account") + public void saveAccount(@Valid @RequestBody UserDTO userDTO) { + String userLogin = SecurityUtils.getCurrentUserLogin().orElseThrow(() -> new AccountResourceException("Current user login not found")); + Optional existingUser = userRepository.findOneByEmailIgnoreCase(userDTO.getEmail()); + if (existingUser.isPresent() && (!existingUser.get().getLogin().equalsIgnoreCase(userLogin))) { + throw new EmailAlreadyUsedException(); + } + Optional user = userRepository.findOneByLogin(userLogin); + if (!user.isPresent()) { + throw new AccountResourceException("User could not be found"); + } + userService.updateUser(userDTO.getFirstName(), userDTO.getLastName(), userDTO.getEmail(), + userDTO.getLangKey(), userDTO.getImageUrl()); + } + + /** + * {@code POST /account/change-password} : changes the current user's password. + * + * @param passwordChangeDto current and new password. + * @throws InvalidPasswordException {@code 400 (Bad Request)} if the new password is incorrect. + */ + @PostMapping(path = "/account/change-password") + public void changePassword(@RequestBody PasswordChangeDTO passwordChangeDto) { + if (!checkPasswordLength(passwordChangeDto.getNewPassword())) { + throw new InvalidPasswordException(); + } + userService.changePassword(passwordChangeDto.getCurrentPassword(), passwordChangeDto.getNewPassword()); + } + + /** + * {@code GET /account/sessions} : get the current open sessions. + * + * @return the current open sessions. + * @throws RuntimeException {@code 500 (Internal Server Error)} if the current open sessions couldn't be retrieved. + */ + @GetMapping("/account/sessions") + public List getCurrentSessions() { + return persistentTokenRepository.findByUser( + userRepository.findOneByLogin(SecurityUtils.getCurrentUserLogin() + .orElseThrow(() -> new AccountResourceException("Current user login not found"))) + .orElseThrow(() -> new AccountResourceException("User could not be found")) + ); + } + + /** + * {@code DELETE /account/sessions?series={series}} : invalidate an existing session. + * + * - You can only delete your own sessions, not any other user's session + * - If you delete one of your existing sessions, and that you are currently logged in on that session, you will + * still be able to use that session, until you quit your browser: it does not work in real time (there is + * no API for that), it only removes the "remember me" cookie + * - This is also true if you invalidate your current session: you will still be able to use it until you close + * your browser or that the session times out. But automatic login (the "remember me" cookie) will not work + * anymore. + * There is an API to invalidate the current session, but there is no API to check which session uses which + * cookie. + * + * @param series the series of an existing session. + * @throws UnsupportedEncodingException if the series couldn't be URL decoded. + */ + @DeleteMapping("/account/sessions/{series}") + public void invalidateSession(@PathVariable String series) throws UnsupportedEncodingException { + String decodedSeries = URLDecoder.decode(series, "UTF-8"); + SecurityUtils.getCurrentUserLogin() + .flatMap(userRepository::findOneByLogin) + .ifPresent(u -> + persistentTokenRepository.findByUser(u).stream() + .filter(persistentToken -> StringUtils.equals(persistentToken.getSeries(), decodedSeries)) + .findAny().ifPresent(t -> persistentTokenRepository.deleteById(decodedSeries))); + } + + /** + * {@code POST /account/reset-password/init} : Send an email to reset the password of the user. + * + * @param mail the mail of the user. + */ + @PostMapping(path = "/account/reset-password/init") + public void requestPasswordReset(@RequestBody String mail) { + Optional user = userService.requestPasswordReset(mail); + if (user.isPresent()) { + mailService.sendPasswordResetMail(user.get()); + } else { + // Pretend the request has been successful to prevent checking which emails really exist + // but log that an invalid attempt has been made + log.warn("Password reset requested for non existing mail"); + } + } + + /** + * {@code POST /account/reset-password/finish} : Finish to reset the password of the user. + * + * @param keyAndPassword the generated key and the new password. + * @throws InvalidPasswordException {@code 400 (Bad Request)} if the password is incorrect. + * @throws RuntimeException {@code 500 (Internal Server Error)} if the password could not be reset. + */ + @PostMapping(path = "/account/reset-password/finish") + public void finishPasswordReset(@RequestBody KeyAndPasswordVM keyAndPassword) { + if (!checkPasswordLength(keyAndPassword.getNewPassword())) { + throw new InvalidPasswordException(); + } + Optional user = + userService.completePasswordReset(keyAndPassword.getNewPassword(), keyAndPassword.getKey()); + + if (!user.isPresent()) { + throw new AccountResourceException("No user was found for this reset key"); + } + } + + private static boolean checkPasswordLength(String password) { + return !StringUtils.isEmpty(password) && + password.length() >= ManagedUserVM.PASSWORD_MIN_LENGTH && + password.length() <= ManagedUserVM.PASSWORD_MAX_LENGTH; + } +} diff --git a/src/main/java/it/cnr/isti/epasmed/web/rest/AuditResource.java b/src/main/java/it/cnr/isti/epasmed/web/rest/AuditResource.java new file mode 100644 index 0000000..1148474 --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/web/rest/AuditResource.java @@ -0,0 +1,79 @@ +package it.cnr.isti.epasmed.web.rest; + +import it.cnr.isti.epasmed.service.AuditEventService; + +import io.github.jhipster.web.util.PaginationUtil; +import io.github.jhipster.web.util.ResponseUtil; +import org.springframework.boot.actuate.audit.AuditEvent; +import org.springframework.data.domain.Page; +import org.springframework.data.domain.Pageable; +import org.springframework.http.HttpHeaders; +import org.springframework.http.HttpStatus; +import org.springframework.http.ResponseEntity; +import org.springframework.web.bind.annotation.*; +import org.springframework.web.servlet.support.ServletUriComponentsBuilder; + +import java.time.Instant; +import java.time.LocalDate; +import java.time.ZoneId; +import java.util.List; + +/** + * REST controller for getting the {@link AuditEvent}s. + */ +@RestController +@RequestMapping("/management/audits") +public class AuditResource { + + private final AuditEventService auditEventService; + + public AuditResource(AuditEventService auditEventService) { + this.auditEventService = auditEventService; + } + + /** + * {@code GET /audits} : get a page of {@link AuditEvent}s. + * + * @param pageable the pagination information. + * @return the {@link ResponseEntity} with status {@code 200 (OK)} and the list of {@link AuditEvent}s in body. + */ + @GetMapping + public ResponseEntity> getAll(Pageable pageable) { + Page page = auditEventService.findAll(pageable); + HttpHeaders headers = PaginationUtil.generatePaginationHttpHeaders(ServletUriComponentsBuilder.fromCurrentRequest(), page); + return new ResponseEntity<>(page.getContent(), headers, HttpStatus.OK); + } + + /** + * {@code GET /audits} : get a page of {@link AuditEvent} between the {@code fromDate} and {@code toDate}. + * + * @param fromDate the start of the time period of {@link AuditEvent} to get. + * @param toDate the end of the time period of {@link AuditEvent} to get. + * @param pageable the pagination information. + * @return the {@link ResponseEntity} with status {@code 200 (OK)} and the list of {@link AuditEvent} in body. + */ + @GetMapping(params = {"fromDate", "toDate"}) + public ResponseEntity> getByDates( + @RequestParam(value = "fromDate") LocalDate fromDate, + @RequestParam(value = "toDate") LocalDate toDate, + Pageable pageable) { + + Instant from = fromDate.atStartOfDay(ZoneId.systemDefault()).toInstant(); + Instant to = toDate.atStartOfDay(ZoneId.systemDefault()).plusDays(1).toInstant(); + + Page page = auditEventService.findByDates(from, to, pageable); + HttpHeaders headers = PaginationUtil.generatePaginationHttpHeaders(ServletUriComponentsBuilder.fromCurrentRequest(), page); + return new ResponseEntity<>(page.getContent(), headers, HttpStatus.OK); + } + + /** + * {@code GET /audits/:id} : get an {@link AuditEvent} by id. + * + * @param id the id of the entity to get. + * @return the {@link ResponseEntity} with status {@code 200 (OK)} and the {@link AuditEvent} in body, or status {@code 404 (Not Found)}. + */ + @GetMapping("/{id:.+}") + public ResponseEntity get(@PathVariable Long id) { + return ResponseUtil.wrapOrNotFound(auditEventService.find(id)); + } +} diff --git a/src/main/java/it/cnr/isti/epasmed/web/rest/ClientForwardController.java b/src/main/java/it/cnr/isti/epasmed/web/rest/ClientForwardController.java new file mode 100644 index 0000000..eb1f985 --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/web/rest/ClientForwardController.java @@ -0,0 +1,17 @@ +package it.cnr.isti.epasmed.web.rest; + +import org.springframework.stereotype.Controller; +import org.springframework.web.bind.annotation.GetMapping; + +@Controller +public class ClientForwardController { + + /** + * Forwards any unmapped paths (except those containing a period) to the client {@code index.html}. + * @return forward to client {@code index.html}. + */ + @GetMapping(value = "/**/{path:[^\\.]*}") + public String forward() { + return "forward:/"; + } +} diff --git a/src/main/java/it/cnr/isti/epasmed/web/rest/SyncResource.java b/src/main/java/it/cnr/isti/epasmed/web/rest/SyncResource.java new file mode 100644 index 0000000..7c88e6a --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/web/rest/SyncResource.java @@ -0,0 +1,70 @@ +package it.cnr.isti.epasmed.web.rest; + +import java.net.URISyntaxException; + +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; +import org.springframework.beans.factory.annotation.Value; +import org.springframework.http.ResponseEntity; +import org.springframework.security.access.prepost.PreAuthorize; +import org.springframework.web.bind.annotation.PostMapping; +import org.springframework.web.bind.annotation.RequestMapping; +import org.springframework.web.bind.annotation.RestController; + +import io.github.jhipster.web.util.HeaderUtil; +import it.cnr.isti.epasmed.domain.TabsSI; +import it.cnr.isti.epasmed.security.AuthoritiesConstants; +import it.cnr.isti.epasmed.sync.SyncService; + +/** + * REST controller for managing Table SI. + *

+ * This class accesses the {@link TabsSI} entity. + * + * + */ +@RestController +@RequestMapping("/api") +public class SyncResource { + + private final Logger log = LoggerFactory.getLogger(SyncResource.class); + + @Value("${jhipster.clientApp.name}") + private String applicationName; + + private final SyncService syncReads; + + public SyncResource(SyncService syncReads) { + this.syncReads = syncReads; + } + + /** + * {@code POST /sync/reads} : Retrieve new flux from Sistema Informativo and + * update ePAS. + * + * @return the {@link ResponseEntity} with status {@code 201 (Executed)} + * or with status {@code 400 (Bad Request)} if there is a error. + * + + */ + @PostMapping("/sync/reads") + @PreAuthorize("hasAuthority(\"" + AuthoritiesConstants.ADMIN + "\")") + public ResponseEntity syncReads() throws URISyntaxException { + log.info("REST request syncReads()"); + ResponseEntity res; + try { + syncReads.executeReads(); + log.info("Sincronizzazione delle Letture eseguita correttamente."); + res=ResponseEntity.noContent().headers(HeaderUtil.createAlert(applicationName, + "Sincronizzazione delle letture eseguita corretamente.","")).build(); + + } catch (Throwable e) { + log.error("Errore nella sincronizzazione delle letture: {}", e.getLocalizedMessage(), e); + res=ResponseEntity.noContent().headers(HeaderUtil.createAlert(applicationName, + "Errore nella sincronizzazione delle letture: {}", e.getLocalizedMessage())).build(); + } + return res; + + + } +} diff --git a/src/main/java/it/cnr/isti/epasmed/web/rest/TabsSIResource.java b/src/main/java/it/cnr/isti/epasmed/web/rest/TabsSIResource.java new file mode 100644 index 0000000..7c39c95 --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/web/rest/TabsSIResource.java @@ -0,0 +1,172 @@ +package it.cnr.isti.epasmed.web.rest; + +import java.net.URI; +import java.net.URISyntaxException; +import java.util.Arrays; +import java.util.Collections; +import java.util.List; +import java.util.Optional; + +import javax.validation.Valid; + +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; +import org.springframework.beans.factory.annotation.Value; +import org.springframework.data.domain.Page; +import org.springframework.data.domain.Pageable; +import org.springframework.data.domain.Sort; +import org.springframework.http.HttpHeaders; +import org.springframework.http.HttpStatus; +import org.springframework.http.ResponseEntity; +import org.springframework.security.access.prepost.PreAuthorize; +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.RestController; +import org.springframework.web.servlet.support.ServletUriComponentsBuilder; + +import io.github.jhipster.web.util.HeaderUtil; +import io.github.jhipster.web.util.PaginationUtil; +import io.github.jhipster.web.util.ResponseUtil; +import it.cnr.isti.epasmed.config.Constants; +import it.cnr.isti.epasmed.domain.TabsSI; +import it.cnr.isti.epasmed.security.AuthoritiesConstants; +import it.cnr.isti.epasmed.service.TabsSIService; +import it.cnr.isti.epasmed.web.rest.errors.BadRequestAlertException; + +/** + * REST controller for managing Table SI. + *

+ * This class accesses the {@link TabsSI} entity. + * + * + */ +@RestController +@RequestMapping("/api") +public class TabsSIResource { + private static final List ALLOWED_ORDERED_PROPERTIES = Collections + .unmodifiableList(Arrays.asList("id", "login", "firstName", "lastName", "email", "activated", "langKey")); + + private final Logger log = LoggerFactory.getLogger(TabsSIResource.class); + + @Value("${jhipster.clientApp.name}") + private String applicationName; + + private final TabsSIService tabsSIService; + + //private final TabsSIRepository tabsSIRepository; + + public TabsSIResource(TabsSIService tabsSIService) { + this.tabsSIService = tabsSIService; + //this.tabsSIRepository = tabsSIRepository; + + } + + /** + * {@code GET /tabssi} : get all tabs of sistema informativo. + * + * @param pageable the pagination information. + * @return the {@link ResponseEntity} with status {@code 200 (OK)} and with body + * all tabs of sistema informativo. + */ + @GetMapping("/tabssi") + public ResponseEntity> getAllTabsSI(Pageable pageable) { + log.info("REST request to read Table SI : {}", pageable); + + if (!onlyContainsAllowedProperties(pageable)) { + return ResponseEntity.badRequest().build(); + } + + final Page page = tabsSIService.getAllTabsSI(pageable); + HttpHeaders headers = PaginationUtil + .generatePaginationHttpHeaders(ServletUriComponentsBuilder.fromCurrentRequest(), page); + return new ResponseEntity<>(page.getContent(), headers, HttpStatus.OK); + } + + private boolean onlyContainsAllowedProperties(Pageable pageable) { + return pageable.getSort().stream().map(Sort.Order::getProperty).allMatch(ALLOWED_ORDERED_PROPERTIES::contains); + } + + /** + * {@code POST /tabssi} : Creates a new table for SI. + *

+ * Creates a new user if the login and email are not already used, and sends an + * mail with an activation link. The user needs to be activated on creation. + * + * @param tabsSIDTO the user to create. + * @return the {@link ResponseEntity} with status {@code 201 (Created)} and with + * body the new table SI, or with status {@code 400 (Bad Request)} if + * the id is already in use. + * @throws URISyntaxException if the Location URI syntax is incorrect. + * @throws BadRequestAlertException {@code 400 (Bad Request)} if the id is + * already in use. + */ + @PostMapping("/tabssi") + @PreAuthorize("hasAuthority(\"" + AuthoritiesConstants.ADMIN + "\")") + public ResponseEntity createTabsSI(@Valid @RequestBody TabsSI tabsSIDTO) throws URISyntaxException { + log.info("REST request to save Table SI : {}", tabsSIDTO); + + if (tabsSIDTO.getId() != null) { + throw new BadRequestAlertException("A new table SI cannot already have an ID", "TabsSI", "idexists"); + } else { + TabsSI newTabsSI = tabsSIService.createTabsSI(tabsSIDTO); + return ResponseEntity.created(new URI("/api/tabssi/" + newTabsSI.getId())) + .headers(HeaderUtil.createAlert(applicationName, + "A user is created with identifier " + newTabsSI.getId(), newTabsSI.getNome())) + .body(newTabsSI); + } + } + + /** + * {@code PUT /users} : Updates an existing User. + * + * @param tabsSIDTO the table of SI to update. + * @return the {@link ResponseEntity} with status {@code 200 (OK)} and with body + * the updated table of IS. + * + */ + @PutMapping("/tabssi") + @PreAuthorize("hasAuthority(\"" + AuthoritiesConstants.ADMIN + "\")") + public ResponseEntity updateTabsSI(@Valid @RequestBody TabsSI tabsSIDTO) { + log.debug("REST request to update Table SI : {}", tabsSIDTO); + Optional updatedTabsSI = tabsSIService.updateTabsSI(tabsSIDTO); + + return ResponseUtil.wrapOrNotFound(updatedTabsSI, HeaderUtil.createAlert(applicationName, + "A table SI is updated with identifier " + tabsSIDTO.getId(), tabsSIDTO.getNome())); + } + + /** + * {@code GET /tabssi/:id} : get the "id" of table SI. + * + * @param id the id of the table SI to find. + * @return the {@link ResponseEntity} with status {@code 200 (OK)} and with body + * the "login" user, or with status {@code 404 (Not Found)}. + */ + @GetMapping("/tabssi/{id:" + Constants.LOGIN_REGEX + "}") + public ResponseEntity getTabsSI(@PathVariable String id) { + log.debug("REST request to get Table SI : {}", id); + return ResponseUtil.wrapOrNotFound(tabsSIService.getTabsSIById(Long.valueOf(id))); + + } + + /** + * {@code DELETE /tabssi/:id} : delete the "id" TabsSI. + * + * @param id the id of the table SI to delete. + * @return the {@link ResponseEntity} with status {@code 204 (NO_CONTENT)}. + */ + @DeleteMapping("/tabssi/{id:" + Constants.LOGIN_REGEX + "}") + @PreAuthorize("hasAuthority(\"" + AuthoritiesConstants.ADMIN + "\")") + public ResponseEntity deleteUser(@PathVariable String id) { + log.debug("REST request to delete Table SI id: {}", id); + tabsSIService.deleteTabsSI(Long.valueOf(id)); + return ResponseEntity.noContent() + .headers(HeaderUtil.createAlert(applicationName, "A Table IS is deleted with identifier " + id, id)) + .build(); + } + +} diff --git a/src/main/java/it/cnr/isti/epasmed/web/rest/UserResource.java b/src/main/java/it/cnr/isti/epasmed/web/rest/UserResource.java new file mode 100644 index 0000000..fd38a17 --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/web/rest/UserResource.java @@ -0,0 +1,200 @@ +package it.cnr.isti.epasmed.web.rest; + +import it.cnr.isti.epasmed.config.Constants; +import it.cnr.isti.epasmed.domain.User; +import it.cnr.isti.epasmed.repository.UserRepository; +import it.cnr.isti.epasmed.security.AuthoritiesConstants; +import it.cnr.isti.epasmed.service.MailService; +import org.springframework.data.domain.Sort; +import java.util.Collections; +import it.cnr.isti.epasmed.service.UserService; +import it.cnr.isti.epasmed.service.dto.UserDTO; +import it.cnr.isti.epasmed.web.rest.errors.BadRequestAlertException; +import it.cnr.isti.epasmed.web.rest.errors.EmailAlreadyUsedException; +import it.cnr.isti.epasmed.web.rest.errors.LoginAlreadyUsedException; + +import io.github.jhipster.web.util.HeaderUtil; +import io.github.jhipster.web.util.PaginationUtil; +import io.github.jhipster.web.util.ResponseUtil; + +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; +import org.springframework.beans.factory.annotation.Value; +import org.springframework.data.domain.Page; +import org.springframework.data.domain.Pageable; +import org.springframework.http.HttpHeaders; +import org.springframework.http.HttpStatus; +import org.springframework.http.ResponseEntity; +import org.springframework.security.access.prepost.PreAuthorize; +import org.springframework.web.bind.annotation.*; +import org.springframework.web.servlet.support.ServletUriComponentsBuilder; + +import javax.validation.Valid; +import java.net.URI; +import java.net.URISyntaxException; +import java.util.*; + +/** + * REST controller for managing users. + *

+ * This class accesses the {@link User} entity, and needs to fetch its collection of authorities. + *

+ * For a normal use-case, it would be better to have an eager relationship between User and Authority, + * and send everything to the client side: there would be no View Model and DTO, a lot less code, and an outer-join + * which would be good for performance. + *

+ * We use a View Model and a DTO for 3 reasons: + *

    + *
  • We want to keep a lazy association between the user and the authorities, because people will + * quite often do relationships with the user, and we don't want them to get the authorities all + * the time for nothing (for performance reasons). This is the #1 goal: we should not impact our users' + * application because of this use-case.
  • + *
  • Not having an outer join causes n+1 requests to the database. This is not a real issue as + * we have by default a second-level cache. This means on the first HTTP call we do the n+1 requests, + * but then all authorities come from the cache, so in fact it's much better than doing an outer join + * (which will get lots of data from the database, for each HTTP call).
  • + *
  • As this manages users, for security reasons, we'd rather have a DTO layer.
  • + *
+ *

+ * Another option would be to have a specific JPA entity graph to handle this case. + */ +@RestController +@RequestMapping("/api") +public class UserResource { + private static final List ALLOWED_ORDERED_PROPERTIES = Collections.unmodifiableList(Arrays.asList("id", "login", "firstName", "lastName", "email", "activated", "langKey")); + + private final Logger log = LoggerFactory.getLogger(UserResource.class); + + @Value("${jhipster.clientApp.name}") + private String applicationName; + + private final UserService userService; + + private final UserRepository userRepository; + + private final MailService mailService; + + public UserResource(UserService userService, UserRepository userRepository, MailService mailService) { + this.userService = userService; + this.userRepository = userRepository; + this.mailService = mailService; + } + + /** + * {@code POST /users} : Creates a new user. + *

+ * Creates a new user if the login and email are not already used, and sends an + * mail with an activation link. + * The user needs to be activated on creation. + * + * @param userDTO the user to create. + * @return the {@link ResponseEntity} with status {@code 201 (Created)} and with body the new user, or with status {@code 400 (Bad Request)} if the login or email is already in use. + * @throws URISyntaxException if the Location URI syntax is incorrect. + * @throws BadRequestAlertException {@code 400 (Bad Request)} if the login or email is already in use. + */ + @PostMapping("/users") + @PreAuthorize("hasAuthority(\"" + AuthoritiesConstants.ADMIN + "\")") + public ResponseEntity createUser(@Valid @RequestBody UserDTO userDTO) throws URISyntaxException { + log.debug("REST request to save User : {}", userDTO); + + if (userDTO.getId() != null) { + throw new BadRequestAlertException("A new user cannot already have an ID", "userManagement", "idexists"); + // Lowercase the user login before comparing with database + } else if (userRepository.findOneByLogin(userDTO.getLogin().toLowerCase()).isPresent()) { + throw new LoginAlreadyUsedException(); + } else if (userRepository.findOneByEmailIgnoreCase(userDTO.getEmail()).isPresent()) { + throw new EmailAlreadyUsedException(); + } else { + User newUser = userService.createUser(userDTO); + mailService.sendCreationEmail(newUser); + return ResponseEntity.created(new URI("/api/users/" + newUser.getLogin())) + .headers(HeaderUtil.createAlert(applicationName, "A user is created with identifier " + newUser.getLogin(), newUser.getLogin())) + .body(newUser); + } + } + + /** + * {@code PUT /users} : Updates an existing User. + * + * @param userDTO the user to update. + * @return the {@link ResponseEntity} with status {@code 200 (OK)} and with body the updated user. + * @throws EmailAlreadyUsedException {@code 400 (Bad Request)} if the email is already in use. + * @throws LoginAlreadyUsedException {@code 400 (Bad Request)} if the login is already in use. + */ + @PutMapping("/users") + @PreAuthorize("hasAuthority(\"" + AuthoritiesConstants.ADMIN + "\")") + public ResponseEntity updateUser(@Valid @RequestBody UserDTO userDTO) { + log.debug("REST request to update User : {}", userDTO); + Optional existingUser = userRepository.findOneByEmailIgnoreCase(userDTO.getEmail()); + if (existingUser.isPresent() && (!existingUser.get().getId().equals(userDTO.getId()))) { + throw new EmailAlreadyUsedException(); + } + existingUser = userRepository.findOneByLogin(userDTO.getLogin().toLowerCase()); + if (existingUser.isPresent() && (!existingUser.get().getId().equals(userDTO.getId()))) { + throw new LoginAlreadyUsedException(); + } + Optional updatedUser = userService.updateUser(userDTO); + + return ResponseUtil.wrapOrNotFound(updatedUser, + HeaderUtil.createAlert(applicationName, "A user is updated with identifier " + userDTO.getLogin(), userDTO.getLogin())); + } + + /** + * {@code GET /users} : get all users. + * + * @param pageable the pagination information. + * @return the {@link ResponseEntity} with status {@code 200 (OK)} and with body all users. + */ + @GetMapping("/users") + public ResponseEntity> getAllUsers(Pageable pageable) { + if (!onlyContainsAllowedProperties(pageable)) { + return ResponseEntity.badRequest().build(); + } + + final Page page = userService.getAllManagedUsers(pageable); + HttpHeaders headers = PaginationUtil.generatePaginationHttpHeaders(ServletUriComponentsBuilder.fromCurrentRequest(), page); + return new ResponseEntity<>(page.getContent(), headers, HttpStatus.OK); + } + + private boolean onlyContainsAllowedProperties(Pageable pageable) { + return pageable.getSort().stream().map(Sort.Order::getProperty).allMatch(ALLOWED_ORDERED_PROPERTIES::contains); + } + + /** + * Gets a list of all roles. + * @return a string list of all roles. + */ + @GetMapping("/users/authorities") + @PreAuthorize("hasAuthority(\"" + AuthoritiesConstants.ADMIN + "\")") + public List getAuthorities() { + return userService.getAuthorities(); + } + + /** + * {@code GET /users/:login} : get the "login" user. + * + * @param login the login of the user to find. + * @return the {@link ResponseEntity} with status {@code 200 (OK)} and with body the "login" user, or with status {@code 404 (Not Found)}. + */ + @GetMapping("/users/{login:" + Constants.LOGIN_REGEX + "}") + public ResponseEntity getUser(@PathVariable String login) { + log.debug("REST request to get User : {}", login); + return ResponseUtil.wrapOrNotFound( + userService.getUserWithAuthoritiesByLogin(login) + .map(UserDTO::new)); + } + + /** + * {@code DELETE /users/:login} : delete the "login" User. + * + * @param login the login of the user to delete. + * @return the {@link ResponseEntity} with status {@code 204 (NO_CONTENT)}. + */ + @DeleteMapping("/users/{login:" + Constants.LOGIN_REGEX + "}") + @PreAuthorize("hasAuthority(\"" + AuthoritiesConstants.ADMIN + "\")") + public ResponseEntity deleteUser(@PathVariable String login) { + log.debug("REST request to delete User: {}", login); + userService.deleteUser(login); + return ResponseEntity.noContent().headers(HeaderUtil.createAlert(applicationName, "A user is deleted with identifier " + login, login)).build(); + } +} diff --git a/src/main/java/it/cnr/isti/epasmed/web/rest/epas/EPASAffiliationsResource.java b/src/main/java/it/cnr/isti/epasmed/web/rest/epas/EPASAffiliationsResource.java new file mode 100644 index 0000000..56d38b8 --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/web/rest/epas/EPASAffiliationsResource.java @@ -0,0 +1,185 @@ +package it.cnr.isti.epasmed.web.rest.epas; + +import java.net.URI; +import java.net.URISyntaxException; +import java.util.List; +import java.util.Optional; + +import javax.validation.Valid; + +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; +import org.springframework.beans.factory.annotation.Value; +import org.springframework.http.ResponseEntity; +import org.springframework.security.access.prepost.PreAuthorize; +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 io.github.jhipster.web.util.HeaderUtil; +import io.github.jhipster.web.util.ResponseUtil; +import it.cnr.isti.epasmed.epas.dto.EPASAffiliationsDTO; +import it.cnr.isti.epasmed.epas.model.EPASAffiliations; +import it.cnr.isti.epasmed.epas.service.EPASAffiliationsService; +import it.cnr.isti.epasmed.security.AuthoritiesConstants; +import it.cnr.isti.epasmed.web.rest.errors.BadRequestAlertException; + +@RestController +@RequestMapping("/api/epas") +public class EPASAffiliationsResource { + + private final Logger log = LoggerFactory.getLogger(getClass()); + + @Value("${jhipster.clientApp.name}") + private String applicationName; + + private final EPASAffiliationsService epasAffiliationsService; + + public EPASAffiliationsResource(EPASAffiliationsService epasAffiliationsServiceService) { + this.epasAffiliationsService = epasAffiliationsServiceService; + + } + + /** + * {@code GET /affiliations/:id} : get affiliations by id. + * + * @param id the id of the affiliations to find. + * @return the {@link ResponseEntity} with status {@code 200 (OK)} and with body + * the EPAS Affiliations, or with status {@code 404 (Not Found)}. + */ + @GetMapping("/affiliations/{id}") + public ResponseEntity getEPASAffiliationsById(@PathVariable String id) { + log.info("REST request to get ePAS Affiliations by Id: {}", id); + EPASAffiliations epasAffiliations = epasAffiliationsService.getById(id); + log.info("Retrieved Affiliations: {}", epasAffiliations); + return ResponseUtil.wrapOrNotFound(Optional.of(epasAffiliations)); + + } + + /** + * {@code GET /affiliations/byGroup} : get the affiliations list by group. + * + * @param id the group id. + * @param includeInactive true if include inactive affiliations. + * @return the {@link ResponseEntity} with status {@code 200 (OK)} and with body + * the list of EPAS Affiliations, or with status + * {@code 404 (Not Found)}. + */ + @GetMapping("/affiliations/byGroup") + public ResponseEntity> getEPASAffiliationsByGroup(@RequestParam("id") String id, + @RequestParam("includeInactive") Optional includeInactive) { + log.info("REST request to get ePAS Affiliations list by Group Id: {}", id); + boolean includeInctv=false; + if (includeInactive.isPresent()) { + log.info("Include Inactive: {}", includeInactive.get()); + includeInctv=includeInactive.get(); + } + List epasAffiliationsList = epasAffiliationsService.getByGroup(id, includeInctv); + return ResponseUtil.wrapOrNotFound(Optional.of(epasAffiliationsList)); + } + + /** + * {@code GET /affiliations/byPerson} : get the affiliations list by person. + * + * @param id the person id. + * @param fiscalCode the person fiscal code. + * @return the {@link ResponseEntity} with status {@code 200 (OK)} and with body + * the list of EPAS Affiliations, or with status + * {@code 404 (Not Found)}. + */ + @GetMapping("/affiliations/byPerson") + public ResponseEntity> getEPASAffiliationsByPerson(@RequestParam("id") Optional id, + @RequestParam("fiscalCode") Optional fiscalCode) { + List epasAffiliationsList = null; + + if (id.isPresent()) { + log.info("REST request to get ePAS Affiliations by Person id: {}", id.get()); + epasAffiliationsList = epasAffiliationsService.getByPersonId(id.get()); + } else { + if (fiscalCode.isPresent()) { + log.info("REST request to get ePAS Affiliations by Person fiscal code: {}", fiscalCode.get()); + epasAffiliationsList = epasAffiliationsService.getByPersonFiscalcode(fiscalCode.get()); + } + } + return ResponseUtil.wrapOrNotFound(Optional.of(epasAffiliationsList)); + } + + /** + * {@code POST /affiliations} : Creates a new affiliations. + *

+ * Creates a new affiliations if the id is not already used. + * + * @param epasAffiliationsDTO the affiliation to create. + * @return the {@link ResponseEntity} with status {@code 201 (Created)} and with + * body the new ePAS Affiliation, or with status + * {@code 400 (Bad Request)} if the id is already in use. + * @throws URISyntaxException if the Location URI syntax is incorrect. + * @throws BadRequestAlertException {@code 400 (Bad Request)} if the id is + * already in use. + */ + @PostMapping("/affiliations") + @PreAuthorize("hasAuthority(\"" + AuthoritiesConstants.ADMIN + "\")") + public ResponseEntity createEPASAffiliations( + @Valid @RequestBody EPASAffiliationsDTO epasAffiliationsDTO) throws URISyntaxException { + log.debug("REST request to create EPAS Affiliation: {}", epasAffiliationsDTO); + + if (epasAffiliationsDTO.getId() != null) { + throw new BadRequestAlertException("A new ePAS Affiliation already have an ID", "EPASAffiliations", + "idexists"); + } else { + EPASAffiliations createdEPASAffiliations = epasAffiliationsService.create(epasAffiliationsDTO); + return ResponseEntity.created(new URI("/api/epas/affiliations/" + createdEPASAffiliations.getId())) + .headers(HeaderUtil.createAlert(applicationName, + "A ePAS Affiliations is created with identifier " + createdEPASAffiliations.getId(), + createdEPASAffiliations.getId())) + .body(createdEPASAffiliations); + } + } + + /** + * {@code PUT /affiliations/:id} : Updates an existing ePAS Affiliation by id. + * + * @param id the Affiliation id + * @param epasAffiliationsDTO the new ePAS Affiliation info. + * @return the {@link ResponseEntity} with status {@code 200 (OK)} and with body + * the updated ePAS Affiliations. + * + */ + @PutMapping("/affiliations/{id}") + @PreAuthorize("hasAuthority(\"" + AuthoritiesConstants.ADMIN + "\")") + public ResponseEntity updateEPASAffiliationsById(@PathVariable String id, + @Valid @RequestBody EPASAffiliationsDTO epasAffiliationsDTO) { + log.debug("REST request to update ePAS Affiliations : {} by {}", id, epasAffiliationsDTO); + epasAffiliationsService.updateById(id, epasAffiliationsDTO); + EPASAffiliations updatedEPASAffiliations = epasAffiliationsService.getById(id); + Optional oEPASAffiliations = Optional.of(updatedEPASAffiliations); + + return ResponseUtil.wrapOrNotFound(oEPASAffiliations, + HeaderUtil.createAlert(applicationName, + "A ePAS Affiliation is updated with identifier " + updatedEPASAffiliations.getId(), + updatedEPASAffiliations.getId())); + } + + /** + * {@code DELETE /affiliations/:id} : delete ePAS Affiliation by id. + * + * @param id the id of the ePAS Affiliation to delete. + * @return the {@link ResponseEntity} with status {@code 204 (NO_CONTENT)}. + */ + @DeleteMapping("/affiliations/{id}") + @PreAuthorize("hasAuthority(\"" + AuthoritiesConstants.ADMIN + "\")") + public ResponseEntity deleteEPASAffiliationById(@PathVariable String id) { + log.debug("REST request to delete ePAS Affiliation by id: {}", id); + epasAffiliationsService.deleteById(id); + return ResponseEntity.noContent() + .headers(HeaderUtil.createAlert(applicationName, "A Affiliation is deleted with identifier " + id, id)) + .build(); + } + +} diff --git a/src/main/java/it/cnr/isti/epasmed/web/rest/epas/EPASContractsResource.java b/src/main/java/it/cnr/isti/epasmed/web/rest/epas/EPASContractsResource.java new file mode 100644 index 0000000..a1b8f1c --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/web/rest/epas/EPASContractsResource.java @@ -0,0 +1,192 @@ +package it.cnr.isti.epasmed.web.rest.epas; + +import java.net.URI; +import java.net.URISyntaxException; +import java.util.List; +import java.util.Optional; + +import javax.validation.Valid; + +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; +import org.springframework.beans.factory.annotation.Value; +import org.springframework.http.ResponseEntity; +import org.springframework.security.access.prepost.PreAuthorize; +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 io.github.jhipster.web.util.HeaderUtil; +import io.github.jhipster.web.util.ResponseUtil; +import it.cnr.isti.epasmed.epas.dto.EPASContractsDTO; +import it.cnr.isti.epasmed.epas.model.EPASContracts; +import it.cnr.isti.epasmed.epas.service.EPASContractsService; +import it.cnr.isti.epasmed.security.AuthoritiesConstants; +import it.cnr.isti.epasmed.web.rest.errors.BadRequestAlertException; + +@RestController +@RequestMapping("/api/epas") +public class EPASContractsResource { + + private final Logger log = LoggerFactory.getLogger(getClass()); + + @Value("${jhipster.clientApp.name}") + private String applicationName; + + private final EPASContractsService epasContractsService; + + public EPASContractsResource(EPASContractsService epasContractsServiceService) { + this.epasContractsService = epasContractsServiceService; + + } + + /** + * {@code GET /contracts/:id} : get contracts by id. + * + * @param id the id of the contracts to find. + * @return the {@link ResponseEntity} with status {@code 200 (OK)} and with body + * the EPAS Contracts, or with status {@code 404 (Not Found)}. + */ + @GetMapping("/contracts/{id}") + public ResponseEntity getEPASContractsById(@PathVariable String id) { + log.info("REST request to get ePAS Contracts by Id: {}", id); + EPASContracts epasContracts = epasContractsService.getById(id); + log.info("Retrieved Contracts: {}", epasContracts); + return ResponseUtil.wrapOrNotFound(Optional.of(epasContracts)); + + } + + /** + * {@code GET /contracts/byPerson} : get contracts by Person id + * + * @param id the id of the person. + * @param fiscalCode the fiscal code of the person. + * @return the {@link ResponseEntity} with status {@code 200 (OK)} and with body + * the EPAS Contracts, or with status {@code 404 (Not Found)}. + */ + @GetMapping("/contracts/byPerson") + public ResponseEntity> getEPASContractsByPerson(@RequestParam("id") Optional id, + @RequestParam("fiscalCode") Optional fiscalCode) { + List epasContractsList = null; + if (id.isPresent()) { + log.info("REST request to get ePAS Contracts by Person: {}", id.get()); + epasContractsList = epasContractsService.getByPersonId(id.get()); + } else { + if (fiscalCode.isPresent()) { + log.info("REST request to get ePAS Contracts by Person fiscal code: {}", fiscalCode.get()); + epasContractsList = epasContractsService.getByPersonFiscalcode(fiscalCode.get()); + } + } + log.info("Retrieved List of Contracts: {}", epasContractsList); + return ResponseUtil.wrapOrNotFound(Optional.of(epasContractsList)); + } + + + /** + * {@code POST /contracts} : Creates a new contracts. + *

+ * Creates a new contracts if the id is not already used. + * + * @param epasContractsDTO the user to create. + * @return the {@link ResponseEntity} with status {@code 201 (Created)} and with + * body the new ePAS Contract, or with status {@code 400 (Bad Request)} + * if the id is already in use. + * @throws URISyntaxException if the Location URI syntax is incorrect. + * @throws BadRequestAlertException {@code 400 (Bad Request)} if the id is + * already in use. + */ + @PostMapping("/contracts") + @PreAuthorize("hasAuthority(\"" + AuthoritiesConstants.ADMIN + "\")") + public ResponseEntity createEPASContracts(@Valid @RequestBody EPASContractsDTO epasContractsDTO) + throws URISyntaxException { + log.debug("REST request to create EPAS Contract: {}", epasContractsDTO); + + if (epasContractsDTO.getId() != null) { + throw new BadRequestAlertException("A new ePAS Contract already have an ID", "EPASContracts", "idexists"); + } else { + EPASContracts createdEPASContracts = epasContractsService.create(epasContractsDTO); + return ResponseEntity.created(new URI("/api/epas/contracts/" + createdEPASContracts.getId())) + .headers(HeaderUtil.createAlert(applicationName, + "A ePAS Contracts is created with identifier " + createdEPASContracts.getId(), + createdEPASContracts.getPerson().getFullname())) + .body(createdEPASContracts); + } + } + + /** + * {@code PUT /contracts/:id} : Updates an existing ePAS Contract by id. + * + * @param epasContracts the new ePAS Contract info. + * @return the {@link ResponseEntity} with status {@code 200 (OK)} and with body + * the updated ePAS Contracts. + * + */ + @PutMapping("/contracts/{id}") + @PreAuthorize("hasAuthority(\"" + AuthoritiesConstants.ADMIN + "\")") + public ResponseEntity updateEPASContractsById(@PathVariable String id, + @Valid @RequestBody EPASContractsDTO epasContractsDTO) { + log.debug("REST request to update ePAS Contracts : {} by {}", id, epasContractsDTO); + epasContractsService.updateById(id, epasContractsDTO); + EPASContracts updatedEPASContracts = epasContractsService.getById(id); + Optional oEPASContracts = Optional.of(updatedEPASContracts); + + return ResponseUtil.wrapOrNotFound(oEPASContracts, + HeaderUtil.createAlert(applicationName, + "A ePAS Contract is updated with identifier " + updatedEPASContracts.getId(), + updatedEPASContracts.getPerson().getFullname())); + } + + /** + * {@code DELETE /contracts/setPreviousContract} : set ePAS Previous Contract. + * + * @param id the id of the ePAS Contract to set. + * @return the {@link ResponseEntity} with status {@code 204 (NO_CONTENT)}. + */ + @DeleteMapping("/contracts/setPreviousContract") + @PreAuthorize("hasAuthority(\"" + AuthoritiesConstants.ADMIN + "\")") + public ResponseEntity setEPASContractPrevious(@RequestParam("id") String id) { + log.debug("REST request Set Previous Contract ePAS by id: {}", id); + epasContractsService.setPreviousContract(id); + return ResponseEntity.noContent().headers(HeaderUtil.createAlert(applicationName, + "Setted to previous the EPAS Contract with identifier " + id, id)).build(); + } + + /** + * {@code DELETE /contracts/unsetPreviousContract} : unset ePAS Previous + * Contract. + * + * @param id the id of the ePAS Contract to unset. + * @return the {@link ResponseEntity} with status {@code 204 (NO_CONTENT)}. + */ + @DeleteMapping("/contracts/unsetPreviousContract") + @PreAuthorize("hasAuthority(\"" + AuthoritiesConstants.ADMIN + "\")") + public ResponseEntity setEPASContractUnsetPrevious(@RequestParam("id") String id) { + log.debug("REST request Unset Previous Contract ePAS by id: {}", id); + epasContractsService.setUnsetPreviousContract(id); + return ResponseEntity.noContent().headers(HeaderUtil.createAlert(applicationName, + "Unsetted to previous the EPAS Contract with identifier " + id, id)).build(); + } + + /** + * {@code DELETE /contracts/:id} : delete ePAS Contract by id. + * + * @param id the id of the ePAS Contract to delete. + * @return the {@link ResponseEntity} with status {@code 204 (NO_CONTENT)}. + */ + @DeleteMapping("/contracts/{id}") + @PreAuthorize("hasAuthority(\"" + AuthoritiesConstants.ADMIN + "\")") + public ResponseEntity deleteEPASContractById(@PathVariable String id) { + log.debug("REST request to delete ePAS Contract by id: {}", id); + epasContractsService.deleteById(id); + return ResponseEntity.noContent() + .headers(HeaderUtil.createAlert(applicationName, "A Contract is deleted with identifier " + id, id)) + .build(); + } + +} diff --git a/src/main/java/it/cnr/isti/epasmed/web/rest/epas/EPASGroupsResource.java b/src/main/java/it/cnr/isti/epasmed/web/rest/epas/EPASGroupsResource.java new file mode 100644 index 0000000..25869ef --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/web/rest/epas/EPASGroupsResource.java @@ -0,0 +1,158 @@ +package it.cnr.isti.epasmed.web.rest.epas; + +import java.net.URI; +import java.net.URISyntaxException; +import java.util.List; +import java.util.Optional; + +import javax.validation.Valid; + +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; +import org.springframework.beans.factory.annotation.Value; +import org.springframework.http.ResponseEntity; +import org.springframework.security.access.prepost.PreAuthorize; +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 io.github.jhipster.web.util.HeaderUtil; +import io.github.jhipster.web.util.ResponseUtil; +import it.cnr.isti.epasmed.epas.dto.EPASGroupsDTO; +import it.cnr.isti.epasmed.epas.model.EPASGroups; +import it.cnr.isti.epasmed.epas.service.EPASGroupsService; +import it.cnr.isti.epasmed.security.AuthoritiesConstants; +import it.cnr.isti.epasmed.web.rest.errors.BadRequestAlertException; + +@RestController +@RequestMapping("/api/epas") +public class EPASGroupsResource { + + private final Logger log = LoggerFactory.getLogger(getClass()); + + @Value("${jhipster.clientApp.name}") + private String applicationName; + + private final EPASGroupsService epasGroupsService; + + public EPASGroupsResource(EPASGroupsService epasGroupsServiceService) { + this.epasGroupsService = epasGroupsServiceService; + + } + + /** + * {@code GET /groups/:id} : get groups by id. + * + * @param id the id of the groups to find. + * @return the {@link ResponseEntity} with status {@code 200 (OK)} and with body + * the EPAS Groups, or with status {@code 404 (Not Found)}. + */ + @GetMapping("/groups/{id}") + public ResponseEntity getEPASGroupsById(@PathVariable String id) { + log.info("REST request to get ePAS Groups by Id: {}", id); + EPASGroups epasGroups = epasGroupsService.getById(id); + log.info("Retrieved Groups: {}", epasGroups); + return ResponseUtil.wrapOrNotFound(Optional.of(epasGroups)); + + } + + + + /** + * {@code GET /groups} : get the groups list. + * + * @param officeId the office id. + * @return the {@link ResponseEntity} with status {@code 200 (OK)} and with body + * the list of EPAS Groups, or with status {@code 404 (Not Found)}. + */ + @GetMapping("/groups") + public ResponseEntity> getEPASGroups(@RequestParam("officeId") String id) { + log.info("REST request to get ePAS Groups list for Office: {}",id); + List epasGroupsList = epasGroupsService.getList(id); + return ResponseUtil.wrapOrNotFound(Optional.of(epasGroupsList)); + + } + + /** + * {@code POST /groups} : Creates a new groups. + *

+ * Creates a new groups if the id is not already used. + * + * @param epasGroupsDTO the user to create. + * @return the {@link ResponseEntity} with status {@code 201 (Created)} and with + * body the new ePAS Group, or with status {@code 400 (Bad Request)} if + * the id is already in use. + * @throws URISyntaxException if the Location URI syntax is incorrect. + * @throws BadRequestAlertException {@code 400 (Bad Request)} if the id is + * already in use. + */ + @PostMapping("/groups") + @PreAuthorize("hasAuthority(\"" + AuthoritiesConstants.ADMIN + "\")") + public ResponseEntity createEPASGroups(@Valid @RequestBody EPASGroupsDTO epasGroupsDTO) + throws URISyntaxException { + log.debug("REST request to create EPAS Group: {}", epasGroupsDTO); + + if (epasGroupsDTO.getId() != null) { + throw new BadRequestAlertException("A new ePAS Group already have an ID", "EPASGroups", "idexists"); + } else { + EPASGroups createdEPASGroups = epasGroupsService.create(epasGroupsDTO); + return ResponseEntity.created(new URI("/api/epas/groups/" + createdEPASGroups.getId())) + .headers(HeaderUtil.createAlert(applicationName, + "A ePAS Groups is created with identifier " + createdEPASGroups.getId(), + createdEPASGroups.getName())) + .body(createdEPASGroups); + } + } + + /** + * {@code PUT /groups/:id} : Updates an existing ePAS Group by id. + * + * @param epasGroups the new ePAS Group info. + * @return the {@link ResponseEntity} with status {@code 200 (OK)} and with body + * the updated ePAS Groups. + * + */ + @PutMapping("/groups/{id}") + @PreAuthorize("hasAuthority(\"" + AuthoritiesConstants.ADMIN + "\")") + public ResponseEntity updateEPASGroupsById(@PathVariable String id, + @Valid @RequestBody EPASGroupsDTO epasGroupsDTO) { + log.debug("REST request to update ePAS Groups : {} by {}", id, epasGroupsDTO); + epasGroupsService.updateById(id,epasGroupsDTO); + EPASGroups updatedEPASGroups=epasGroupsService.getById(id); + Optional oEPASGroups = Optional.of(updatedEPASGroups); + + return ResponseUtil.wrapOrNotFound(oEPASGroups, HeaderUtil.createAlert(applicationName, + "A ePAS Group is updated with identifier " + updatedEPASGroups.getId(), updatedEPASGroups.getName())); + } + + + + + + /** + * {@code DELETE /groups/:id} : delete ePAS Group by id. + * + * @param id the id of the ePAS Group to delete. + * @return the {@link ResponseEntity} with status {@code 204 (NO_CONTENT)}. + */ + @DeleteMapping("/groups/{id}") + @PreAuthorize("hasAuthority(\"" + AuthoritiesConstants.ADMIN + "\")") + public ResponseEntity deleteEPASGroupById(@PathVariable String id) { + log.debug("REST request to delete ePAS Group by id: {}", id); + epasGroupsService.deleteById(id); + return ResponseEntity.noContent() + .headers(HeaderUtil.createAlert(applicationName, "A Group is deleted with identifier " + id, id)) + .build(); + } + + + + + +} diff --git a/src/main/java/it/cnr/isti/epasmed/web/rest/epas/EPASPersonsResource.java b/src/main/java/it/cnr/isti/epasmed/web/rest/epas/EPASPersonsResource.java new file mode 100644 index 0000000..7107864 --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/web/rest/epas/EPASPersonsResource.java @@ -0,0 +1,202 @@ +package it.cnr.isti.epasmed.web.rest.epas; + +import java.net.URI; +import java.net.URISyntaxException; +import java.util.List; +import java.util.Optional; + +import javax.validation.Valid; + +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; +import org.springframework.beans.factory.annotation.Value; +import org.springframework.http.ResponseEntity; +import org.springframework.security.access.prepost.PreAuthorize; +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 io.github.jhipster.web.util.HeaderUtil; +import io.github.jhipster.web.util.ResponseUtil; +import it.cnr.isti.epasmed.epas.dto.EPASPersonsDTO; +import it.cnr.isti.epasmed.epas.model.EPASPersons; +import it.cnr.isti.epasmed.epas.service.EPASPersonsService; +import it.cnr.isti.epasmed.security.AuthoritiesConstants; +import it.cnr.isti.epasmed.web.rest.errors.BadRequestAlertException; + +@RestController +@RequestMapping("/api/epas") +public class EPASPersonsResource { + + private final Logger log = LoggerFactory.getLogger(getClass()); + + @Value("${jhipster.clientApp.name}") + private String applicationName; + + private final EPASPersonsService epasPersonsService; + + public EPASPersonsResource(EPASPersonsService epasPersonsServiceService) { + this.epasPersonsService = epasPersonsServiceService; + + } + + /** + * {@code GET /persons/:id} : get persons by id. + * + * @param id the id of the persons to find. + * @return the {@link ResponseEntity} with status {@code 200 (OK)} and with body + * the EPAS Persons, or with status {@code 404 (Not Found)}. + */ + @GetMapping("/persons/{id}") + public ResponseEntity getEPASPersonsById(@PathVariable String id) { + log.info("REST request to get ePAS Persons by Id: {}", id); + EPASPersons epasPersons = epasPersonsService.getById(id); + log.info("Retrieved Persons: {}", epasPersons); + return ResponseUtil.wrapOrNotFound(Optional.of(epasPersons)); + + } + + /** + * {@code GET /persons/show} : get the persons by fiscal code. + * + * @param fc the fiscal code of the persons to find. + * @return the {@link ResponseEntity} with status {@code 200 (OK)} and with body + * the ePAS Persons, or with status {@code 404 (Not Found)}. + */ + @GetMapping("/persons/show") + public ResponseEntity getEPASPersonsByFiscalCode(@RequestParam("fiscalCode") String fc) { + log.info("REST request to get ePAS Persons by fiscal code: {}", fc); + EPASPersons epasPersons = epasPersonsService.getByFiscalCode(fc); + log.info("Retrieved Persons: {}", epasPersons); + return ResponseUtil.wrapOrNotFound(Optional.of(epasPersons)); + + } + + /** + * {@code GET /persons} : get the persons list. + * + * @param officeId the office id. + * @return the {@link ResponseEntity} with status {@code 200 (OK)} and with body + * the list of EPAS Persons, or with status {@code 404 (Not Found)}. + */ + @GetMapping("/persons") + public ResponseEntity> getEPASPersons(@RequestParam("officeId") String id) { + log.info("REST request to get ePAS Persons list for Office: {}", id); + List epasPersonsList = epasPersonsService.getList(id); + return ResponseUtil.wrapOrNotFound(Optional.of(epasPersonsList)); + + } + + /** + * {@code POST /persons} : Creates a new persons. + *

+ * Creates a new persons if the id is not already used. + * + * @param epasPersonsDTO the user to create. + * @return the {@link ResponseEntity} with status {@code 201 (Created)} and with + * body the new ePAS Person, or with status {@code 400 (Bad Request)} if + * the id is already in use. + * @throws URISyntaxException if the Location URI syntax is incorrect. + * @throws BadRequestAlertException {@code 400 (Bad Request)} if the id is + * already in use. + */ + @PostMapping("/persons") + @PreAuthorize("hasAuthority(\"" + AuthoritiesConstants.ADMIN + "\")") + public ResponseEntity createEPASPersons(@Valid @RequestBody EPASPersonsDTO epasPersonsDTO) + throws URISyntaxException { + log.debug("REST request to create EPAS Person: {}", epasPersonsDTO); + + if (epasPersonsDTO.getId() != null) { + throw new BadRequestAlertException("A new ePAS Person already have an ID", "EPASPersons", "idexists"); + } else { + EPASPersons createdEPASPersons = epasPersonsService.create(epasPersonsDTO); + return ResponseEntity.created(new URI("/api/epas/persons/" + createdEPASPersons.getId())) + .headers(HeaderUtil.createAlert(applicationName, + "A ePAS Persons is created with identifier " + createdEPASPersons.getId(), + createdEPASPersons.getFullname())) + .body(createdEPASPersons); + } + } + + /** + * {@code PUT /persons/:id} : Updates an existing ePAS Person by id. + * + * @param epasPersonsDTO the new ePAS Person info. + * @return the {@link ResponseEntity} with status {@code 200 (OK)} and with body + * the updated ePAS Persons. + * + */ + @PutMapping("/persons/{id}") + @PreAuthorize("hasAuthority(\"" + AuthoritiesConstants.ADMIN + "\")") + public ResponseEntity updateEPASPersonsById(@PathVariable String id, + @Valid @RequestBody EPASPersonsDTO epasPersonsDTO) { + log.debug("REST request to update ePAS Persons : {} by {}", id, epasPersonsDTO); + epasPersonsService.updateById(id, epasPersonsDTO); + EPASPersons updatedEPASPersons=epasPersonsService.getById(id); + Optional oEPASPersons = Optional.of(updatedEPASPersons); + + return ResponseUtil.wrapOrNotFound(oEPASPersons, HeaderUtil.createAlert(applicationName, + "A ePAS Person is updated with identifier " + updatedEPASPersons.getId(), updatedEPASPersons.getFullname())); + } + + /** + * {@code PUT /persons/update} : Updates an existing ePAS Person by fiscal code. + * + * @param fc the fiscal code of the persons to update. + * @param epasPersonsDTO the new ePAS Person info. + * @return the {@link ResponseEntity} with status {@code 200 (OK)} and with body + * the updated ePAS Persons. + * + */ + @PutMapping("/persons/update") + @PreAuthorize("hasAuthority(\"" + AuthoritiesConstants.ADMIN + "\")") + public ResponseEntity updateEPASPersons(@RequestParam("fiscalCode") String fc, + @Valid @RequestBody EPASPersonsDTO epasPersonsDTO) { + log.debug("REST request to update ePAS Persons : {}", epasPersonsDTO); + epasPersonsService.updateByFiscalCode(fc, epasPersonsDTO); + EPASPersons updatedEPASPersons=epasPersonsService.getByFiscalCode(fc); + Optional oEPASPersons = Optional.of(updatedEPASPersons); + + return ResponseUtil.wrapOrNotFound(oEPASPersons, HeaderUtil.createAlert(applicationName, + "A ePAS Person is updated with id " + updatedEPASPersons.getId(), updatedEPASPersons.getFullname())); + } + + /** + * {@code DELETE /persons/:id} : delete ePAS Person by id. + * + * @param id the id of the ePAS Person to delete. + * @return the {@link ResponseEntity} with status {@code 204 (NO_CONTENT)}. + */ + @DeleteMapping("/persons/{id}") + @PreAuthorize("hasAuthority(\"" + AuthoritiesConstants.ADMIN + "\")") + public ResponseEntity deleteEPASPersonById(@PathVariable String id) { + log.debug("REST request to delete ePAS Person by id: {}", id); + epasPersonsService.deleteById(id); + return ResponseEntity.noContent() + .headers(HeaderUtil.createAlert(applicationName, "A Person is deleted with identifier " + id, id)) + .build(); + } + + /** + * {@code DELETE /persons/delete} : delete ePAS Person by fiscal code. + * + * @param id the fiscal code of the ePAS Person to delete. + * @return the {@link ResponseEntity} with status {@code 204 (NO_CONTENT)}. + */ + @DeleteMapping("/persons/delete") + @PreAuthorize("hasAuthority(\"" + AuthoritiesConstants.ADMIN + "\")") + public ResponseEntity deleteEPASPersonByFiscalCode(@RequestParam("fiscalCode") String fc) { + log.debug("REST request to delete ePAS Person by fiscal code: {}", fc); + epasPersonsService.deleteByFiscalCode(fc); + return ResponseEntity.noContent() + .headers(HeaderUtil.createAlert(applicationName, "A Person is deleted with fiscal code " + fc, fc)) + .build(); + } + +} diff --git a/src/main/java/it/cnr/isti/epasmed/web/rest/epas/EPASWorkingTimeTypesResource.java b/src/main/java/it/cnr/isti/epasmed/web/rest/epas/EPASWorkingTimeTypesResource.java new file mode 100644 index 0000000..d5e8e16 --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/web/rest/epas/EPASWorkingTimeTypesResource.java @@ -0,0 +1,67 @@ +package it.cnr.isti.epasmed.web.rest.epas; + +import java.util.List; +import java.util.Optional; + +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; +import org.springframework.beans.factory.annotation.Value; +import org.springframework.http.ResponseEntity; +import org.springframework.web.bind.annotation.GetMapping; +import org.springframework.web.bind.annotation.PathVariable; +import org.springframework.web.bind.annotation.RequestMapping; +import org.springframework.web.bind.annotation.RequestParam; +import org.springframework.web.bind.annotation.RestController; + +import io.github.jhipster.web.util.ResponseUtil; +import it.cnr.isti.epasmed.epas.model.EPASWorkingTimeTypes; +import it.cnr.isti.epasmed.epas.service.EPASWorkingTimeTypesService; + +@RestController +@RequestMapping("/api/epas") +public class EPASWorkingTimeTypesResource { + + private final Logger log = LoggerFactory.getLogger(getClass()); + + @Value("${jhipster.clientApp.name}") + private String applicationName; + + private final EPASWorkingTimeTypesService epasWorkingTimeTypesService; + + public EPASWorkingTimeTypesResource(EPASWorkingTimeTypesService epasWorkingTimeTypesServiceService) { + this.epasWorkingTimeTypesService = epasWorkingTimeTypesServiceService; + + } + + /** + * {@code GET /workingtimetypes/:id} : get workingtimetypes by id. + * + * @param id the id of the workingtimetypes to find. + * @return the {@link ResponseEntity} with status {@code 200 (OK)} and with body + * the EPAS WorkingTimeTypes, or with status {@code 404 (Not Found)}. + */ + @GetMapping("/workingtimetypes/{id}") + public ResponseEntity getEPASWorkingTimeTypesById(@PathVariable String id) { + log.info("REST request to get ePAS WorkingTimeTypes by Id: {}", id); + EPASWorkingTimeTypes epasWorkingTimeTypes = epasWorkingTimeTypesService.getById(id); + log.info("Retrieved WorkingTimeTypes: {}", epasWorkingTimeTypes); + return ResponseUtil.wrapOrNotFound(Optional.of(epasWorkingTimeTypes)); + + } + + /** + * {@code GET /workingtimetypes} : get the workingtimetypes list. + * + * @param officeId the office id. + * @return the {@link ResponseEntity} with status {@code 200 (OK)} and with body + * the list of EPASWorkingTimeTypes, or with status {@code 404 (Not Found)}. + */ + @GetMapping("/workingtimetypes") + public ResponseEntity> getEPASWorkingTimeTypes(@RequestParam("officeId") String id) { + log.info("REST request to get ePAS WorkingTimeTypes list for Office: {}", id); + List epasWorkingTimeTypesList = epasWorkingTimeTypesService.getList(id); + return ResponseUtil.wrapOrNotFound(Optional.of(epasWorkingTimeTypesList)); + + } + +} diff --git a/src/main/java/it/cnr/isti/epasmed/web/rest/errors/BadRequestAlertException.java b/src/main/java/it/cnr/isti/epasmed/web/rest/errors/BadRequestAlertException.java new file mode 100644 index 0000000..165d621 --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/web/rest/errors/BadRequestAlertException.java @@ -0,0 +1,42 @@ +package it.cnr.isti.epasmed.web.rest.errors; + +import org.zalando.problem.AbstractThrowableProblem; +import org.zalando.problem.Status; + +import java.net.URI; +import java.util.HashMap; +import java.util.Map; + +public class BadRequestAlertException extends AbstractThrowableProblem { + + private static final long serialVersionUID = 1L; + + private final String entityName; + + private final String errorKey; + + public BadRequestAlertException(String defaultMessage, String entityName, String errorKey) { + this(ErrorConstants.DEFAULT_TYPE, defaultMessage, entityName, errorKey); + } + + public BadRequestAlertException(URI type, String defaultMessage, String entityName, String errorKey) { + super(type, defaultMessage, Status.BAD_REQUEST, null, null, null, getAlertParameters(entityName, errorKey)); + this.entityName = entityName; + this.errorKey = errorKey; + } + + public String getEntityName() { + return entityName; + } + + public String getErrorKey() { + return errorKey; + } + + private static Map getAlertParameters(String entityName, String errorKey) { + Map parameters = new HashMap<>(); + parameters.put("message", "error." + errorKey); + parameters.put("params", entityName); + return parameters; + } +} diff --git a/src/main/java/it/cnr/isti/epasmed/web/rest/errors/EmailAlreadyUsedException.java b/src/main/java/it/cnr/isti/epasmed/web/rest/errors/EmailAlreadyUsedException.java new file mode 100644 index 0000000..3392520 --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/web/rest/errors/EmailAlreadyUsedException.java @@ -0,0 +1,10 @@ +package it.cnr.isti.epasmed.web.rest.errors; + +public class EmailAlreadyUsedException extends BadRequestAlertException { + + private static final long serialVersionUID = 1L; + + public EmailAlreadyUsedException() { + super(ErrorConstants.EMAIL_ALREADY_USED_TYPE, "Email is already in use!", "userManagement", "emailexists"); + } +} diff --git a/src/main/java/it/cnr/isti/epasmed/web/rest/errors/ErrorConstants.java b/src/main/java/it/cnr/isti/epasmed/web/rest/errors/ErrorConstants.java new file mode 100644 index 0000000..4593366 --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/web/rest/errors/ErrorConstants.java @@ -0,0 +1,18 @@ +package it.cnr.isti.epasmed.web.rest.errors; + +import java.net.URI; + +public final class ErrorConstants { + + public static final String ERR_CONCURRENCY_FAILURE = "error.concurrencyFailure"; + public static final String ERR_VALIDATION = "error.validation"; + public static final String PROBLEM_BASE_URL = "https://www.jhipster.tech/problem"; + public static final URI DEFAULT_TYPE = URI.create(PROBLEM_BASE_URL + "/problem-with-message"); + public static final URI CONSTRAINT_VIOLATION_TYPE = URI.create(PROBLEM_BASE_URL + "/constraint-violation"); + public static final URI INVALID_PASSWORD_TYPE = URI.create(PROBLEM_BASE_URL + "/invalid-password"); + public static final URI EMAIL_ALREADY_USED_TYPE = URI.create(PROBLEM_BASE_URL + "/email-already-used"); + public static final URI LOGIN_ALREADY_USED_TYPE = URI.create(PROBLEM_BASE_URL + "/login-already-used"); + + private ErrorConstants() { + } +} diff --git a/src/main/java/it/cnr/isti/epasmed/web/rest/errors/ExceptionTranslator.java b/src/main/java/it/cnr/isti/epasmed/web/rest/errors/ExceptionTranslator.java new file mode 100644 index 0000000..f10c340 --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/web/rest/errors/ExceptionTranslator.java @@ -0,0 +1,200 @@ +package it.cnr.isti.epasmed.web.rest.errors; + +import io.github.jhipster.config.JHipsterConstants; +import io.github.jhipster.web.util.HeaderUtil; + +import org.apache.commons.lang3.StringUtils; +import org.springframework.beans.factory.annotation.Value; +import org.springframework.dao.ConcurrencyFailureException; +import org.springframework.dao.DataAccessException; +import org.springframework.http.ResponseEntity; +import org.springframework.http.converter.HttpMessageConversionException; +import org.springframework.validation.BindingResult; +import org.springframework.web.bind.MethodArgumentNotValidException; +import org.springframework.web.bind.annotation.ControllerAdvice; +import org.springframework.web.bind.annotation.ExceptionHandler; +import org.springframework.web.context.request.NativeWebRequest; +import org.springframework.core.env.Environment; +import org.zalando.problem.DefaultProblem; +import org.zalando.problem.Problem; +import org.zalando.problem.ProblemBuilder; +import org.zalando.problem.Status; +import org.zalando.problem.StatusType; +import org.zalando.problem.spring.web.advice.ProblemHandling; +import org.zalando.problem.spring.web.advice.security.SecurityAdviceTrait; +import org.zalando.problem.violations.ConstraintViolationProblem; + +import javax.annotation.Nonnull; +import javax.annotation.Nullable; +import javax.servlet.http.HttpServletRequest; +import java.net.URI; +import java.util.Arrays; +import java.util.Collection; +import java.util.List; +import java.util.Optional; +import java.util.stream.Collectors; + +/** + * Controller advice to translate the server side exceptions to client-friendly json structures. + * The error response follows RFC7807 - Problem Details for HTTP APIs (https://tools.ietf.org/html/rfc7807). + */ +@ControllerAdvice +public class ExceptionTranslator implements ProblemHandling, SecurityAdviceTrait { + + private static final String FIELD_ERRORS_KEY = "fieldErrors"; + private static final String MESSAGE_KEY = "message"; + private static final String PATH_KEY = "path"; + private static final String VIOLATIONS_KEY = "violations"; + + @Value("${jhipster.clientApp.name}") + private String applicationName; + + private final Environment env; + + public ExceptionTranslator(Environment env) { + this.env = env; + } + + /** + * Post-process the Problem payload to add the message key for the front-end if needed. + */ + @Override + public ResponseEntity process(@Nullable ResponseEntity entity, NativeWebRequest request) { + if (entity == null) { + return entity; + } + Problem problem = entity.getBody(); + if (!(problem instanceof ConstraintViolationProblem || problem instanceof DefaultProblem)) { + return entity; + } + ProblemBuilder builder = Problem.builder() + .withType(Problem.DEFAULT_TYPE.equals(problem.getType()) ? ErrorConstants.DEFAULT_TYPE : problem.getType()) + .withStatus(problem.getStatus()) + .withTitle(problem.getTitle()) + .with(PATH_KEY, request.getNativeRequest(HttpServletRequest.class).getRequestURI()); + + if (problem instanceof ConstraintViolationProblem) { + builder + .with(VIOLATIONS_KEY, ((ConstraintViolationProblem) problem).getViolations()) + .with(MESSAGE_KEY, ErrorConstants.ERR_VALIDATION); + } else { + builder + .withCause(((DefaultProblem) problem).getCause()) + .withDetail(problem.getDetail()) + .withInstance(problem.getInstance()); + problem.getParameters().forEach(builder::with); + if (!problem.getParameters().containsKey(MESSAGE_KEY) && problem.getStatus() != null) { + builder.with(MESSAGE_KEY, "error.http." + problem.getStatus().getStatusCode()); + } + } + return new ResponseEntity<>(builder.build(), entity.getHeaders(), entity.getStatusCode()); + } + + @Override + public ResponseEntity handleMethodArgumentNotValid(MethodArgumentNotValidException ex, @Nonnull NativeWebRequest request) { + BindingResult result = ex.getBindingResult(); + List fieldErrors = result.getFieldErrors().stream() + .map(f -> new FieldErrorVM(f.getObjectName().replaceFirst("DTO$", ""), f.getField(), f.getCode())) + .collect(Collectors.toList()); + + Problem problem = Problem.builder() + .withType(ErrorConstants.CONSTRAINT_VIOLATION_TYPE) + .withTitle("Method argument not valid") + .withStatus(defaultConstraintViolationStatus()) + .with(MESSAGE_KEY, ErrorConstants.ERR_VALIDATION) + .with(FIELD_ERRORS_KEY, fieldErrors) + .build(); + return create(ex, problem, request); + } + + @ExceptionHandler + public ResponseEntity handleEmailAlreadyUsedException(it.cnr.isti.epasmed.service.EmailAlreadyUsedException ex, NativeWebRequest request) { + EmailAlreadyUsedException problem = new EmailAlreadyUsedException(); + return create(problem, request, HeaderUtil.createFailureAlert(applicationName, false, problem.getEntityName(), problem.getErrorKey(), problem.getMessage())); + } + + @ExceptionHandler + public ResponseEntity handleUsernameAlreadyUsedException(it.cnr.isti.epasmed.service.UsernameAlreadyUsedException ex, NativeWebRequest request) { + LoginAlreadyUsedException problem = new LoginAlreadyUsedException(); + return create(problem, request, HeaderUtil.createFailureAlert(applicationName, false, problem.getEntityName(), problem.getErrorKey(), problem.getMessage())); + } + + @ExceptionHandler + public ResponseEntity handleInvalidPasswordException(it.cnr.isti.epasmed.service.InvalidPasswordException ex, NativeWebRequest request) { + return create(new InvalidPasswordException(), request); + } + + @ExceptionHandler + public ResponseEntity handleBadRequestAlertException(BadRequestAlertException ex, NativeWebRequest request) { + return create(ex, request, HeaderUtil.createFailureAlert(applicationName, false, ex.getEntityName(), ex.getErrorKey(), ex.getMessage())); + } + + @ExceptionHandler + public ResponseEntity handleConcurrencyFailure(ConcurrencyFailureException ex, NativeWebRequest request) { + Problem problem = Problem.builder() + .withStatus(Status.CONFLICT) + .with(MESSAGE_KEY, ErrorConstants.ERR_CONCURRENCY_FAILURE) + .build(); + return create(ex, problem, request); + } + + @Override + public ProblemBuilder prepare(final Throwable throwable, final StatusType status, final URI type) { + + Collection activeProfiles = Arrays.asList(env.getActiveProfiles()); + + if (activeProfiles.contains(JHipsterConstants.SPRING_PROFILE_PRODUCTION)) { + if (throwable instanceof HttpMessageConversionException) { + return Problem.builder() + .withType(type) + .withTitle(status.getReasonPhrase()) + .withStatus(status) + .withDetail("Unable to convert http message") + .withCause(Optional.ofNullable(throwable.getCause()) + .filter(cause -> isCausalChainsEnabled()) + .map(this::toProblem) + .orElse(null)); + } + + if (throwable instanceof DataAccessException) { + return Problem.builder() + .withType(type) + .withTitle(status.getReasonPhrase()) + .withStatus(status) + .withDetail("Failure during data access") + .withCause(Optional.ofNullable(throwable.getCause()) + .filter(cause -> isCausalChainsEnabled()) + .map(this::toProblem) + .orElse(null)); + } + + if (containsPackageName(throwable.getMessage())) { + return Problem.builder() + .withType(type) + .withTitle(status.getReasonPhrase()) + .withStatus(status) + .withDetail("Unexpected runtime exception") + .withCause(Optional.ofNullable(throwable.getCause()) + .filter(cause -> isCausalChainsEnabled()) + .map(this::toProblem) + .orElse(null)); + } + } + + return Problem.builder() + .withType(type) + .withTitle(status.getReasonPhrase()) + .withStatus(status) + .withDetail(throwable.getMessage()) + .withCause(Optional.ofNullable(throwable.getCause()) + .filter(cause -> isCausalChainsEnabled()) + .map(this::toProblem) + .orElse(null)); + } + + private boolean containsPackageName(String message) { + + // This list is for sure not complete + return StringUtils.containsAny(message, "org.", "java.", "net.", "javax.", "com.", "io.", "de.", "it.cnr.isti.epasmed"); + } +} diff --git a/src/main/java/it/cnr/isti/epasmed/web/rest/errors/FieldErrorVM.java b/src/main/java/it/cnr/isti/epasmed/web/rest/errors/FieldErrorVM.java new file mode 100644 index 0000000..2b90fd2 --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/web/rest/errors/FieldErrorVM.java @@ -0,0 +1,33 @@ +package it.cnr.isti.epasmed.web.rest.errors; + +import java.io.Serializable; + +public class FieldErrorVM implements Serializable { + + private static final long serialVersionUID = 1L; + + private final String objectName; + + private final String field; + + private final String message; + + public FieldErrorVM(String dto, String field, String message) { + this.objectName = dto; + this.field = field; + this.message = message; + } + + public String getObjectName() { + return objectName; + } + + public String getField() { + return field; + } + + public String getMessage() { + return message; + } + +} diff --git a/src/main/java/it/cnr/isti/epasmed/web/rest/errors/InvalidPasswordException.java b/src/main/java/it/cnr/isti/epasmed/web/rest/errors/InvalidPasswordException.java new file mode 100644 index 0000000..21f2c80 --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/web/rest/errors/InvalidPasswordException.java @@ -0,0 +1,13 @@ +package it.cnr.isti.epasmed.web.rest.errors; + +import org.zalando.problem.AbstractThrowableProblem; +import org.zalando.problem.Status; + +public class InvalidPasswordException extends AbstractThrowableProblem { + + private static final long serialVersionUID = 1L; + + public InvalidPasswordException() { + super(ErrorConstants.INVALID_PASSWORD_TYPE, "Incorrect password", Status.BAD_REQUEST); + } +} diff --git a/src/main/java/it/cnr/isti/epasmed/web/rest/errors/LoginAlreadyUsedException.java b/src/main/java/it/cnr/isti/epasmed/web/rest/errors/LoginAlreadyUsedException.java new file mode 100644 index 0000000..99d560a --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/web/rest/errors/LoginAlreadyUsedException.java @@ -0,0 +1,10 @@ +package it.cnr.isti.epasmed.web.rest.errors; + +public class LoginAlreadyUsedException extends BadRequestAlertException { + + private static final long serialVersionUID = 1L; + + public LoginAlreadyUsedException() { + super(ErrorConstants.LOGIN_ALREADY_USED_TYPE, "Login name already used!", "userManagement", "userexists"); + } +} diff --git a/src/main/java/it/cnr/isti/epasmed/web/rest/errors/package-info.java b/src/main/java/it/cnr/isti/epasmed/web/rest/errors/package-info.java new file mode 100644 index 0000000..3cc1090 --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/web/rest/errors/package-info.java @@ -0,0 +1,6 @@ +/** + * Specific errors used with Zalando's "problem-spring-web" library. + * + * More information on https://github.com/zalando/problem-spring-web + */ +package it.cnr.isti.epasmed.web.rest.errors; diff --git a/src/main/java/it/cnr/isti/epasmed/web/rest/package-info.java b/src/main/java/it/cnr/isti/epasmed/web/rest/package-info.java new file mode 100644 index 0000000..1b553f5 --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/web/rest/package-info.java @@ -0,0 +1,4 @@ +/** + * Spring MVC REST controllers. + */ +package it.cnr.isti.epasmed.web.rest; diff --git a/src/main/java/it/cnr/isti/epasmed/web/rest/sistemainformativo/SIMasterLogResource.java b/src/main/java/it/cnr/isti/epasmed/web/rest/sistemainformativo/SIMasterLogResource.java new file mode 100644 index 0000000..9966c0d --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/web/rest/sistemainformativo/SIMasterLogResource.java @@ -0,0 +1,29 @@ +package it.cnr.isti.epasmed.web.rest.sistemainformativo; + +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; +import org.springframework.web.bind.annotation.GetMapping; +import org.springframework.web.bind.annotation.RequestMapping; +import org.springframework.web.bind.annotation.RestController; + +import it.cnr.isti.epasmed.sistemainformativo.service.SIMasterLogService; + +@RestController +@RequestMapping("/api/sistemainformativo") +public class SIMasterLogResource { + + private final Logger log = LoggerFactory.getLogger(SIMasterLogResource.class); + + private final SIMasterLogService masterLogService; + + public SIMasterLogResource(SIMasterLogService masterLogService) { + super(); + this.masterLogService = masterLogService; + } + + @GetMapping("/masterlog/count") + public int getCount() { + return masterLogService.count(); + } + +} diff --git a/src/main/java/it/cnr/isti/epasmed/web/rest/vm/KeyAndPasswordVM.java b/src/main/java/it/cnr/isti/epasmed/web/rest/vm/KeyAndPasswordVM.java new file mode 100644 index 0000000..b5f3dca --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/web/rest/vm/KeyAndPasswordVM.java @@ -0,0 +1,27 @@ +package it.cnr.isti.epasmed.web.rest.vm; + +/** + * View Model object for storing the user's key and password. + */ +public class KeyAndPasswordVM { + + private String key; + + private String newPassword; + + public String getKey() { + return key; + } + + public void setKey(String key) { + this.key = key; + } + + public String getNewPassword() { + return newPassword; + } + + public void setNewPassword(String newPassword) { + this.newPassword = newPassword; + } +} diff --git a/src/main/java/it/cnr/isti/epasmed/web/rest/vm/ManagedUserVM.java b/src/main/java/it/cnr/isti/epasmed/web/rest/vm/ManagedUserVM.java new file mode 100644 index 0000000..88aa11b --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/web/rest/vm/ManagedUserVM.java @@ -0,0 +1,35 @@ +package it.cnr.isti.epasmed.web.rest.vm; + +import it.cnr.isti.epasmed.service.dto.UserDTO; +import javax.validation.constraints.Size; + +/** + * View Model extending the UserDTO, which is meant to be used in the user management UI. + */ +public class ManagedUserVM extends UserDTO { + + public static final int PASSWORD_MIN_LENGTH = 4; + + public static final int PASSWORD_MAX_LENGTH = 100; + + @Size(min = PASSWORD_MIN_LENGTH, max = PASSWORD_MAX_LENGTH) + private String password; + + public ManagedUserVM() { + // Empty constructor needed for Jackson. + } + + public String getPassword() { + return password; + } + + public void setPassword(String password) { + this.password = password; + } + + // prettier-ignore + @Override + public String toString() { + return "ManagedUserVM{" + super.toString() + "} "; + } +} diff --git a/src/main/java/it/cnr/isti/epasmed/web/rest/vm/package-info.java b/src/main/java/it/cnr/isti/epasmed/web/rest/vm/package-info.java new file mode 100644 index 0000000..c4327a8 --- /dev/null +++ b/src/main/java/it/cnr/isti/epasmed/web/rest/vm/package-info.java @@ -0,0 +1,4 @@ +/** + * View Models used by Spring MVC REST controllers. + */ +package it.cnr.isti.epasmed.web.rest.vm; diff --git a/src/main/javasecurity/java.security b/src/main/javasecurity/java.security new file mode 100644 index 0000000..f399517 --- /dev/null +++ b/src/main/javasecurity/java.security @@ -0,0 +1,1223 @@ +# +# This is the "master security properties file". +# +# An alternate java.security properties file may be specified +# from the command line via the system property +# +# -Djava.security.properties= +# +# This properties file appends to the master security properties file. +# If both properties files specify values for the same key, the value +# from the command-line properties file is selected, as it is the last +# one loaded. +# +# Also, if you specify +# +# -Djava.security.properties== (2 equals), +# +# then that properties file completely overrides the master security +# properties file. +# +# To disable the ability to specify an additional properties file from +# the command line, set the key security.overridePropertiesFile +# to false in the master security properties file. It is set to true +# by default. + +# In this file, various security properties are set for use by +# java.security classes. This is where users can statically register +# Cryptography Package Providers ("providers" for short). The term +# "provider" refers to a package or set of packages that supply a +# concrete implementation of a subset of the cryptography aspects of +# the Java Security API. A provider may, for example, implement one or +# more digital signature algorithms or message digest algorithms. +# +# Each provider must implement a subclass of the Provider class. +# To register a provider in this master security properties file, +# specify the Provider subclass name and priority in the format +# +# security.provider.= +# +# This declares a provider, and specifies its preference +# order n. The preference order is the order in which providers are +# searched for requested algorithms (when no specific provider is +# requested). The order is 1-based; 1 is the most preferred, followed +# by 2, and so on. +# +# must specify the subclass of the Provider class whose +# constructor sets the values of various properties that are required +# for the Java Security API to look up the algorithms or other +# facilities implemented by the provider. +# +# There must be at least one provider specification in java.security. +# There is a default provider that comes standard with the JDK. It +# is called the "SUN" provider, and its Provider subclass +# named Sun appears in the sun.security.provider package. Thus, the +# "SUN" provider is registered via the following: +# +# security.provider.1=sun.security.provider.Sun +# +# (The number 1 is used for the default provider.) +# +# Note: Providers can be dynamically registered instead by calls to +# either the addProvider or insertProviderAt method in the Security +# class. + +# +# List of providers and their preference orders (see above): +# +security.provider.1=sun.security.provider.Sun +security.provider.2=sun.security.rsa.SunRsaSign +security.provider.3=sun.security.ec.SunEC +security.provider.4=com.sun.net.ssl.internal.ssl.Provider +security.provider.5=com.sun.crypto.provider.SunJCE +security.provider.6=sun.security.jgss.SunProvider +security.provider.7=com.sun.security.sasl.Provider +security.provider.8=org.jcp.xml.dsig.internal.dom.XMLDSigRI +security.provider.9=sun.security.smartcardio.SunPCSC + +# +# Sun Provider SecureRandom seed source. +# +# Select the primary source of seed data for the "SHA1PRNG" and +# "NativePRNG" SecureRandom implementations in the "Sun" provider. +# (Other SecureRandom implementations might also use this property.) +# +# On Unix-like systems (for example, Solaris/Linux/MacOS), the +# "NativePRNG" and "SHA1PRNG" implementations obtains seed data from +# special device files such as file:/dev/random. +# +# On Windows systems, specifying the URLs "file:/dev/random" or +# "file:/dev/urandom" will enable the native Microsoft CryptoAPI seeding +# mechanism for SHA1PRNG. +# +# By default, an attempt is made to use the entropy gathering device +# specified by the "securerandom.source" Security property. If an +# exception occurs while accessing the specified URL: +# +# SHA1PRNG: +# the traditional system/thread activity algorithm will be used. +# +# NativePRNG: +# a default value of /dev/random will be used. If neither +# are available, the implementation will be disabled. +# "file" is the only currently supported protocol type. +# +# The entropy gathering device can also be specified with the System +# property "java.security.egd". For example: +# +# % java -Djava.security.egd=file:/dev/random MainClass +# +# Specifying this System property will override the +# "securerandom.source" Security property. +# +# In addition, if "file:/dev/random" or "file:/dev/urandom" is +# specified, the "NativePRNG" implementation will be more preferred than +# SHA1PRNG in the Sun provider. +# +securerandom.source=file:/dev/random + +# +# A list of known strong SecureRandom implementations. +# +# To help guide applications in selecting a suitable strong +# java.security.SecureRandom implementation, Java distributions should +# indicate a list of known strong implementations using the property. +# +# This is a comma-separated list of algorithm and/or algorithm:provider +# entries. +# +securerandom.strongAlgorithms=NativePRNGBlocking:SUN + +# +# Class to instantiate as the javax.security.auth.login.Configuration +# provider. +# +login.configuration.provider=sun.security.provider.ConfigFile + +# +# Default login configuration file +# +#login.config.url.1=file:${user.home}/.java.login.config + +# +# Class to instantiate as the system Policy. This is the name of the class +# that will be used as the Policy object. +# +policy.provider=sun.security.provider.PolicyFile + +# The default is to have a single system-wide policy file, +# and a policy file in the user's home directory. +policy.url.1=file:${java.home}/lib/security/java.policy +policy.url.2=file:${user.home}/.java.policy + +# whether or not we expand properties in the policy file +# if this is set to false, properties (${...}) will not be expanded in policy +# files. +policy.expandProperties=true + +# whether or not we allow an extra policy to be passed on the command line +# with -Djava.security.policy=somefile. Comment out this line to disable +# this feature. +policy.allowSystemProperty=true + +# whether or not we look into the IdentityScope for trusted Identities +# when encountering a 1.1 signed JAR file. If the identity is found +# and is trusted, we grant it AllPermission. +policy.ignoreIdentityScope=false + +# +# Default keystore type. +# +keystore.type=jks + +# +# Controls compatibility mode for the JKS keystore type. +# +# When set to 'true', the JKS keystore type supports loading +# keystore files in either JKS or PKCS12 format. When set to 'false' +# it supports loading only JKS keystore files. +# +keystore.type.compat=true + +# +# List of comma-separated packages that start with or equal this string +# will cause a security exception to be thrown when +# passed to checkPackageAccess unless the +# corresponding RuntimePermission ("accessClassInPackage."+package) has +# been granted. +package.access=sun.,\ + org.GNOME.Accessibility.,\ + com.sun.xml.internal.,\ + com.sun.imageio.,\ + com.sun.istack.internal.,\ + com.sun.jmx.,\ + com.sun.media.sound.,\ + com.sun.naming.internal.,\ + com.sun.proxy.,\ + com.sun.corba.se.,\ + com.sun.org.apache.bcel.internal.,\ + com.sun.org.apache.regexp.internal.,\ + com.sun.org.apache.xerces.internal.,\ + com.sun.org.apache.xpath.internal.,\ + com.sun.org.apache.xalan.internal.extensions.,\ + com.sun.org.apache.xalan.internal.lib.,\ + com.sun.org.apache.xalan.internal.res.,\ + com.sun.org.apache.xalan.internal.templates.,\ + com.sun.org.apache.xalan.internal.utils.,\ + com.sun.org.apache.xalan.internal.xslt.,\ + com.sun.org.apache.xalan.internal.xsltc.cmdline.,\ + com.sun.org.apache.xalan.internal.xsltc.compiler.,\ + com.sun.org.apache.xalan.internal.xsltc.trax.,\ + com.sun.org.apache.xalan.internal.xsltc.util.,\ + com.sun.org.apache.xml.internal.res.,\ + com.sun.org.apache.xml.internal.resolver.helpers.,\ + com.sun.org.apache.xml.internal.resolver.readers.,\ + com.sun.org.apache.xml.internal.security.,\ + com.sun.org.apache.xml.internal.serializer.utils.,\ + com.sun.org.apache.xml.internal.utils.,\ + com.sun.org.glassfish.,\ + com.oracle.xmlns.internal.,\ + com.oracle.webservices.internal.,\ + oracle.jrockit.jfr.,\ + org.jcp.xml.dsig.internal.,\ + jdk.internal.,\ + jdk.nashorn.internal.,\ + jdk.nashorn.tools.,\ + jdk.xml.internal.,\ + com.sun.activation.registries.,\ + jdk.jfr.events.,\ + jdk.jfr.internal.,\ + jdk.management.jfr.internal. + +# +# List of comma-separated packages that start with or equal this string +# will cause a security exception to be thrown when +# passed to checkPackageDefinition unless the +# corresponding RuntimePermission ("defineClassInPackage."+package) has +# been granted. +# +# by default, none of the class loaders supplied with the JDK call +# checkPackageDefinition. +# +package.definition=sun.,\ + com.sun.xml.internal.,\ + com.sun.imageio.,\ + com.sun.istack.internal.,\ + com.sun.jmx.,\ + com.sun.media.sound.,\ + com.sun.naming.internal.,\ + com.sun.proxy.,\ + com.sun.corba.se.,\ + com.sun.org.apache.bcel.internal.,\ + com.sun.org.apache.regexp.internal.,\ + com.sun.org.apache.xerces.internal.,\ + com.sun.org.apache.xpath.internal.,\ + com.sun.org.apache.xalan.internal.extensions.,\ + com.sun.org.apache.xalan.internal.lib.,\ + com.sun.org.apache.xalan.internal.res.,\ + com.sun.org.apache.xalan.internal.templates.,\ + com.sun.org.apache.xalan.internal.utils.,\ + com.sun.org.apache.xalan.internal.xslt.,\ + com.sun.org.apache.xalan.internal.xsltc.cmdline.,\ + com.sun.org.apache.xalan.internal.xsltc.compiler.,\ + com.sun.org.apache.xalan.internal.xsltc.trax.,\ + com.sun.org.apache.xalan.internal.xsltc.util.,\ + com.sun.org.apache.xml.internal.res.,\ + com.sun.org.apache.xml.internal.resolver.helpers.,\ + com.sun.org.apache.xml.internal.resolver.readers.,\ + com.sun.org.apache.xml.internal.security.,\ + com.sun.org.apache.xml.internal.serializer.utils.,\ + com.sun.org.apache.xml.internal.utils.,\ + com.sun.org.glassfish.,\ + com.oracle.xmlns.internal.,\ + com.oracle.webservices.internal.,\ + oracle.jrockit.jfr.,\ + org.jcp.xml.dsig.internal.,\ + jdk.internal.,\ + jdk.nashorn.internal.,\ + jdk.nashorn.tools.,\ + jdk.xml.internal.,\ + com.sun.activation.registries.,\ + jdk.jfr.events.,\ + jdk.jfr.internal.,\ + jdk.management.jfr.internal. + +# +# Determines whether this properties file can be appended to +# or overridden on the command line via -Djava.security.properties +# +security.overridePropertiesFile=true + +# +# Determines the default key and trust manager factory algorithms for +# the javax.net.ssl package. +# +ssl.KeyManagerFactory.algorithm=SunX509 +ssl.TrustManagerFactory.algorithm=PKIX + +# +# The Java-level namelookup cache policy for successful lookups: +# +# any negative value: caching forever +# any positive value: the number of seconds to cache an address for +# zero: do not cache +# +# default value is forever (FOREVER). For security reasons, this +# caching is made forever when a security manager is set. When a security +# manager is not set, the default behavior in this implementation +# is to cache for 30 seconds. +# +# NOTE: setting this to anything other than the default value can have +# serious security implications. Do not set it unless +# you are sure you are not exposed to DNS spoofing attack. +# +#networkaddress.cache.ttl=-1 + +# The Java-level namelookup cache policy for failed lookups: +# +# any negative value: cache forever +# any positive value: the number of seconds to cache negative lookup results +# zero: do not cache +# +# In some Microsoft Windows networking environments that employ +# the WINS name service in addition to DNS, name service lookups +# that fail may take a noticeably long time to return (approx. 5 seconds). +# For this reason the default caching policy is to maintain these +# results for 10 seconds. +# +# +networkaddress.cache.negative.ttl=10 + +# +# Properties to configure OCSP for certificate revocation checking +# + +# Enable OCSP +# +# By default, OCSP is not used for certificate revocation checking. +# This property enables the use of OCSP when set to the value "true". +# +# NOTE: SocketPermission is required to connect to an OCSP responder. +# +# Example, +# ocsp.enable=true + +# +# Location of the OCSP responder +# +# By default, the location of the OCSP responder is determined implicitly +# from the certificate being validated. This property explicitly specifies +# the location of the OCSP responder. The property is used when the +# Authority Information Access extension (defined in RFC 5280) is absent +# from the certificate or when it requires overriding. +# +# Example, +# ocsp.responderURL=http://ocsp.example.net:80 + +# +# Subject name of the OCSP responder's certificate +# +# By default, the certificate of the OCSP responder is that of the issuer +# of the certificate being validated. This property identifies the certificate +# of the OCSP responder when the default does not apply. Its value is a string +# distinguished name (defined in RFC 2253) which identifies a certificate in +# the set of certificates supplied during cert path validation. In cases where +# the subject name alone is not sufficient to uniquely identify the certificate +# then both the "ocsp.responderCertIssuerName" and +# "ocsp.responderCertSerialNumber" properties must be used instead. When this +# property is set then those two properties are ignored. +# +# Example, +# ocsp.responderCertSubjectName="CN=OCSP Responder, O=XYZ Corp" + +# +# Issuer name of the OCSP responder's certificate +# +# By default, the certificate of the OCSP responder is that of the issuer +# of the certificate being validated. This property identifies the certificate +# of the OCSP responder when the default does not apply. Its value is a string +# distinguished name (defined in RFC 2253) which identifies a certificate in +# the set of certificates supplied during cert path validation. When this +# property is set then the "ocsp.responderCertSerialNumber" property must also +# be set. When the "ocsp.responderCertSubjectName" property is set then this +# property is ignored. +# +# Example, +# ocsp.responderCertIssuerName="CN=Enterprise CA, O=XYZ Corp" + +# +# Serial number of the OCSP responder's certificate +# +# By default, the certificate of the OCSP responder is that of the issuer +# of the certificate being validated. This property identifies the certificate +# of the OCSP responder when the default does not apply. Its value is a string +# of hexadecimal digits (colon or space separators may be present) which +# identifies a certificate in the set of certificates supplied during cert path +# validation. When this property is set then the "ocsp.responderCertIssuerName" +# property must also be set. When the "ocsp.responderCertSubjectName" property +# is set then this property is ignored. +# +# Example, +# ocsp.responderCertSerialNumber=2A:FF:00 + +# +# Policy for failed Kerberos KDC lookups: +# +# When a KDC is unavailable (network error, service failure, etc), it is +# put inside a blacklist and accessed less often for future requests. The +# value (case-insensitive) for this policy can be: +# +# tryLast +# KDCs in the blacklist are always tried after those not on the list. +# +# tryLess[:max_retries,timeout] +# KDCs in the blacklist are still tried by their order in the configuration, +# but with smaller max_retries and timeout values. max_retries and timeout +# are optional numerical parameters (default 1 and 5000, which means once +# and 5 seconds). Please notes that if any of the values defined here is +# more than what is defined in krb5.conf, it will be ignored. +# +# Whenever a KDC is detected as available, it is removed from the blacklist. +# The blacklist is reset when krb5.conf is reloaded. You can add +# refreshKrb5Config=true to a JAAS configuration file so that krb5.conf is +# reloaded whenever a JAAS authentication is attempted. +# +# Example, +# krb5.kdc.bad.policy = tryLast +# krb5.kdc.bad.policy = tryLess:2,2000 +krb5.kdc.bad.policy = tryLast + +# +# Kerberos cross-realm referrals (RFC 6806) +# +# OpenJDK's Kerberos client supports cross-realm referrals as defined in +# RFC 6806. This allows to setup more dynamic environments in which clients +# do not need to know in advance how to reach the realm of a target principal +# (either a user or service). +# +# When a client issues an AS or a TGS request, the "canonicalize" option +# is set to announce support of this feature. A KDC server may fulfill the +# request or reply referring the client to a different one. If referred, +# the client will issue a new request and the cycle repeats. +# +# In addition to referrals, the "canonicalize" option allows the KDC server +# to change the client name in response to an AS request. For security reasons, +# RFC 6806 (section 11) FAST scheme is enforced. +# +# Disable Kerberos cross-realm referrals. Value may be overwritten with a +# System property (-Dsun.security.krb5.disableReferrals). +sun.security.krb5.disableReferrals=false + +# Maximum number of AS or TGS referrals to avoid infinite loops. Value may +# be overwritten with a System property (-Dsun.security.krb5.maxReferrals). +sun.security.krb5.maxReferrals=5 + +# +# This property contains a list of disabled EC Named Curves that can be included +# in the jdk.[tls|certpath|jar].disabledAlgorithms properties. To include this +# list in any of the disabledAlgorithms properties, add the property name as +# an entry. +jdk.disabled.namedCurves = secp112r1, secp112r2, secp128r1, secp128r2, \ + secp160k1, secp160r1, secp160r2, secp192k1, secp192r1, secp224k1, \ + secp224r1, secp256k1, sect113r1, sect113r2, sect131r1, sect131r2, \ + sect163k1, sect163r1, sect163r2, sect193r1, sect193r2, sect233k1, \ + sect233r1, sect239k1, sect283k1, sect283r1, sect409k1, sect409r1, \ + sect571k1, sect571r1, X9.62 c2tnb191v1, X9.62 c2tnb191v2, \ + X9.62 c2tnb191v3, X9.62 c2tnb239v1, X9.62 c2tnb239v2, X9.62 c2tnb239v3, \ + X9.62 c2tnb359v1, X9.62 c2tnb431r1, X9.62 prime192v2, X9.62 prime192v3, \ + X9.62 prime239v1, X9.62 prime239v2, X9.62 prime239v3, brainpoolP256r1, \ + brainpoolP320r1, brainpoolP384r1, brainpoolP512r1 + +# +# Algorithm restrictions for certification path (CertPath) processing +# +# In some environments, certain algorithms or key lengths may be undesirable +# for certification path building and validation. For example, "MD2" is +# generally no longer considered to be a secure hash algorithm. This section +# describes the mechanism for disabling algorithms based on algorithm name +# and/or key length. This includes algorithms used in certificates, as well +# as revocation information such as CRLs and signed OCSP Responses. +# The syntax of the disabled algorithm string is described as follows: +# DisabledAlgorithms: +# " DisabledAlgorithm { , DisabledAlgorithm } " +# +# DisabledAlgorithm: +# AlgorithmName [Constraint] { '&' Constraint } | IncludeProperty +# +# AlgorithmName: +# (see below) +# +# Constraint: +# KeySizeConstraint | CAConstraint | DenyAfterConstraint | +# UsageConstraint +# +# KeySizeConstraint: +# keySize Operator KeyLength +# +# Operator: +# <= | < | == | != | >= | > +# +# KeyLength: +# Integer value of the algorithm's key length in bits +# +# CAConstraint: +# jdkCA +# +# DenyAfterConstraint: +# denyAfter YYYY-MM-DD +# +# UsageConstraint: +# usage [TLSServer] [TLSClient] [SignedJAR] +# +# IncludeProperty: +# include +# +# The "AlgorithmName" is the standard algorithm name of the disabled +# algorithm. See "Java Cryptography Architecture Standard Algorithm Name +# Documentation" for information about Standard Algorithm Names. Matching +# is performed using a case-insensitive sub-element matching rule. (For +# example, in "SHA1withECDSA" the sub-elements are "SHA1" for hashing and +# "ECDSA" for signatures.) If the assertion "AlgorithmName" is a +# sub-element of the certificate algorithm name, the algorithm will be +# rejected during certification path building and validation. For example, +# the assertion algorithm name "DSA" will disable all certificate algorithms +# that rely on DSA, such as NONEwithDSA, SHA1withDSA. However, the assertion +# will not disable algorithms related to "ECDSA". +# +# The "IncludeProperty" allows a implementation-defined security property that +# can be included in the disabledAlgorithms properties. These properties are +# to help manage common actions easier across multiple disabledAlgorithm +# properties. +# There is one defined security property: jdk.disabled.NamedCurves +# See the property for more specific details. +# +# +# A "Constraint" defines restrictions on the keys and/or certificates for +# a specified AlgorithmName: +# +# KeySizeConstraint: +# keySize Operator KeyLength +# The constraint requires a key of a valid size range if the +# "AlgorithmName" is of a key algorithm. The "KeyLength" indicates +# the key size specified in number of bits. For example, +# "RSA keySize <= 1024" indicates that any RSA key with key size less +# than or equal to 1024 bits should be disabled, and +# "RSA keySize < 1024, RSA keySize > 2048" indicates that any RSA key +# with key size less than 1024 or greater than 2048 should be disabled. +# This constraint is only used on algorithms that have a key size. +# +# CAConstraint: +# jdkCA +# This constraint prohibits the specified algorithm only if the +# algorithm is used in a certificate chain that terminates at a marked +# trust anchor in the lib/security/cacerts keystore. If the jdkCA +# constraint is not set, then all chains using the specified algorithm +# are restricted. jdkCA may only be used once in a DisabledAlgorithm +# expression. +# Example: To apply this constraint to SHA-1 certificates, include +# the following: "SHA1 jdkCA" +# +# DenyAfterConstraint: +# denyAfter YYYY-MM-DD +# This constraint prohibits a certificate with the specified algorithm +# from being used after the date regardless of the certificate's +# validity. JAR files that are signed and timestamped before the +# constraint date with certificates containing the disabled algorithm +# will not be restricted. The date is processed in the UTC timezone. +# This constraint can only be used once in a DisabledAlgorithm +# expression. +# Example: To deny usage of RSA 2048 bit certificates after Feb 3 2020, +# use the following: "RSA keySize == 2048 & denyAfter 2020-02-03" +# +# UsageConstraint: +# usage [TLSServer] [TLSClient] [SignedJAR] +# This constraint prohibits the specified algorithm for +# a specified usage. This should be used when disabling an algorithm +# for all usages is not practical. 'TLSServer' restricts the algorithm +# in TLS server certificate chains when server authentication is +# performed. 'TLSClient' restricts the algorithm in TLS client +# certificate chains when client authentication is performed. +# 'SignedJAR' constrains use of certificates in signed jar files. +# The usage type follows the keyword and more than one usage type can +# be specified with a whitespace delimiter. +# Example: "SHA1 usage TLSServer TLSClient" +# +# When an algorithm must satisfy more than one constraint, it must be +# delimited by an ampersand '&'. For example, to restrict certificates in a +# chain that terminate at a distribution provided trust anchor and contain +# RSA keys that are less than or equal to 1024 bits, add the following +# constraint: "RSA keySize <= 1024 & jdkCA". +# +# All DisabledAlgorithms expressions are processed in the order defined in the +# property. This requires lower keysize constraints to be specified +# before larger keysize constraints of the same algorithm. For example: +# "RSA keySize < 1024 & jdkCA, RSA keySize < 2048". +# +# Note: The algorithm restrictions do not apply to trust anchors or +# self-signed certificates. +# +# Note: This property is currently used by Oracle's PKIX implementation. It +# is not guaranteed to be examined and used by other implementations. +# +# Example: +# jdk.certpath.disabledAlgorithms=MD2, DSA, RSA keySize < 2048 +# +# +jdk.certpath.disabledAlgorithms=MD2, MD5, SHA1 jdkCA & usage TLSServer, \ + RSA keySize < 1024, DSA keySize < 1024, EC keySize < 224, \ + include jdk.disabled.namedCurves + +# +# Legacy algorithms for certification path (CertPath) processing and +# signed JAR files. +# +# In some environments, a certain algorithm or key length may be undesirable +# but is not yet disabled. +# +# Tools such as keytool and jarsigner may emit warnings when these legacy +# algorithms are used. See the man pages for those tools for more information. +# +# The syntax is the same as the "jdk.certpath.disabledAlgorithms" and +# "jdk.jar.disabledAlgorithms" security properties. +# +# Note: This property is currently used by the JDK Reference +# implementation. It is not guaranteed to be examined and used by other +# implementations. + +jdk.security.legacyAlgorithms=SHA1, \ + RSA keySize < 2048, DSA keySize < 2048 + +# +# Algorithm restrictions for signed JAR files +# +# In some environments, certain algorithms or key lengths may be undesirable +# for signed JAR validation. For example, "MD2" is generally no longer +# considered to be a secure hash algorithm. This section describes the +# mechanism for disabling algorithms based on algorithm name and/or key length. +# JARs signed with any of the disabled algorithms or key sizes will be treated +# as unsigned. +# +# The syntax of the disabled algorithm string is described as follows: +# DisabledAlgorithms: +# " DisabledAlgorithm { , DisabledAlgorithm } " +# +# DisabledAlgorithm: +# AlgorithmName [Constraint] { '&' Constraint } +# +# AlgorithmName: +# (see below) +# +# Constraint: +# KeySizeConstraint | DenyAfterConstraint +# +# KeySizeConstraint: +# keySize Operator KeyLength +# +# DenyAfterConstraint: +# denyAfter YYYY-MM-DD +# +# Operator: +# <= | < | == | != | >= | > +# +# KeyLength: +# Integer value of the algorithm's key length in bits +# +# Note: This property is currently used by the JDK Reference +# implementation. It is not guaranteed to be examined and used by other +# implementations. +# +# See "jdk.certpath.disabledAlgorithms" for syntax descriptions. +# +jdk.jar.disabledAlgorithms=MD2, MD5, RSA keySize < 1024, \ + DSA keySize < 1024, include jdk.disabled.namedCurves + +# +# Algorithm restrictions for Secure Socket Layer/Transport Layer Security +# (SSL/TLS) processing +# +# In some environments, certain algorithms or key lengths may be undesirable +# when using SSL/TLS. This section describes the mechanism for disabling +# algorithms during SSL/TLS security parameters negotiation, including +# protocol version negotiation, cipher suites selection, peer authentication +# and key exchange mechanisms. +# +# Disabled algorithms will not be negotiated for SSL/TLS connections, even +# if they are enabled explicitly in an application. +# +# For PKI-based peer authentication and key exchange mechanisms, this list +# of disabled algorithms will also be checked during certification path +# building and validation, including algorithms used in certificates, as +# well as revocation information such as CRLs and signed OCSP Responses. +# This is in addition to the jdk.certpath.disabledAlgorithms property above. +# +# See the specification of "jdk.certpath.disabledAlgorithms" for the +# syntax of the disabled algorithm string. +# +# Note: The algorithm restrictions do not apply to trust anchors or +# self-signed certificates. +# +# Note: This property is currently used by the JDK Reference implementation. +# It is not guaranteed to be examined and used by other implementations. +# +# Example: +# jdk.tls.disabledAlgorithms=MD5, SSLv3, DSA, RSA keySize < 2048 +jdk.tls.disabledAlgorithms=SSLv3, RC4, DES, MD5withRSA, \ + DH keySize < 1024, EC keySize < 224, 3DES_EDE_CBC, anon, NULL, \ + include jdk.disabled.namedCurves + +# Legacy algorithms for Secure Socket Layer/Transport Layer Security (SSL/TLS) +# processing in JSSE implementation. +# +# In some environments, a certain algorithm may be undesirable but it +# cannot be disabled because of its use in legacy applications. Legacy +# algorithms may still be supported, but applications should not use them +# as the security strength of legacy algorithms are usually not strong enough +# in practice. +# +# During SSL/TLS security parameters negotiation, legacy algorithms will +# not be negotiated unless there are no other candidates. +# +# The syntax of the legacy algorithms string is described as this Java +# BNF-style: +# LegacyAlgorithms: +# " LegacyAlgorithm { , LegacyAlgorithm } " +# +# LegacyAlgorithm: +# AlgorithmName (standard JSSE algorithm name) +# +# See the specification of security property "jdk.certpath.disabledAlgorithms" +# for the syntax and description of the "AlgorithmName" notation. +# +# Per SSL/TLS specifications, cipher suites have the form: +# SSL_KeyExchangeAlg_WITH_CipherAlg_MacAlg +# or +# TLS_KeyExchangeAlg_WITH_CipherAlg_MacAlg +# +# For example, the cipher suite TLS_RSA_WITH_AES_128_CBC_SHA uses RSA as the +# key exchange algorithm, AES_128_CBC (128 bits AES cipher algorithm in CBC +# mode) as the cipher (encryption) algorithm, and SHA-1 as the message digest +# algorithm for HMAC. +# +# The LegacyAlgorithm can be one of the following standard algorithm names: +# 1. JSSE cipher suite name, e.g., TLS_RSA_WITH_AES_128_CBC_SHA +# 2. JSSE key exchange algorithm name, e.g., RSA +# 3. JSSE cipher (encryption) algorithm name, e.g., AES_128_CBC +# 4. JSSE message digest algorithm name, e.g., SHA +# +# See SSL/TLS specifications and "Java Cryptography Architecture Standard +# Algorithm Name Documentation" for information about the algorithm names. +# +# Note: This property is currently used by the JDK Reference implementation. +# It is not guaranteed to be examined and used by other implementations. +# There is no guarantee the property will continue to exist or be of the +# same syntax in future releases. +# +# Example: +# jdk.tls.legacyAlgorithms=DH_anon, DES_CBC, SSL_RSA_WITH_RC4_128_MD5 +# +jdk.tls.legacyAlgorithms= TLSv1, TLSv1.1, \ + K_NULL, C_NULL, M_NULL, \ + DH_anon, ECDH_anon, \ + RC4_128, RC4_40, DES_CBC, DES40_CBC, \ + 3DES_EDE_CBC + +# The pre-defined default finite field Diffie-Hellman ephemeral (DHE) +# parameters for Transport Layer Security (SSL/TLS/DTLS) processing. +# +# In traditional SSL/TLS/DTLS connections where finite field DHE parameters +# negotiation mechanism is not used, the server offers the client group +# parameters, base generator g and prime modulus p, for DHE key exchange. +# It is recommended to use dynamic group parameters. This property defines +# a mechanism that allows you to specify custom group parameters. +# +# The syntax of this property string is described as this Java BNF-style: +# DefaultDHEParameters: +# DefinedDHEParameters { , DefinedDHEParameters } +# +# DefinedDHEParameters: +# "{" DHEPrimeModulus , DHEBaseGenerator "}" +# +# DHEPrimeModulus: +# HexadecimalDigits +# +# DHEBaseGenerator: +# HexadecimalDigits +# +# HexadecimalDigits: +# HexadecimalDigit { HexadecimalDigit } +# +# HexadecimalDigit: one of +# 0 1 2 3 4 5 6 7 8 9 A B C D E F a b c d e f +# +# Whitespace characters are ignored. +# +# The "DefinedDHEParameters" defines the custom group parameters, prime +# modulus p and base generator g, for a particular size of prime modulus p. +# The "DHEPrimeModulus" defines the hexadecimal prime modulus p, and the +# "DHEBaseGenerator" defines the hexadecimal base generator g of a group +# parameter. It is recommended to use safe primes for the custom group +# parameters. +# +# If this property is not defined or the value is empty, the underlying JSSE +# provider's default group parameter is used for each connection. +# +# If the property value does not follow the grammar, or a particular group +# parameter is not valid, the connection will fall back and use the +# underlying JSSE provider's default group parameter. +# +# Note: This property is currently used by OpenJDK's JSSE implementation. It +# is not guaranteed to be examined and used by other implementations. +# +# Example: +# jdk.tls.server.defaultDHEParameters= +# { \ +# FFFFFFFF FFFFFFFF C90FDAA2 2168C234 C4C6628B 80DC1CD1 \ +# 29024E08 8A67CC74 020BBEA6 3B139B22 514A0879 8E3404DD \ +# EF9519B3 CD3A431B 302B0A6D F25F1437 4FE1356D 6D51C245 \ +# E485B576 625E7EC6 F44C42E9 A637ED6B 0BFF5CB6 F406B7ED \ +# EE386BFB 5A899FA5 AE9F2411 7C4B1FE6 49286651 ECE65381 \ +# FFFFFFFF FFFFFFFF, 2} + +# +# TLS key limits on symmetric cryptographic algorithms +# +# This security property sets limits on algorithms key usage in TLS 1.3. +# When the amount of data encrypted exceeds the algorithm value listed below, +# a KeyUpdate message will trigger a key change. This is for symmetric ciphers +# with TLS 1.3 only. +# +# The syntax for the property is described below: +# KeyLimits: +# " KeyLimit { , KeyLimit } " +# +# WeakKeyLimit: +# AlgorithmName Action Length +# +# AlgorithmName: +# A full algorithm transformation. +# +# Action: +# KeyUpdate +# +# Length: +# The amount of encrypted data in a session before the Action occurs +# This value may be an integer value in bytes, or as a power of two, 2^29. +# +# KeyUpdate: +# The TLS 1.3 KeyUpdate handshake process begins when the Length amount +# is fulfilled. +# +# Note: This property is currently used by OpenJDK's JSSE implementation. It +# is not guaranteed to be examined and used by other implementations. +# +jdk.tls.keyLimits=AES/GCM/NoPadding KeyUpdate 2^37 + +# Cryptographic Jurisdiction Policy defaults +# +# Import and export control rules on cryptographic software vary from +# country to country. By default, the JDK provides two different sets of +# cryptographic policy files: +# +# unlimited: These policy files contain no restrictions on cryptographic +# strengths or algorithms. +# +# limited: These policy files contain more restricted cryptographic +# strengths, and are still available if your country or +# usage requires the traditional restrictive policy. +# +# The JDK JCE framework uses the unlimited policy files by default. +# However the user may explicitly choose a set either by defining the +# "crypto.policy" Security property or by installing valid JCE policy +# jar files into the traditional JDK installation location. To better +# support older JDK Update releases, the "crypto.policy" property is not +# defined by default. See below for more information. +# +# The following logic determines which policy files are used: +# +# refers to the directory where the JRE was +# installed and may be determined using the "java.home" +# System property. +# +# 1. If the Security property "crypto.policy" has been defined, +# then the following mechanism is used: +# +# The policy files are stored as jar files in subdirectories of +# /lib/security/policy. Each directory contains a complete +# set of policy files. +# +# The "crypto.policy" Security property controls the directory +# selection, and thus the effective cryptographic policy. +# +# The default set of directories is: +# +# limited | unlimited +# +# 2. If the "crypto.policy" property is not set and the traditional +# US_export_policy.jar and local_policy.jar files +# (e.g. limited/unlimited) are found in the legacy +# /lib/security directory, then the rules embedded within +# those jar files will be used. This helps preserve compatibility +# for users upgrading from an older installation. +# +# 3. If the jar files are not present in the legacy location +# and the "crypto.policy" Security property is not defined, +# then the JDK will use the unlimited settings (equivalent to +# crypto.policy=unlimited) +# +# Please see the JCA documentation for additional information on these +# files and formats. +# +# YOU ARE ADVISED TO CONSULT YOUR EXPORT/IMPORT CONTROL COUNSEL OR ATTORNEY +# TO DETERMINE THE EXACT REQUIREMENTS. +# +# Please note that the JCE for Java SE, including the JCE framework, +# cryptographic policy files, and standard JCE providers provided with +# the Java SE, have been reviewed and approved for export as mass market +# encryption item by the US Bureau of Industry and Security. +# +# Note: This property is currently used by the JDK Reference implementation. +# It is not guaranteed to be examined and used by other implementations. +# +crypto.policy=unlimited + +# +# The policy for the XML Signature secure validation mode. The mode is +# enabled by setting the property "org.jcp.xml.dsig.secureValidation" to +# true with the javax.xml.crypto.XMLCryptoContext.setProperty() method, +# or by running the code with a SecurityManager. +# +# Policy: +# Constraint {"," Constraint } +# Constraint: +# AlgConstraint | MaxTransformsConstraint | MaxReferencesConstraint | +# ReferenceUriSchemeConstraint | KeySizeConstraint | OtherConstraint +# AlgConstraint +# "disallowAlg" Uri +# MaxTransformsConstraint: +# "maxTransforms" Integer +# MaxReferencesConstraint: +# "maxReferences" Integer +# ReferenceUriSchemeConstraint: +# "disallowReferenceUriSchemes" String { String } +# KeySizeConstraint: +# "minKeySize" KeyAlg Integer +# OtherConstraint: +# "noDuplicateIds" | "noRetrievalMethodLoops" +# +# For AlgConstraint, Uri is the algorithm URI String that is not allowed. +# See the XML Signature Recommendation for more information on algorithm +# URI Identifiers. For KeySizeConstraint, KeyAlg is the standard algorithm +# name of the key type (ex: "RSA"). If the MaxTransformsConstraint, +# MaxReferencesConstraint or KeySizeConstraint (for the same key type) is +# specified more than once, only the last entry is enforced. +# +# Note: This property is currently used by the JDK Reference implementation. It +# is not guaranteed to be examined and used by other implementations. +# +jdk.xml.dsig.secureValidationPolicy=\ + disallowAlg http://www.w3.org/TR/1999/REC-xslt-19991116,\ + disallowAlg http://www.w3.org/2001/04/xmldsig-more#rsa-md5,\ + disallowAlg http://www.w3.org/2001/04/xmldsig-more#hmac-md5,\ + disallowAlg http://www.w3.org/2001/04/xmldsig-more#md5,\ + maxTransforms 5,\ + maxReferences 30,\ + disallowReferenceUriSchemes file http https,\ + minKeySize RSA 1024,\ + minKeySize DSA 1024,\ + minKeySize EC 224,\ + noDuplicateIds,\ + noRetrievalMethodLoops + +# +# Serialization process-wide filter +# +# A filter, if configured, is used by java.io.ObjectInputStream during +# deserialization to check the contents of the stream. +# A filter is configured as a sequence of patterns, each pattern is either +# matched against the name of a class in the stream or defines a limit. +# Patterns are separated by ";" (semicolon). +# Whitespace is significant and is considered part of the pattern. +# +# If the system property jdk.serialFilter is also specified on the command +# line, it supersedes the security property value defined here. +# +# If a pattern includes a "=", it sets a limit. +# If a limit appears more than once the last value is used. +# Limits are checked before classes regardless of the order in the sequence of patterns. +# If any of the limits are exceeded, the filter status is REJECTED. +# +# maxdepth=value - the maximum depth of a graph +# maxrefs=value - the maximum number of internal references +# maxbytes=value - the maximum number of bytes in the input stream +# maxarray=value - the maximum array length allowed +# +# Other patterns, from left to right, match the class or package name as +# returned from Class.getName. +# If the class is an array type, the class or package to be matched is the element type. +# Arrays of any number of dimensions are treated the same as the element type. +# For example, a pattern of "!example.Foo", rejects creation of any instance or +# array of example.Foo. +# +# If the pattern starts with "!", the status is REJECTED if the remaining pattern +# is matched; otherwise the status is ALLOWED if the pattern matches. +# If the pattern ends with ".**" it matches any class in the package and all subpackages. +# If the pattern ends with ".*" it matches any class in the package. +# If the pattern ends with "*", it matches any class with the pattern as a prefix. +# If the pattern is equal to the class name, it matches. +# Otherwise, the status is UNDECIDED. +# +# Primitive types are not configurable with this filter. +# +#jdk.serialFilter=pattern;pattern + +# +# RMI Registry Serial Filter +# +# The filter pattern uses the same format as jdk.serialFilter. +# This filter can override the builtin filter if additional types need to be +# allowed or rejected from the RMI Registry or to decrease limits but not +# to increase limits. +# If the limits (maxdepth, maxrefs, or maxbytes) are exceeded, the object is rejected. +# +# The maxdepth of any array passed to the RMI Registry is set to +# 10000. The maximum depth of the graph is set to 20. +# These limits can be reduced via the maxarray, maxdepth limits. +# +#sun.rmi.registry.registryFilter=pattern;pattern + +# +# Array construction of any component type, including subarrays and arrays of +# primitives, are allowed unless the length is greater than the maxarray limit. +# The filter is applied to each array element. +# +# The built-in filter allows subclasses of allowed classes and +# can approximately be represented as the pattern: +# +#sun.rmi.registry.registryFilter=\ +# maxarray=1000000;\ +# maxdepth=20;\ +# java.lang.String;\ +# java.lang.Number;\ +# java.lang.reflect.Proxy;\ +# java.rmi.Remote;\ +# sun.rmi.server.UnicastRef;\ +# sun.rmi.server.RMIClientSocketFactory;\ +# sun.rmi.server.RMIServerSocketFactory;\ +# java.rmi.activation.ActivationID;\ +# java.rmi.server.UID +# +# RMI Distributed Garbage Collector (DGC) Serial Filter +# +# The filter pattern uses the same format as jdk.serialFilter. +# This filter can override the builtin filter if additional types need to be +# allowed or rejected from the RMI DGC. +# +# The builtin DGC filter can approximately be represented as the filter pattern: +# +#sun.rmi.transport.dgcFilter=\ +# java.rmi.server.ObjID;\ +# java.rmi.server.UID;\ +# java.rmi.dgc.VMID;\ +# java.rmi.dgc.Lease;\ +# maxdepth=5;maxarray=10000 + +# CORBA ORBIorTypeCheckRegistryFilter +# Type check enhancement for ORB::string_to_object processing +# +# An IOR type check filter, if configured, is used by an ORB during +# an ORB::string_to_object invocation to check the veracity of the type encoded +# in the ior string. +# +# The filter pattern consists of a semi-colon separated list of class names. +# The configured list contains the binary class names of the IDL interface types +# corresponding to the IDL stub class to be instantiated. +# As such, a filter specifies a list of IDL stub classes that will be +# allowed by an ORB when an ORB::string_to_object is invoked. +# It is used to specify a white list configuration of acceptable +# IDL stub types which may be contained in a stringified IOR +# parameter passed as input to an ORB::string_to_object method. +# +# Note: This property is currently used by the JDK Reference implementation. +# It is not guaranteed to be examined and used by other implementations. +# +#com.sun.CORBA.ORBIorTypeCheckRegistryFilter=binary_class_name;binary_class_name + +# +# JCEKS Encrypted Key Serial Filter +# +# This filter, if configured, is used by the JCEKS KeyStore during the +# deserialization of the encrypted Key object stored inside a key entry. +# If not configured or the filter result is UNDECIDED (i.e. none of the patterns +# matches), the filter configured by jdk.serialFilter will be consulted. +# +# If the system property jceks.key.serialFilter is also specified, it supersedes +# the security property value defined here. +# +# The filter pattern uses the same format as jdk.serialFilter. The default +# pattern allows java.lang.Enum, java.security.KeyRep, java.security.KeyRep$Type, +# and javax.crypto.spec.SecretKeySpec and rejects all the others. +jceks.key.serialFilter = java.lang.Enum;java.security.KeyRep;\ + java.security.KeyRep$Type;javax.crypto.spec.SecretKeySpec;!* + +# The iteration count used for password-based encryption (PBE) in JCEKS +# keystores. Values in the range 10000 to 5000000 are considered valid. +# If the value is out of this range, or is not a number, or is unspecified; +# a default of 200000 is used. +# +# If the system property jdk.jceks.iterationCount is also specified, it +# supersedes the security property value defined here. +# +#jdk.jceks.iterationCount = 200000 + +# +# Disabled mechanisms for the Simple Authentication and Security Layer (SASL) +# +# Disabled mechanisms will not be negotiated by both SASL clients and servers. +# These mechanisms will be ignored if they are specified in the "mechanisms" +# argument of "Sasl.createSaslClient" or the "mechanism" argument of +# "Sasl.createSaslServer". +# +# The value of this property is a comma-separated list of SASL mechanisms. +# The mechanisms are case-sensitive. Whitespaces around the commas are ignored. +# +# Note: This property is currently used by the JDK Reference implementation. +# It is not guaranteed to be examined and used by other implementations. +# +# Example: +# jdk.sasl.disabledMechanisms=PLAIN, CRAM-MD5, DIGEST-MD5 +jdk.sasl.disabledMechanisms= + +# +# Policies for distrusting Certificate Authorities (CAs). +# +# This is a comma separated value of one or more case-sensitive strings, each +# of which represents a policy for determining if a CA should be distrusted. +# The supported values are: +# +# +# SYMANTEC_TLS : Distrust TLS Server certificates anchored by a Symantec +# root CA and issued after April 16, 2019 unless issued by one of the +# following subordinate CAs which have a later distrust date: +# 1. Apple IST CA 2 - G1, SHA-256 fingerprint: +# AC2B922ECFD5E01711772FEA8ED372DE9D1E2245FCE3F57A9CDBEC77296A424B +# Distrust after December 31, 2019. +# 2. Apple IST CA 8 - G1, SHA-256 fingerprint: +# A4FE7C7F15155F3F0AEF7AAA83CF6E06DEB97CA3F909DF920AC1490882D488ED +# Distrust after December 31, 2019. +# Leading and trailing whitespace surrounding each value are ignored. +# Unknown values are ignored. If the property is commented out or set to the +# empty String, no policies are enforced. +# +# Note: This property is currently used by the JDK Reference implementation. +# It is not guaranteed to be supported by other SE implementations. Also, this +# property does not override other security properties which can restrict +# certificates such as jdk.tls.disabledAlgorithms or +# jdk.certpath.disabledAlgorithms; those restrictions are still enforced even +# if this property is not enabled. +# +jdk.security.caDistrustPolicies=SYMANTEC_TLS + +# +# Policies for the proxy_impersonator Kerberos ccache configuration entry +# +# The proxy_impersonator ccache configuration entry indicates that the ccache +# is a synthetic delegated credential for use with S4U2Proxy by an intermediate +# server. The ccache file should also contain the TGT of this server and +# an evidence ticket from the default principal of the ccache to this server. +# +# This security property determines how Java uses this configuration entry. +# There are 3 possible values: +# +# no-impersonate - Ignore this configuration entry, and always act as +# the owner of the TGT (if it exists). +# +# try-impersonate - Try impersonation when this configuration entry exists. +# If no matching TGT or evidence ticket is found, +# fallback to no-impersonate. +# +# always-impersonate - Always impersonate when this configuration entry exists. +# If no matching TGT or evidence ticket is found, +# no initial credential is read from the ccache. +# +# The default value is "always-impersonate". +# +# If a system property of the same name is also specified, it supersedes the +# security property value defined here. +# +#jdk.security.krb5.default.initiate.credential=always-impersonate + +# +# Trust Anchor Certificates - CA Basic Constraint check +# +# X.509 v3 certificates used as Trust Anchors (to validate signed code or TLS +# connections) must have the cA Basic Constraint field set to 'true'. Also, if +# they include a Key Usage extension, the keyCertSign bit must be set. These +# checks, enabled by default, can be disabled for backward-compatibility +# purposes with the jdk.security.allowNonCaAnchor System and Security +# properties. In the case that both properties are simultaneously set, the +# System value prevails. The default value of the property is "false". +# +#jdk.security.allowNonCaAnchor=true + +# +# JNDI Object Factories Filter +# +# This filter is used by the JNDI runtime to control the set of object factory classes +# which will be allowed to instantiate objects from object references returned by +# naming/directory systems. The factory class named by the reference instance will be +# matched against this filter. The filter property supports pattern-based filter syntax +# with the same format as jdk.serialFilter. +# +# Each pattern is matched against the factory class name to allow or disallow it's +# instantiation. The access to a factory class is allowed unless the filter returns +# REJECTED. +# +# Note: This property is currently used by the JDK Reference implementation. +# It is not guaranteed to be examined and used by other implementations. +# +# If the system property jdk.jndi.object.factoriesFilter is also specified, it supersedes +# the security property value defined here. The default value of the property is "*". +# +# The default pattern value allows any object factory class specified by the reference +# instance to recreate the referenced object. +#jdk.jndi.object.factoriesFilter=* diff --git a/src/main/jib/entrypoint.sh b/src/main/jib/entrypoint.sh new file mode 100644 index 0000000..fd76f91 --- /dev/null +++ b/src/main/jib/entrypoint.sh @@ -0,0 +1,10 @@ +#!/bin/sh + +echo "The application will start in ${JHIPSTER_SLEEP}s..." && sleep ${JHIPSTER_SLEEP} + +# TLS Security Config +#-Djdk.tls.disabledAlgorithms=SSLv3,RC4,DES,MD5withRSA,anon,NULL +#-Djdk.tls.legacyAlgorithms=TLSv1,TLSv1.1,K_NULL,C_NULL,M_NULL,DH_anon,ECDH_anon,RC4_128,RC4_40,DES_CBC,DES40_CBC,3DES_EDE_CBC +#-Ddeployment.security.SSLv3=true -Ddeployment.security.TLSv1=true +#-Djavax.net.debug=ssl:handshake:verbose +exec java ${JAVA_OPTS} -noverify -XX:+AlwaysPreTouch -Djava.security.egd=file:/dev/./urandom -cp /app/resources/:/app/classes/:/app/libs/* "it.cnr.isti.epasmed.EpasmedApp" "$@" diff --git a/src/main/resources/.h2.server.properties b/src/main/resources/.h2.server.properties new file mode 100644 index 0000000..029fc8b --- /dev/null +++ b/src/main/resources/.h2.server.properties @@ -0,0 +1,6 @@ +#H2 Server Properties +#Wed May 12 18:37:07 CEST 2021 +0=JHipster H2 (Disk)|org.h2.Driver|jdbc\:h2\:file\:./target/h2db/db/epasmed| +webAllowOthers=true +webPort=8082 +webSSL=false diff --git a/src/main/resources/banner.txt b/src/main/resources/banner.txt new file mode 100644 index 0000000..640e29f --- /dev/null +++ b/src/main/resources/banner.txt @@ -0,0 +1,11 @@ + + ${AnsiColor.GREEN}███████╗██████╗ █████╗ ███████╗${AnsiColor.RED} ███╗ ███╗███████╗██████╗ + ${AnsiColor.GREEN}██╔════╝██╔══██╗██╔══██╗██╔════╝${AnsiColor.RED} ████╗ ████║██╔════╝██╔══██╗ + ${AnsiColor.GREEN}█████╗ ██████╔╝███████║███████╗${AnsiColor.RED} ██╔████╔██║█████╗ ██║ ██║ + ${AnsiColor.GREEN}██╔══╝ ██╔═══╝ ██╔══██║╚════██║${AnsiColor.RED} ██║╚██╔╝██║██╔══╝ ██║ ██║ + ${AnsiColor.GREEN}███████╗██║ ██║ ██║███████║${AnsiColor.RED} ██║ ╚═╝ ██║███████╗██████╔╝ + ${AnsiColor.GREEN}╚══════╝╚═╝ ╚═╝ ╚═╝╚══════╝${AnsiColor.RED} ╚═╝ ╚═╝╚══════╝╚═════╝ + + +${AnsiColor.BRIGHT_BLUE}:: Running Spring Boot ${spring-boot.version} :: +${AnsiColor.DEFAULT} diff --git a/src/main/resources/config/application-dev.yml b/src/main/resources/config/application-dev.yml new file mode 100644 index 0000000..fe8c83d --- /dev/null +++ b/src/main/resources/config/application-dev.yml @@ -0,0 +1,132 @@ +# =================================================================== +# Spring Boot configuration for the "dev" profile. +# +# This configuration overrides the application.yml file. +# +# More information on profiles: https://www.jhipster.tech/profiles/ +# More information on configuration properties: https://www.jhipster.tech/common-application-properties/ +# =================================================================== + +# =================================================================== +# Standard Spring Boot properties. +# Full reference is available at: +# http://docs.spring.io/spring-boot/docs/current/reference/html/common-application-properties.html +# =================================================================== + +logging: + level: + ROOT: DEBUG + io.github.jhipster: DEBUG + it.cnr.isti.epasmed: DEBUG + +spring: + profiles: + active: dev + include: + - swagger + # For disable liquibase + # - no-liquibase + # Uncomment to activate TLS for the dev profile + #- tls + devtools: + restart: + enabled: true + additional-exclude: static/**,.h2.server.properties + livereload: + enabled: false # we use Webpack dev server + BrowserSync for livereload + jackson: + serialization: + indent-output: true + mail: + host: localhost + port: 25 + username: + password: + messages: + cache-duration: PT1S # 1 second, see the ISO 8601 standard + thymeleaf: + cache: false + +server: + port: 8080 + +# =================================================================== +# JHipster specific properties +# +# Full reference is available at: https://www.jhipster.tech/common-application-properties/ +# =================================================================== + +jhipster: + # CORS is only enabled by default with the "dev" profile, so BrowserSync can access the API + cors: + allowed-origins: '*' + allowed-methods: '*' + allowed-headers: '*' + exposed-headers: 'Link,X-Total-Count' + allow-credentials: true + max-age: 1800 + security: + remember-me: + # security key (this key should be unique for your application, and kept secret) + key: + mail: # specific JHipster mail property, for standard properties see MailProperties + base-url: http://127.0.0.1:8080 + metrics: + logs: # Reports metrics in the logs + enabled: false + report-frequency: 60 # in seconds + logging: + use-json-format: false # By default, logs are not in Json format + logstash: # Forward logs to logstash over a socket, used by LoggingConfiguration + enabled: false + host: localhost + port: 5000 + queue-size: 512 + audit-events: + retention-period: 30 # Number of days before audit events are deleted. + +# =================================================================== +# Application specific properties +# Add your own application properties here, see the ApplicationProperties class +# to have type-safe configuration, like in the JHipsterProperties above +# +# More documentation is available at: +# https://www.jhipster.tech/common-application-properties/ +# =================================================================== + +application: + datasource-epasmed: + datasource: + platform: io.github.jhipster.domain.util.FixedH2Dialect + url: jdbc:h2:file:./target/h2db/db/epasmed;DB_CLOSE_DELAY=-1 + username: + password: + hikari: + #type: com.zaxxer.hikari.HikariDataSource + poolName: epasmedPool + auto-commit: false + maximum-pool-size: 10 + liquibase: + # Remove 'faker' if you do not want the sample data to be loaded automatically + contexts: dev, faker + change-log: classpath:config/liquibase/epasmed/master.xml + enabled: true + + datasource-sistemainformativo: + datasource: + url: jdbc:postgresql://sistemainformativo-dev.isti.cnr.it:5432/sistemainformativo + driver-class-name: org.postgresql.Driver + username: epasmed + password: + hikari: + poolName: sistemainformativoPool + auto-commit: false + maximum-pool-size: 10 + + datasource-epas-rest: + #rest-url: https://epas-demo.devel.iit.cnr.it/rest + rest-url: https://epas.isti.cnr.it/rest + rest-username1: isti_registry_manager + rest-password1: + rest-username2: isti_person_day_reader + rest-password2: diff --git a/src/main/resources/config/application-prod.yml b/src/main/resources/config/application-prod.yml new file mode 100644 index 0000000..5e144e9 --- /dev/null +++ b/src/main/resources/config/application-prod.yml @@ -0,0 +1,164 @@ +# =================================================================== +# Spring Boot configuration for the "prod" profile. +# +# This configuration overrides the application.yml file. +# +# More information on profiles: https://www.jhipster.tech/profiles/ +# More information on configuration properties: https://www.jhipster.tech/common-application-properties/ +# =================================================================== + +# =================================================================== +# Standard Spring Boot properties. +# Full reference is available at: +# http://docs.spring.io/spring-boot/docs/current/reference/html/common-application-properties.html +# =================================================================== + +logging: + level: + ROOT: DEBUG + io.github.jhipster: DEBUG + it.cnr.isti.epasmed: DEBUG + +management: + metrics: + export: + prometheus: + enabled: false + +spring: + devtools: + restart: + enabled: false + livereload: + enabled: false + mail: + host: localhost + port: 25 + username: + password: + thymeleaf: + cache: true + +# datasource: +# type: com.zaxxer.hikari.HikariDataSource +# url: jdbc:postgresql://localhost:5432/epasmed +# username: epasmed +# password: +# hikari: +# poolName: Hikari +# auto-commit: false +# jpa: +# database-platform: io.github.jhipster.domain.util.FixedPostgreSQL10Dialect +# show-sql: false + +# =================================================================== +# To enable TLS in production, generate a certificate using: +# keytool -genkey -alias epasmed -storetype PKCS12 -keyalg RSA -keysize 2048 -keystore keystore.p12 -validity 3650 +# +# You can also use Let's Encrypt: +# https://maximilian-boehm.com/hp2121/Create-a-Java-Keystore-JKS-from-Let-s-Encrypt-Certificates.htm +# +# Then, modify the server.ssl properties so your "server" configuration looks like: +# +# server: +# port: 443 +# ssl: +# key-store: classpath:config/tls/keystore.p12 +# key-store-password: password +# key-store-type: PKCS12 +# key-alias: selfsigned +# # The ciphers suite enforce the security by deactivating some old and deprecated SSL cipher, this list was tested against SSL Labs (https://www.ssllabs.com/ssltest/) +# ciphers: TLS_ECDHE_RSA_WITH_AES_128_CBC_SHA,TLS_ECDHE_RSA_WITH_AES_256_CBC_SHA,TLS_ECDHE_RSA_WITH_AES_128_GCM_SHA256,TLS_ECDHE_RSA_WITH_AES_256_GCM_SHA384 ,TLS_DHE_RSA_WITH_AES_128_GCM_SHA256 ,TLS_DHE_RSA_WITH_AES_256_GCM_SHA384 ,TLS_ECDHE_RSA_WITH_AES_128_CBC_SHA256,TLS_ECDHE_RSA_WITH_AES_256_CBC_SHA384,TLS_DHE_RSA_WITH_AES_128_CBC_SHA256,TLS_DHE_RSA_WITH_AES_128_CBC_SHA,TLS_DHE_RSA_WITH_AES_256_CBC_SHA256,TLS_DHE_RSA_WITH_AES_256_CBC_SHA,TLS_RSA_WITH_AES_128_GCM_SHA256,TLS_RSA_WITH_AES_256_GCM_SHA384,TLS_RSA_WITH_AES_128_CBC_SHA256,TLS_RSA_WITH_AES_256_CBC_SHA256,TLS_RSA_WITH_AES_128_CBC_SHA,TLS_RSA_WITH_AES_256_CBC_SHA,TLS_DHE_RSA_WITH_CAMELLIA_256_CBC_SHA,TLS_RSA_WITH_CAMELLIA_256_CBC_SHA,TLS_DHE_RSA_WITH_CAMELLIA_128_CBC_SHA,TLS_RSA_WITH_CAMELLIA_128_CBC_SHA +# =================================================================== +server: + port: 8080 + compression: + enabled: true + mime-types: text/html,text/xml,text/plain,text/css, application/javascript, application/json + min-response-size: 1024 + +# =================================================================== +# JHipster specific properties +# +# Full reference is available at: https://www.jhipster.tech/common-application-properties/ +# =================================================================== + +jhipster: + http: + cache: # Used by the CachingHttpHeadersFilter + timeToLiveInDays: 1461 + security: + remember-me: + # security key (this key should be unique for your application, and kept secret) + key: + mail: # specific JHipster mail property, for standard properties see MailProperties + base-url: http://epasmed.isti.cnr.it # Modify according to your server's URL + metrics: + logs: # Reports metrics in the logs + enabled: false + report-frequency: 60 # in seconds + logging: + use-json-format: false # By default, logs are not in Json format + logstash: # Forward logs to logstash over a socket, used by LoggingConfiguration + enabled: false + host: localhost + port: 5000 + queue-size: 512 + audit-events: + retention-period: 30 # Number of days before audit events are deleted. + +# =================================================================== +# Application specific properties +# Add your own application properties here, see the ApplicationProperties class +# to have type-safe configuration, like in the JHipsterProperties above +# +# More documentation is available at: +# https://www.jhipster.tech/common-application-properties/ +# =================================================================== + +application: + datasource-epasmed: + datasource: + #type: com.zaxxer.hikari.HikariDataSource + platform: org.hibernate.dialect.PostgreSQLDialect + url: jdbc:postgresql://127.0.0.1:5432/epasmed + username: epasmed + password: + hikari: + poolName: epasmedPool + auto-commit: false + maximum-pool-size: 10 + liquibase: + # Replace by 'prod, faker' to add the faker context and have sample data loaded in production + contexts: prod, faker + change-log: classpath:config/liquibase/epasmed/master.xml + enabled: true + datasource-sistemainformativo: + datasource: + url: jdbc:postgresql://sistemainformativo-dev.isti.cnr.it:5432/sistemainformativo + driver-class-name: org.postgresql.Driver + username: epasmed + password: + hikari: + poolName: sistemainformativoPool + auto-commit: false + maximum-pool-size: 10 + datasource-epas-rest: + #rest-url: https://epas-demo.devel.iit.cnr.it/rest + rest-url: https://epas.isti.cnr.it/rest + rest-username1: isti_registry_manager + rest-password1: + rest-username2: isti_person_day_reader + rest-password2: +# datasource: +# type: com.zaxxer.hikari.HikariDataSource +# url: jdbc:postgresql://localhost:5432/epasmed +# username: epasmed +# password: +# hikari: +# poolName: Hikari +# auto-commit: false +# jpa: +# database-platform: io.github.jhipster.domain.util.FixedPostgreSQL10Dialect +# show-sql: false + diff --git a/src/main/resources/config/application-tls.yml b/src/main/resources/config/application-tls.yml new file mode 100644 index 0000000..27c9cd8 --- /dev/null +++ b/src/main/resources/config/application-tls.yml @@ -0,0 +1,19 @@ +# =================================================================== +# Activate this profile to enable TLS and HTTP/2. +# +# JHipster has generated a self-signed certificate, which will be used to encrypt traffic. +# As your browser will not understand this certificate, you will need to import it. +# +# Another (easiest) solution with Chrome is to enable the "allow-insecure-localhost" flag +# at chrome://flags/#allow-insecure-localhost +# =================================================================== +server: + ssl: + key-store: classpath:config/tls/keystore.p12 + key-store-password: password + key-store-type: PKCS12 + key-alias: selfsigned + ciphers: TLS_ECDHE_ECDSA_WITH_AES_128_GCM_SHA256, TLS_ECDHE_ECDSA_WITH_AES_256_GCM_SHA384, TLS_ECDHE_ECDSA_WITH_AES_128_CBC_SHA, TLS_ECDHE_ECDSA_WITH_AES_256_CBC_SHA, TLS_ECDHE_ECDSA_WITH_AES_128_CBC_SHA256, TLS_ECDHE_ECDSA_WITH_AES_256_CBC_SHA384, TLS_ECDHE_RSA_WITH_AES_128_GCM_SHA256, TLS_ECDHE_RSA_WITH_AES_256_GCM_SHA384, TLS_ECDHE_RSA_WITH_AES_128_CBC_SHA, TLS_ECDHE_RSA_WITH_AES_256_CBC_SHA, TLS_ECDHE_RSA_WITH_AES_128_CBC_SHA256, TLS_ECDHE_RSA_WITH_AES_256_CBC_SHA384, TLS_DHE_RSA_WITH_AES_128_GCM_SHA256, TLS_DHE_RSA_WITH_AES_256_GCM_SHA384, TLS_DHE_RSA_WITH_AES_128_CBC_SHA, TLS_DHE_RSA_WITH_AES_256_CBC_SHA, TLS_DHE_RSA_WITH_AES_128_CBC_SHA256, TLS_DHE_RSA_WITH_AES_256_CBC_SHA256 + enabled-protocols: TLSv1.2 + http2: + enabled: true diff --git a/src/main/resources/config/application.yml b/src/main/resources/config/application.yml new file mode 100644 index 0000000..fb42175 --- /dev/null +++ b/src/main/resources/config/application.yml @@ -0,0 +1,141 @@ +# =================================================================== +# Spring Boot configuration. +# +# This configuration will be overridden by the Spring profile you use, +# for example application-dev.yml if you use the "dev" profile. +# +# More information on profiles: https://www.jhipster.tech/profiles/ +# More information on configuration properties: https://www.jhipster.tech/common-application-properties/ +# =================================================================== + +# =================================================================== +# Standard Spring Boot properties. +# Full reference is available at: +# http://docs.spring.io/spring-boot/docs/current/reference/html/common-application-properties.html +# =================================================================== + +management: + endpoints: + web: + base-path: /management + exposure: + include: ['configprops', 'env', 'health', 'info', 'jhimetrics', 'logfile', 'loggers', 'prometheus', 'threaddump'] + endpoint: + health: + show-details: when-authorized + roles: 'ROLE_ADMIN' + jhimetrics: + enabled: true + info: + git: + mode: full + health: + mail: + enabled: false # When using the MailService, configure an SMTP server and set this to true + metrics: + export: + # Prometheus is the default metrics backend + prometheus: + enabled: true + step: 60 + enable: + http: true + jvm: true + logback: true + process: true + system: true + distribution: + percentiles-histogram: + all: true + percentiles: + all: 0, 0.5, 0.75, 0.95, 0.99, 1.0 + tags: + application: ${spring.application.name} + web: + server: + request: + autotime: + enabled: true + +spring: + application: + name: epasmed + profiles: + # The commented value for `active` can be replaced with valid Spring profiles to load. + # Otherwise, it will be filled in by maven when building the JAR file + # Either way, it can be overridden by `--spring.profiles.active` value passed in the commandline or `-Dspring.profiles.active` set in `JAVA_OPTS` + active: #spring.profiles.active# + jmx: + enabled: false + messages: + basename: i18n/messages + main: + allow-bean-definition-overriding: true + task: + execution: + thread-name-prefix: epasmed-task- + pool: + core-size: 2 + max-size: 50 + queue-capacity: 10000 + scheduling: + thread-name-prefix: epasmed-scheduling- + pool: + size: 2 + thymeleaf: + mode: HTML + output: + ansi: + enabled: always + +server: + servlet: + session: + cookie: + http-only: true + +# Properties to be exposed on the /info management endpoint +info: + # Comma separated list of profiles that will trigger the ribbon to show + display-ribbon-on-profiles: 'dev' + +# =================================================================== +# JHipster specific properties +# +# Full reference is available at: https://www.jhipster.tech/common-application-properties/ +# =================================================================== + +jhipster: + clientApp: + name: 'epasmedApp' + # By default CORS is disabled. Uncomment to enable. + # cors: + # allowed-origins: "*" + # allowed-methods: "*" + # allowed-headers: "*" + # exposed-headers: "Link,X-Total-Count" + # allow-credentials: true + # max-age: 1800 + mail: + from: epasmed@localhost + swagger: + default-include-pattern: /api/.* + title: ePASMed API + description: ePASMed API documentation + version: 0.0.1 + terms-of-service-url: + contact-name: + contact-url: + contact-email: + license: unlicensed + license-url: +# =================================================================== +# Application specific properties +# Add your own application properties here, see the ApplicationProperties class +# to have type-safe configuration, like in the JHipsterProperties above +# +# More documentation is available at: +# https://www.jhipster.tech/common-application-properties/ +# =================================================================== + +#application: diff --git a/src/main/resources/config/liquibase/epasmed/changelog/00000000000000_initial_schema.xml b/src/main/resources/config/liquibase/epasmed/changelog/00000000000000_initial_schema.xml new file mode 100644 index 0000000..66beaf1 --- /dev/null +++ b/src/main/resources/config/liquibase/epasmed/changelog/00000000000000_initial_schema.xml @@ -0,0 +1,202 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/src/main/resources/config/liquibase/epasmed/data/authority.csv b/src/main/resources/config/liquibase/epasmed/data/authority.csv new file mode 100644 index 0000000..6400545 --- /dev/null +++ b/src/main/resources/config/liquibase/epasmed/data/authority.csv @@ -0,0 +1,4 @@ +name +ROLE_ADMIN +ROLE_USER +ROLE_PROMETHEUS diff --git a/src/main/resources/config/liquibase/epasmed/data/tabssi.csv b/src/main/resources/config/liquibase/epasmed/data/tabssi.csv new file mode 100644 index 0000000..0270fdb --- /dev/null +++ b/src/main/resources/config/liquibase/epasmed/data/tabssi.csv @@ -0,0 +1,17 @@ +id,nome,operazioni,id_flusso,last_update +1,anagrafico,R,116002,2020-12-07 06:33:01 +2,mail,R,116214,2020-12-14 06:33:01 +3,telefoni,R,115866,2020-12-02 06:33:01 +4,recapiti,R,116214,2020-12-14 06:33:01 +5,titoli_studio,R,116214,2020-12-14 06:33:01 +6,gruppi,R,114623,2020-11-15 06:33:03 +7,gruppo_pers,R,116214,2020-12-14 06:33:01 +8,posizioni,R,115866,2020-12-02 06:33:01 +9,proroghe,R,115692,2020-11-28 06:33:02 +10,cartellini,W,116175,2020-12-12 04:58:52 +11,cartellini_rendicontazioni,W,116174,2020-12-12 04:58:52 +12,pers_orario,W,115869,2020-12-02 04:33:02 +13,lavoro_fuori_sede,W,116115,2020-12-11 04:33:01 +14,festivita,W,104649,2020-04-21 04:33:02 +15,orario,W,84546,2019-02-19 17:20:56 +16,aspettative,W,113219,2020-10-15 04:33:03 diff --git a/src/main/resources/config/liquibase/epasmed/data/user.csv b/src/main/resources/config/liquibase/epasmed/data/user.csv new file mode 100644 index 0000000..c07a405 --- /dev/null +++ b/src/main/resources/config/liquibase/epasmed/data/user.csv @@ -0,0 +1,6 @@ +id;login;password_hash;first_name;last_name;email;image_url;activated;lang_key;created_by;last_modified_by +1;system;$2a$10$mE.qmcV0mFU5NcKh73TZx.z4ueI/.bDWbj0T1BYyqP481kGGarKLG;System;System;system@localhost;;true;en;system;system +2;anonymoususer;$2a$10$j8S5d7Sr7.8VTOYNviDPOeWX8KcYILUVJBsYV83Y5NtECayypx9lO;Anonymous;User;anonymous@localhost;;true;en;system;system +3;admin;$2a$10$gSAhZrxMllrbgj/kkK9UceBPpChGWJA7SYIb1Mqo.n5aNLq1/oRrC;Administrator;Administrator;admin@localhost;;true;en;system;system +4;user;$2a$10$VEjxo0jq2YG9Rbk2HmX9S.k1uZBGYUHdUcid3g/vfiEl7lwWgOH/K;User;User;user@localhost;;true;en;system;system +5;prometheus;$2a$10$/NMhceiqWrpL2BXXaZ2CFOcpL73AdrbzU9n2qylXDdEYI/A/M57dC;Prometheus;Prometheus;prometheus@localhost;;true;en;system;system \ No newline at end of file diff --git a/src/main/resources/config/liquibase/epasmed/data/user_authority.csv b/src/main/resources/config/liquibase/epasmed/data/user_authority.csv new file mode 100644 index 0000000..2a59780 --- /dev/null +++ b/src/main/resources/config/liquibase/epasmed/data/user_authority.csv @@ -0,0 +1,8 @@ +user_id;authority_name +1;ROLE_ADMIN +1;ROLE_USER +3;ROLE_ADMIN +3;ROLE_USER +4;ROLE_USER +5;ROLE_PROMETHEUS +5;ROLE_USER diff --git a/src/main/resources/config/liquibase/epasmed/master.xml b/src/main/resources/config/liquibase/epasmed/master.xml new file mode 100644 index 0000000..522d332 --- /dev/null +++ b/src/main/resources/config/liquibase/epasmed/master.xml @@ -0,0 +1,20 @@ + + + + + + + + + + + + + + + + + diff --git a/src/main/resources/config/tls/keystore.p12 b/src/main/resources/config/tls/keystore.p12 new file mode 100644 index 0000000000000000000000000000000000000000..a9942482d3fedd5f835f3127834ed2831cd875d5 GIT binary patch literal 2607 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 D?4`g4 literal 0 HcmV?d00001 diff --git a/src/main/resources/i18n/messages.properties b/src/main/resources/i18n/messages.properties new file mode 100644 index 0000000..25b79df --- /dev/null +++ b/src/main/resources/i18n/messages.properties @@ -0,0 +1,21 @@ +# Error page +error.title=Your request cannot be processed +error.subtitle=Sorry, an error has occurred. +error.status=Status: +error.message=Message: + +# Activation email +email.activation.title=epasmed account activation is required +email.activation.greeting=Dear {0} +email.activation.text1=Your epasmed account has been created, please click on the URL below to activate it: +email.activation.text2=Regards, +email.signature=epasmed Team. + +# Creation email +email.creation.text1=Your epasmed account has been created, please click on the URL below to access it: + +# Reset email +email.reset.title=epasmed password reset +email.reset.greeting=Dear {0} +email.reset.text1=For your epasmed account a password reset was requested, please click on the URL below to reset it: +email.reset.text2=Regards, diff --git a/src/main/resources/logback-spring.xml b/src/main/resources/logback-spring.xml new file mode 100644 index 0000000..46f66b0 --- /dev/null +++ b/src/main/resources/logback-spring.xml @@ -0,0 +1,68 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + true + + + diff --git a/src/main/resources/swagger/api.yml b/src/main/resources/swagger/api.yml new file mode 100644 index 0000000..3fea6e9 --- /dev/null +++ b/src/main/resources/swagger/api.yml @@ -0,0 +1,33 @@ +# API-first development with OpenAPI +# This file will be used at compile time to generate Spring-MVC endpoint stubs using openapi-generator +openapi: '3.0.1' +info: + title: 'ePASMed' + version: 0.0.1 +servers: + - url: http://localhost:8080/api + description: Development server + - url: https://localhost:8080/api + description: Development server with TLS Profile +paths: {} +components: + responses: + Problem: + description: error occurred - see status code and problem object for more information. + content: + 'application/problem+json': + schema: + $ref: 'https://opensource.zalando.com/problem/schema.yaml#/Problem' + + securitySchemes: + sessionId: + type: apiKey + name: JSESSIONID + in: cookie + csrf: + type: apiKey + name: XSRF-TOKEN + in: cookie +security: + - sessionId: [] + - csrf: [] diff --git a/src/main/resources/templates/error.html b/src/main/resources/templates/error.html new file mode 100644 index 0000000..a421bd2 --- /dev/null +++ b/src/main/resources/templates/error.html @@ -0,0 +1,87 @@ + + + + + + Your request cannot be processed + + + +

+

Your request cannot be processed :(

+ +

Sorry, an error has occurred.

+ + Status:  ()
+ + Message: 
+
+
+ + diff --git a/src/main/resources/templates/mail/activationEmail.html b/src/main/resources/templates/mail/activationEmail.html new file mode 100644 index 0000000..07d8d3b --- /dev/null +++ b/src/main/resources/templates/mail/activationEmail.html @@ -0,0 +1,25 @@ + + + + JHipster activation + + + + +

+ Dear +

+

+ Your JHipster account has been created, please click on the URL below to activate it: +

+

+ Activation link +

+

+ Regards, +
+ JHipster. +

+ + diff --git a/src/main/resources/templates/mail/creationEmail.html b/src/main/resources/templates/mail/creationEmail.html new file mode 100644 index 0000000..1d9276e --- /dev/null +++ b/src/main/resources/templates/mail/creationEmail.html @@ -0,0 +1,25 @@ + + + + JHipster creation + + + + +

+ Dear +

+

+ Your JHipster account has been created, please click on the URL below to access it: +

+

+ Login link +

+

+ Regards, +
+ JHipster. +

+ + diff --git a/src/main/resources/templates/mail/passwordResetEmail.html b/src/main/resources/templates/mail/passwordResetEmail.html new file mode 100644 index 0000000..4e709cf --- /dev/null +++ b/src/main/resources/templates/mail/passwordResetEmail.html @@ -0,0 +1,25 @@ + + + + JHipster password reset + + + + +

+ Dear +

+

+ For your JHipster account a password reset was requested, please click on the URL below to reset it: +

+

+ Login link +

+

+ Regards, +
+ JHipster. +

+ + diff --git a/src/main/webapp/404.html b/src/main/webapp/404.html new file mode 100644 index 0000000..21a47e5 --- /dev/null +++ b/src/main/webapp/404.html @@ -0,0 +1,61 @@ + + + + + Page Not Found + + + + + +

Page Not Found

+

Sorry, but the page you were trying to view does not exist.

+ + + diff --git a/src/main/webapp/WEB-INF/web.xml b/src/main/webapp/WEB-INF/web.xml new file mode 100644 index 0000000..f1611b5 --- /dev/null +++ b/src/main/webapp/WEB-INF/web.xml @@ -0,0 +1,13 @@ + + + + + html + text/html;charset=utf-8 + + + diff --git a/src/main/webapp/app/account/account.module.ts b/src/main/webapp/app/account/account.module.ts new file mode 100644 index 0000000..80e11ee --- /dev/null +++ b/src/main/webapp/app/account/account.module.ts @@ -0,0 +1,29 @@ +import { NgModule } from '@angular/core'; +import { RouterModule } from '@angular/router'; + +import { EpasmedSharedModule } from 'app/shared/shared.module'; + +import { SessionsComponent } from './sessions/sessions.component'; +import { PasswordStrengthBarComponent } from './password/password-strength-bar.component'; +import { RegisterComponent } from './register/register.component'; +import { ActivateComponent } from './activate/activate.component'; +import { PasswordComponent } from './password/password.component'; +import { PasswordResetInitComponent } from './password-reset/init/password-reset-init.component'; +import { PasswordResetFinishComponent } from './password-reset/finish/password-reset-finish.component'; +import { SettingsComponent } from './settings/settings.component'; +import { accountState } from './account.route'; + +@NgModule({ + imports: [EpasmedSharedModule, RouterModule.forChild(accountState)], + declarations: [ + ActivateComponent, + RegisterComponent, + PasswordComponent, + PasswordStrengthBarComponent, + PasswordResetInitComponent, + PasswordResetFinishComponent, + SessionsComponent, + SettingsComponent, + ], +}) +export class AccountModule {} diff --git a/src/main/webapp/app/account/account.route.ts b/src/main/webapp/app/account/account.route.ts new file mode 100644 index 0000000..2e51b90 --- /dev/null +++ b/src/main/webapp/app/account/account.route.ts @@ -0,0 +1,26 @@ +import { Routes } from '@angular/router'; + +import { activateRoute } from './activate/activate.route'; +import { passwordRoute } from './password/password.route'; +import { passwordResetFinishRoute } from './password-reset/finish/password-reset-finish.route'; +import { passwordResetInitRoute } from './password-reset/init/password-reset-init.route'; +import { registerRoute } from './register/register.route'; +import { sessionsRoute } from './sessions/sessions.route'; +import { settingsRoute } from './settings/settings.route'; + +const ACCOUNT_ROUTES = [ + activateRoute, + passwordRoute, + passwordResetFinishRoute, + passwordResetInitRoute, + registerRoute, + sessionsRoute, + settingsRoute, +]; + +export const accountState: Routes = [ + { + path: '', + children: ACCOUNT_ROUTES, + }, +]; diff --git a/src/main/webapp/app/account/activate/activate.component.html b/src/main/webapp/app/account/activate/activate.component.html new file mode 100644 index 0000000..84d514d --- /dev/null +++ b/src/main/webapp/app/account/activate/activate.component.html @@ -0,0 +1,16 @@ +
+
+
+

Activation

+ +
+ Your user account has been activated. Please + sign in. +
+ +
+ Your user could not be activated. Please use the registration form to sign up. +
+
+
+
diff --git a/src/main/webapp/app/account/activate/activate.component.ts b/src/main/webapp/app/account/activate/activate.component.ts new file mode 100644 index 0000000..8e061f5 --- /dev/null +++ b/src/main/webapp/app/account/activate/activate.component.ts @@ -0,0 +1,28 @@ +import { Component, OnInit } from '@angular/core'; +import { ActivatedRoute } from '@angular/router'; +import { flatMap } from 'rxjs/operators'; + +import { LoginModalService } from 'app/core/login/login-modal.service'; +import { ActivateService } from './activate.service'; + +@Component({ + selector: 'jhi-activate', + templateUrl: './activate.component.html', +}) +export class ActivateComponent implements OnInit { + error = false; + success = false; + + constructor(private activateService: ActivateService, private loginModalService: LoginModalService, private route: ActivatedRoute) {} + + ngOnInit(): void { + this.route.queryParams.pipe(flatMap(params => this.activateService.get(params.key))).subscribe( + () => (this.success = true), + () => (this.error = true) + ); + } + + login(): void { + this.loginModalService.open(); + } +} diff --git a/src/main/webapp/app/account/activate/activate.route.ts b/src/main/webapp/app/account/activate/activate.route.ts new file mode 100644 index 0000000..c08218f --- /dev/null +++ b/src/main/webapp/app/account/activate/activate.route.ts @@ -0,0 +1,12 @@ +import { Route } from '@angular/router'; + +import { ActivateComponent } from './activate.component'; + +export const activateRoute: Route = { + path: 'activate', + component: ActivateComponent, + data: { + authorities: [], + pageTitle: 'Activation', + }, +}; diff --git a/src/main/webapp/app/account/activate/activate.service.ts b/src/main/webapp/app/account/activate/activate.service.ts new file mode 100644 index 0000000..2c6046e --- /dev/null +++ b/src/main/webapp/app/account/activate/activate.service.ts @@ -0,0 +1,16 @@ +import { Injectable } from '@angular/core'; +import { HttpClient, HttpParams } from '@angular/common/http'; +import { Observable } from 'rxjs'; + +import { SERVER_API_URL } from 'app/app.constants'; + +@Injectable({ providedIn: 'root' }) +export class ActivateService { + constructor(private http: HttpClient) {} + + get(key: string): Observable<{}> { + return this.http.get(SERVER_API_URL + 'api/activate', { + params: new HttpParams().set('key', key), + }); + } +} diff --git a/src/main/webapp/app/account/password-reset/finish/password-reset-finish.component.html b/src/main/webapp/app/account/password-reset/finish/password-reset-finish.component.html new file mode 100644 index 0000000..bc91735 --- /dev/null +++ b/src/main/webapp/app/account/password-reset/finish/password-reset-finish.component.html @@ -0,0 +1,84 @@ +
+
+
+

Reset password

+ +
+ The password reset key is missing. +
+ +
+ Choose a new password +
+ +
+ Your password couldn't be reset. Remember a password request is only valid for 24 hours. +
+ +
+ Your password has been reset. Please + sign in. +
+ +
+ The password and its confirmation do not match! +
+ +
+
+
+ + + +
+ + Your password is required. + + + + Your password is required to be at least 4 characters. + + + + Your password cannot be longer than 50 characters. + +
+ + +
+ +
+ + + +
+ + Your password confirmation is required. + + + + Your password confirmation is required to be at least 4 characters. + + + + Your password confirmation cannot be longer than 50 characters. + +
+
+ + +
+
+
+
+
diff --git a/src/main/webapp/app/account/password-reset/finish/password-reset-finish.component.ts b/src/main/webapp/app/account/password-reset/finish/password-reset-finish.component.ts new file mode 100644 index 0000000..015b134 --- /dev/null +++ b/src/main/webapp/app/account/password-reset/finish/password-reset-finish.component.ts @@ -0,0 +1,69 @@ +import { Component, OnInit, AfterViewInit, ElementRef, ViewChild } from '@angular/core'; +import { FormBuilder, Validators } from '@angular/forms'; +import { ActivatedRoute } from '@angular/router'; + +import { LoginModalService } from 'app/core/login/login-modal.service'; +import { PasswordResetFinishService } from './password-reset-finish.service'; + +@Component({ + selector: 'jhi-password-reset-finish', + templateUrl: './password-reset-finish.component.html', +}) +export class PasswordResetFinishComponent implements OnInit, AfterViewInit { + @ViewChild('newPassword', { static: false }) + newPassword?: ElementRef; + + initialized = false; + doNotMatch = false; + error = false; + success = false; + key = ''; + + passwordForm = this.fb.group({ + newPassword: ['', [Validators.required, Validators.minLength(4), Validators.maxLength(50)]], + confirmPassword: ['', [Validators.required, Validators.minLength(4), Validators.maxLength(50)]], + }); + + constructor( + private passwordResetFinishService: PasswordResetFinishService, + private loginModalService: LoginModalService, + private route: ActivatedRoute, + private fb: FormBuilder + ) {} + + ngOnInit(): void { + this.route.queryParams.subscribe(params => { + if (params['key']) { + this.key = params['key']; + } + this.initialized = true; + }); + } + + ngAfterViewInit(): void { + if (this.newPassword) { + this.newPassword.nativeElement.focus(); + } + } + + finishReset(): void { + this.doNotMatch = false; + this.error = false; + + const newPassword = this.passwordForm.get(['newPassword'])!.value; + const confirmPassword = this.passwordForm.get(['confirmPassword'])!.value; + + if (newPassword !== confirmPassword) { + this.doNotMatch = true; + } else { + this.passwordResetFinishService.save(this.key, newPassword).subscribe( + () => (this.success = true), + () => (this.error = true) + ); + } + } + + login(): void { + this.loginModalService.open(); + } +} diff --git a/src/main/webapp/app/account/password-reset/finish/password-reset-finish.route.ts b/src/main/webapp/app/account/password-reset/finish/password-reset-finish.route.ts new file mode 100644 index 0000000..afa0a44 --- /dev/null +++ b/src/main/webapp/app/account/password-reset/finish/password-reset-finish.route.ts @@ -0,0 +1,12 @@ +import { Route } from '@angular/router'; + +import { PasswordResetFinishComponent } from './password-reset-finish.component'; + +export const passwordResetFinishRoute: Route = { + path: 'reset/finish', + component: PasswordResetFinishComponent, + data: { + authorities: [], + pageTitle: 'Password', + }, +}; diff --git a/src/main/webapp/app/account/password-reset/finish/password-reset-finish.service.ts b/src/main/webapp/app/account/password-reset/finish/password-reset-finish.service.ts new file mode 100644 index 0000000..2473629 --- /dev/null +++ b/src/main/webapp/app/account/password-reset/finish/password-reset-finish.service.ts @@ -0,0 +1,14 @@ +import { Injectable } from '@angular/core'; +import { HttpClient } from '@angular/common/http'; +import { Observable } from 'rxjs'; + +import { SERVER_API_URL } from 'app/app.constants'; + +@Injectable({ providedIn: 'root' }) +export class PasswordResetFinishService { + constructor(private http: HttpClient) {} + + save(key: string, newPassword: string): Observable<{}> { + return this.http.post(SERVER_API_URL + 'api/account/reset-password/finish', { key, newPassword }); + } +} diff --git a/src/main/webapp/app/account/password-reset/init/password-reset-init.component.html b/src/main/webapp/app/account/password-reset/init/password-reset-init.component.html new file mode 100644 index 0000000..4149719 --- /dev/null +++ b/src/main/webapp/app/account/password-reset/init/password-reset-init.component.html @@ -0,0 +1,49 @@ +
+
+
+

Reset your password

+ + + +
+ Enter the email address you used to register. +
+ +
+ Check your emails for details on how to reset your password. +
+ +
+
+ + + +
+ + Your email is required. + + + + Your email is invalid. + + + + Your email is required to be at least 5 characters. + + + + Your email cannot be longer than 100 characters. + +
+
+ + +
+
+
+
diff --git a/src/main/webapp/app/account/password-reset/init/password-reset-init.component.ts b/src/main/webapp/app/account/password-reset/init/password-reset-init.component.ts new file mode 100644 index 0000000..ec99fcd --- /dev/null +++ b/src/main/webapp/app/account/password-reset/init/password-reset-init.component.ts @@ -0,0 +1,30 @@ +import { Component, AfterViewInit, ElementRef, ViewChild } from '@angular/core'; +import { FormBuilder, Validators } from '@angular/forms'; + +import { PasswordResetInitService } from './password-reset-init.service'; + +@Component({ + selector: 'jhi-password-reset-init', + templateUrl: './password-reset-init.component.html', +}) +export class PasswordResetInitComponent implements AfterViewInit { + @ViewChild('email', { static: false }) + email?: ElementRef; + + success = false; + resetRequestForm = this.fb.group({ + email: ['', [Validators.required, Validators.minLength(5), Validators.maxLength(254), Validators.email]], + }); + + constructor(private passwordResetInitService: PasswordResetInitService, private fb: FormBuilder) {} + + ngAfterViewInit(): void { + if (this.email) { + this.email.nativeElement.focus(); + } + } + + requestReset(): void { + this.passwordResetInitService.save(this.resetRequestForm.get(['email'])!.value).subscribe(() => (this.success = true)); + } +} diff --git a/src/main/webapp/app/account/password-reset/init/password-reset-init.route.ts b/src/main/webapp/app/account/password-reset/init/password-reset-init.route.ts new file mode 100644 index 0000000..2fe88dd --- /dev/null +++ b/src/main/webapp/app/account/password-reset/init/password-reset-init.route.ts @@ -0,0 +1,12 @@ +import { Route } from '@angular/router'; + +import { PasswordResetInitComponent } from './password-reset-init.component'; + +export const passwordResetInitRoute: Route = { + path: 'reset/request', + component: PasswordResetInitComponent, + data: { + authorities: [], + pageTitle: 'Password', + }, +}; diff --git a/src/main/webapp/app/account/password-reset/init/password-reset-init.service.ts b/src/main/webapp/app/account/password-reset/init/password-reset-init.service.ts new file mode 100644 index 0000000..368500c --- /dev/null +++ b/src/main/webapp/app/account/password-reset/init/password-reset-init.service.ts @@ -0,0 +1,14 @@ +import { Injectable } from '@angular/core'; +import { HttpClient } from '@angular/common/http'; +import { Observable } from 'rxjs'; + +import { SERVER_API_URL } from 'app/app.constants'; + +@Injectable({ providedIn: 'root' }) +export class PasswordResetInitService { + constructor(private http: HttpClient) {} + + save(mail: string): Observable<{}> { + return this.http.post(SERVER_API_URL + 'api/account/reset-password/init', mail); + } +} diff --git a/src/main/webapp/app/account/password/password-strength-bar.component.ts b/src/main/webapp/app/account/password/password-strength-bar.component.ts new file mode 100644 index 0000000..bc4a41b --- /dev/null +++ b/src/main/webapp/app/account/password/password-strength-bar.component.ts @@ -0,0 +1,83 @@ +import { Component, ElementRef, Input, Renderer2 } from '@angular/core'; + +@Component({ + selector: 'jhi-password-strength-bar', + template: `
+ Password strength: +
    +
  • +
  • +
  • +
  • +
  • +
+
`, + styleUrls: ['password-strength-bar.scss'], +}) +export class PasswordStrengthBarComponent { + colors = ['#F00', '#F90', '#FF0', '#9F0', '#0F0']; + + constructor(private renderer: Renderer2, private elementRef: ElementRef) {} + + measureStrength(p: string): number { + let force = 0; + const regex = /[$-/:-?{-~!"^_`[\]]/g; // " + const lowerLetters = /[a-z]+/.test(p); + const upperLetters = /[A-Z]+/.test(p); + const numbers = /[0-9]+/.test(p); + const symbols = regex.test(p); + + const flags = [lowerLetters, upperLetters, numbers, symbols]; + const passedMatches = flags.filter((isMatchedFlag: boolean) => { + return isMatchedFlag === true; + }).length; + + force += 2 * p.length + (p.length >= 10 ? 1 : 0); + force += passedMatches * 10; + + // penalty (short password) + force = p.length <= 6 ? Math.min(force, 10) : force; + + // penalty (poor variety of characters) + force = passedMatches === 1 ? Math.min(force, 10) : force; + force = passedMatches === 2 ? Math.min(force, 20) : force; + force = passedMatches === 3 ? Math.min(force, 40) : force; + + return force; + } + + getColor(s: number): { idx: number; color: string } { + let idx = 0; + if (s <= 10) { + idx = 0; + } else if (s <= 20) { + idx = 1; + } else if (s <= 30) { + idx = 2; + } else if (s <= 40) { + idx = 3; + } else { + idx = 4; + } + return { idx: idx + 1, color: this.colors[idx] }; + } + + @Input() + set passwordToCheck(password: string) { + if (password) { + const c = this.getColor(this.measureStrength(password)); + const element = this.elementRef.nativeElement; + if (element.className) { + this.renderer.removeClass(element, element.className); + } + const lis = element.getElementsByTagName('li'); + for (let i = 0; i < lis.length; i++) { + if (i < c.idx) { + this.renderer.setStyle(lis[i], 'backgroundColor', c.color); + } else { + this.renderer.setStyle(lis[i], 'backgroundColor', '#DDD'); + } + } + } + } +} diff --git a/src/main/webapp/app/account/password/password-strength-bar.scss b/src/main/webapp/app/account/password/password-strength-bar.scss new file mode 100644 index 0000000..67ce468 --- /dev/null +++ b/src/main/webapp/app/account/password/password-strength-bar.scss @@ -0,0 +1,23 @@ +/* ========================================================================== +start Password strength bar style +========================================================================== */ +ul#strength { + display: inline; + list-style: none; + margin: 0; + margin-left: 15px; + padding: 0; + vertical-align: 2px; +} + +.point { + background: #ddd; + border-radius: 2px; + display: inline-block; + height: 5px; + margin-right: 1px; + width: 20px; + &:last-child { + margin: 0 !important; + } +} diff --git a/src/main/webapp/app/account/password/password.component.html b/src/main/webapp/app/account/password/password.component.html new file mode 100644 index 0000000..5014c58 --- /dev/null +++ b/src/main/webapp/app/account/password/password.component.html @@ -0,0 +1,87 @@ +
+
+
+

Password for [{{ account.login }}]

+ +
+ Password changed! +
+ +
+ An error has occurred! The password could not be changed. +
+ +
+ The password and its confirmation do not match! +
+ +
+
+ + + +
+ + Your password is required. + +
+
+ +
+ + + +
+ + Your password is required. + + + + Your password is required to be at least 4 characters. + + + + Your password cannot be longer than 50 characters. + +
+ + +
+ +
+ + + +
+ + Your confirmation password is required. + + + + Your confirmation password is required to be at least 4 characters. + + + + Your confirmation password cannot be longer than 50 characters. + +
+
+ + +
+
+
+
diff --git a/src/main/webapp/app/account/password/password.component.ts b/src/main/webapp/app/account/password/password.component.ts new file mode 100644 index 0000000..ff46cdd --- /dev/null +++ b/src/main/webapp/app/account/password/password.component.ts @@ -0,0 +1,45 @@ +import { Component, OnInit } from '@angular/core'; +import { FormBuilder, Validators } from '@angular/forms'; +import { Observable } from 'rxjs'; + +import { AccountService } from 'app/core/auth/account.service'; +import { Account } from 'app/core/user/account.model'; +import { PasswordService } from './password.service'; + +@Component({ + selector: 'jhi-password', + templateUrl: './password.component.html', +}) +export class PasswordComponent implements OnInit { + doNotMatch = false; + error = false; + success = false; + account$?: Observable; + passwordForm = this.fb.group({ + currentPassword: ['', [Validators.required]], + newPassword: ['', [Validators.required, Validators.minLength(4), Validators.maxLength(50)]], + confirmPassword: ['', [Validators.required, Validators.minLength(4), Validators.maxLength(50)]], + }); + + constructor(private passwordService: PasswordService, private accountService: AccountService, private fb: FormBuilder) {} + + ngOnInit(): void { + this.account$ = this.accountService.identity(); + } + + changePassword(): void { + this.error = false; + this.success = false; + this.doNotMatch = false; + + const newPassword = this.passwordForm.get(['newPassword'])!.value; + if (newPassword !== this.passwordForm.get(['confirmPassword'])!.value) { + this.doNotMatch = true; + } else { + this.passwordService.save(newPassword, this.passwordForm.get(['currentPassword'])!.value).subscribe( + () => (this.success = true), + () => (this.error = true) + ); + } + } +} diff --git a/src/main/webapp/app/account/password/password.route.ts b/src/main/webapp/app/account/password/password.route.ts new file mode 100644 index 0000000..39d4649 --- /dev/null +++ b/src/main/webapp/app/account/password/password.route.ts @@ -0,0 +1,15 @@ +import { Route } from '@angular/router'; + +import { UserRouteAccessService } from 'app/core/auth/user-route-access-service'; +import { PasswordComponent } from './password.component'; +import { Authority } from 'app/shared/constants/authority.constants'; + +export const passwordRoute: Route = { + path: 'password', + component: PasswordComponent, + data: { + authorities: [Authority.USER], + pageTitle: 'Password', + }, + canActivate: [UserRouteAccessService], +}; diff --git a/src/main/webapp/app/account/password/password.service.ts b/src/main/webapp/app/account/password/password.service.ts new file mode 100644 index 0000000..13017f8 --- /dev/null +++ b/src/main/webapp/app/account/password/password.service.ts @@ -0,0 +1,14 @@ +import { Injectable } from '@angular/core'; +import { HttpClient } from '@angular/common/http'; +import { Observable } from 'rxjs'; + +import { SERVER_API_URL } from 'app/app.constants'; + +@Injectable({ providedIn: 'root' }) +export class PasswordService { + constructor(private http: HttpClient) {} + + save(newPassword: string, currentPassword: string): Observable<{}> { + return this.http.post(SERVER_API_URL + 'api/account/change-password', { currentPassword, newPassword }); + } +} diff --git a/src/main/webapp/app/account/register/register.component.html b/src/main/webapp/app/account/register/register.component.html new file mode 100644 index 0000000..1cff15f --- /dev/null +++ b/src/main/webapp/app/account/register/register.component.html @@ -0,0 +1,144 @@ +
+
+
+

Registration

+ +
+ Registration saved! Please check your email for confirmation. +
+ +
+ Registration failed! Please try again later. +
+ +
+ Login name already registered! Please choose another one. +
+ +
+ Email is already in use! Please choose another one. +
+ +
+ The password and its confirmation do not match! +
+
+
+ +
+
+
+
+ + + +
+ + Your username is required. + + + + Your username is required to be at least 1 character. + + + + Your username cannot be longer than 50 characters. + + + + Your username can only contain letters and digits. + +
+
+ +
+ + + +
+ + Your email is required. + + + + Your email is invalid. + + + + Your email is required to be at least 5 characters. + + + + Your email cannot be longer than 100 characters. + +
+
+ +
+ + + +
+ + Your password is required. + + + + Your password is required to be at least 4 characters. + + + + Your password cannot be longer than 50 characters. + +
+ + +
+ +
+ + + +
+ + Your confirmation password is required. + + + + Your confirmation password is required to be at least 4 characters. + + + + Your confirmation password cannot be longer than 50 characters. + +
+
+ + +
+ +
+ If you want to + sign in, you can try the default accounts:
- Administrator (login="admin" and password="admin")
- User (login="user" and password="user").
+
+
+
+
diff --git a/src/main/webapp/app/account/register/register.component.ts b/src/main/webapp/app/account/register/register.component.ts new file mode 100644 index 0000000..b131a1e --- /dev/null +++ b/src/main/webapp/app/account/register/register.component.ts @@ -0,0 +1,78 @@ +import { Component, AfterViewInit, ElementRef, ViewChild } from '@angular/core'; +import { HttpErrorResponse } from '@angular/common/http'; +import { FormBuilder, Validators } from '@angular/forms'; + +import { EMAIL_ALREADY_USED_TYPE, LOGIN_ALREADY_USED_TYPE } from 'app/shared/constants/error.constants'; +import { LoginModalService } from 'app/core/login/login-modal.service'; +import { RegisterService } from './register.service'; + +@Component({ + selector: 'jhi-register', + templateUrl: './register.component.html', +}) +export class RegisterComponent implements AfterViewInit { + @ViewChild('login', { static: false }) + login?: ElementRef; + + doNotMatch = false; + error = false; + errorEmailExists = false; + errorUserExists = false; + success = false; + + registerForm = this.fb.group({ + login: [ + '', + [ + Validators.required, + Validators.minLength(1), + Validators.maxLength(50), + Validators.pattern('^[a-zA-Z0-9!$&*+=?^_`{|}~.-]+@[a-zA-Z0-9-]+(?:\\.[a-zA-Z0-9-]+)*$|^[_.@A-Za-z0-9-]+$'), + ], + ], + email: ['', [Validators.required, Validators.minLength(5), Validators.maxLength(254), Validators.email]], + password: ['', [Validators.required, Validators.minLength(4), Validators.maxLength(50)]], + confirmPassword: ['', [Validators.required, Validators.minLength(4), Validators.maxLength(50)]], + }); + + constructor(private loginModalService: LoginModalService, private registerService: RegisterService, private fb: FormBuilder) {} + + ngAfterViewInit(): void { + if (this.login) { + this.login.nativeElement.focus(); + } + } + + register(): void { + this.doNotMatch = false; + this.error = false; + this.errorEmailExists = false; + this.errorUserExists = false; + + const password = this.registerForm.get(['password'])!.value; + if (password !== this.registerForm.get(['confirmPassword'])!.value) { + this.doNotMatch = true; + } else { + const login = this.registerForm.get(['login'])!.value; + const email = this.registerForm.get(['email'])!.value; + this.registerService.save({ login, email, password, langKey: 'en' }).subscribe( + () => (this.success = true), + response => this.processError(response) + ); + } + } + + openLogin(): void { + this.loginModalService.open(); + } + + private processError(response: HttpErrorResponse): void { + if (response.status === 400 && response.error.type === LOGIN_ALREADY_USED_TYPE) { + this.errorUserExists = true; + } else if (response.status === 400 && response.error.type === EMAIL_ALREADY_USED_TYPE) { + this.errorEmailExists = true; + } else { + this.error = true; + } + } +} diff --git a/src/main/webapp/app/account/register/register.route.ts b/src/main/webapp/app/account/register/register.route.ts new file mode 100644 index 0000000..6c6a3f7 --- /dev/null +++ b/src/main/webapp/app/account/register/register.route.ts @@ -0,0 +1,12 @@ +import { Route } from '@angular/router'; +import { RegisterComponent } from './register.component'; +import { Authority } from 'app/shared/constants/authority.constants'; + +export const registerRoute: Route = { + path: 'register', + component: RegisterComponent, + data: { + authorities: [Authority.ADMIN], + pageTitle: 'Registration', + }, +}; diff --git a/src/main/webapp/app/account/register/register.service.ts b/src/main/webapp/app/account/register/register.service.ts new file mode 100644 index 0000000..9a51915 --- /dev/null +++ b/src/main/webapp/app/account/register/register.service.ts @@ -0,0 +1,15 @@ +import { Injectable } from '@angular/core'; +import { HttpClient } from '@angular/common/http'; +import { Observable } from 'rxjs'; + +import { SERVER_API_URL } from 'app/app.constants'; +import { IUser } from 'app/core/user/user.model'; + +@Injectable({ providedIn: 'root' }) +export class RegisterService { + constructor(private http: HttpClient) {} + + save(account: IUser): Observable<{}> { + return this.http.post(SERVER_API_URL + 'api/register', account); + } +} diff --git a/src/main/webapp/app/account/sessions/session.model.ts b/src/main/webapp/app/account/sessions/session.model.ts new file mode 100644 index 0000000..619cdbd --- /dev/null +++ b/src/main/webapp/app/account/sessions/session.model.ts @@ -0,0 +1,3 @@ +export class Session { + constructor(public series: string, public tokenDate: Date, public ipAddress: string, public userAgent: string) {} +} diff --git a/src/main/webapp/app/account/sessions/sessions.component.html b/src/main/webapp/app/account/sessions/sessions.component.html new file mode 100644 index 0000000..101c9be --- /dev/null +++ b/src/main/webapp/app/account/sessions/sessions.component.html @@ -0,0 +1,38 @@ +
+

Active sessions for [{{ account.login }}]

+ +
+ Session invalidated! +
+ +
+ An error has occurred! The session could not be invalidated. +
+ +
+ + + + + + + + + + + + + + + + + +
IP AddressUser agentDate
{{ session.ipAddress }}{{ session.userAgent }}{{ session.tokenDate | date:'longDate' }} + +
+
+
diff --git a/src/main/webapp/app/account/sessions/sessions.component.ts b/src/main/webapp/app/account/sessions/sessions.component.ts new file mode 100644 index 0000000..1c9459b --- /dev/null +++ b/src/main/webapp/app/account/sessions/sessions.component.ts @@ -0,0 +1,38 @@ +import { Component, OnInit } from '@angular/core'; + +import { AccountService } from 'app/core/auth/account.service'; +import { Session } from './session.model'; +import { SessionsService } from './sessions.service'; +import { Account } from 'app/core/user/account.model'; + +@Component({ + selector: 'jhi-sessions', + templateUrl: './sessions.component.html', +}) +export class SessionsComponent implements OnInit { + account: Account | null = null; + error = false; + success = false; + sessions: Session[] = []; + + constructor(private sessionsService: SessionsService, private accountService: AccountService) {} + + ngOnInit(): void { + this.sessionsService.findAll().subscribe(sessions => (this.sessions = sessions)); + + this.accountService.identity().subscribe(account => (this.account = account)); + } + + invalidate(series: string): void { + this.error = false; + this.success = false; + + this.sessionsService.delete(encodeURIComponent(series)).subscribe( + () => { + this.success = true; + this.sessionsService.findAll().subscribe(sessions => (this.sessions = sessions)); + }, + () => (this.error = true) + ); + } +} diff --git a/src/main/webapp/app/account/sessions/sessions.route.ts b/src/main/webapp/app/account/sessions/sessions.route.ts new file mode 100644 index 0000000..994a9f5 --- /dev/null +++ b/src/main/webapp/app/account/sessions/sessions.route.ts @@ -0,0 +1,15 @@ +import { Route } from '@angular/router'; + +import { Authority } from 'app/shared/constants/authority.constants'; +import { UserRouteAccessService } from 'app/core/auth/user-route-access-service'; +import { SessionsComponent } from './sessions.component'; + +export const sessionsRoute: Route = { + path: 'sessions', + component: SessionsComponent, + data: { + authorities: [Authority.USER], + pageTitle: 'Sessions', + }, + canActivate: [UserRouteAccessService], +}; diff --git a/src/main/webapp/app/account/sessions/sessions.service.ts b/src/main/webapp/app/account/sessions/sessions.service.ts new file mode 100644 index 0000000..426a91f --- /dev/null +++ b/src/main/webapp/app/account/sessions/sessions.service.ts @@ -0,0 +1,21 @@ +import { Injectable } from '@angular/core'; +import { HttpClient } from '@angular/common/http'; +import { Observable } from 'rxjs'; + +import { SERVER_API_URL } from 'app/app.constants'; +import { Session } from './session.model'; + +@Injectable({ providedIn: 'root' }) +export class SessionsService { + public resourceUrl = SERVER_API_URL + 'api/account/sessions/'; + + constructor(private http: HttpClient) {} + + findAll(): Observable { + return this.http.get(this.resourceUrl); + } + + delete(series: string): Observable<{}> { + return this.http.delete(`${this.resourceUrl}${series}`); + } +} diff --git a/src/main/webapp/app/account/settings/settings.component.html b/src/main/webapp/app/account/settings/settings.component.html new file mode 100644 index 0000000..2b2cdf6 --- /dev/null +++ b/src/main/webapp/app/account/settings/settings.component.html @@ -0,0 +1,91 @@ +
+
+
+

User settings for [{{ account.login }}]

+ +
+ Settings saved! +
+ + + +
+
+ + + +
+ + Your first name is required. + + + + Your first name is required to be at least 1 character. + + + + Your first name cannot be longer than 50 characters. + +
+
+ +
+ + + +
+ + Your last name is required. + + + + Your last name is required to be at least 1 character. + + + + Your last name cannot be longer than 50 characters. + +
+
+ +
+ + + +
+ + Your email is required. + + + + Your email is invalid. + + + + Your email is required to be at least 5 characters. + + + + Your email cannot be longer than 100 characters. + +
+
+ + +
+
+
+
diff --git a/src/main/webapp/app/account/settings/settings.component.ts b/src/main/webapp/app/account/settings/settings.component.ts new file mode 100644 index 0000000..e562e18 --- /dev/null +++ b/src/main/webapp/app/account/settings/settings.component.ts @@ -0,0 +1,49 @@ +import { Component, OnInit } from '@angular/core'; +import { FormBuilder, Validators } from '@angular/forms'; + +import { AccountService } from 'app/core/auth/account.service'; +import { Account } from 'app/core/user/account.model'; + +@Component({ + selector: 'jhi-settings', + templateUrl: './settings.component.html', +}) +export class SettingsComponent implements OnInit { + account!: Account; + success = false; + settingsForm = this.fb.group({ + firstName: [undefined, [Validators.required, Validators.minLength(1), Validators.maxLength(50)]], + lastName: [undefined, [Validators.required, Validators.minLength(1), Validators.maxLength(50)]], + email: [undefined, [Validators.required, Validators.minLength(5), Validators.maxLength(254), Validators.email]], + }); + + constructor(private accountService: AccountService, private fb: FormBuilder) {} + + ngOnInit(): void { + this.accountService.identity().subscribe(account => { + if (account) { + this.settingsForm.patchValue({ + firstName: account.firstName, + lastName: account.lastName, + email: account.email, + }); + + this.account = account; + } + }); + } + + save(): void { + this.success = false; + + this.account.firstName = this.settingsForm.get('firstName')!.value; + this.account.lastName = this.settingsForm.get('lastName')!.value; + this.account.email = this.settingsForm.get('email')!.value; + + this.accountService.save(this.account).subscribe(() => { + this.success = true; + + this.accountService.authenticate(this.account); + }); + } +} diff --git a/src/main/webapp/app/account/settings/settings.route.ts b/src/main/webapp/app/account/settings/settings.route.ts new file mode 100644 index 0000000..e16b22e --- /dev/null +++ b/src/main/webapp/app/account/settings/settings.route.ts @@ -0,0 +1,15 @@ +import { Route } from '@angular/router'; + +import { UserRouteAccessService } from 'app/core/auth/user-route-access-service'; +import { SettingsComponent } from './settings.component'; +import { Authority } from 'app/shared/constants/authority.constants'; + +export const settingsRoute: Route = { + path: 'settings', + component: SettingsComponent, + data: { + authorities: [Authority.USER], + pageTitle: 'Settings', + }, + canActivate: [UserRouteAccessService], +}; diff --git a/src/main/webapp/app/admin/admin-routing.module.ts b/src/main/webapp/app/admin/admin-routing.module.ts new file mode 100644 index 0000000..6e5529c --- /dev/null +++ b/src/main/webapp/app/admin/admin-routing.module.ts @@ -0,0 +1,44 @@ +import { NgModule } from '@angular/core'; +import { RouterModule } from '@angular/router'; +/* jhipster-needle-add-admin-module-import - JHipster will add admin modules imports here */ + +@NgModule({ + imports: [ + /* jhipster-needle-add-admin-module - JHipster will add admin modules here */ + RouterModule.forChild([ + { + path: 'user-management', + loadChildren: () => import('./user-management/user-management.module').then(m => m.UserManagementModule), + data: { + pageTitle: 'Users', + }, + }, + { + path: 'audits', + loadChildren: () => import('./audits/audits.module').then(m => m.AuditsModule), + }, + { + path: 'configuration', + loadChildren: () => import('./configuration/configuration.module').then(m => m.ConfigurationModule), + }, + { + path: 'docs', + loadChildren: () => import('./docs/docs.module').then(m => m.DocsModule), + }, + { + path: 'health', + loadChildren: () => import('./health/health.module').then(m => m.HealthModule), + }, + { + path: 'logs', + loadChildren: () => import('./logs/logs.module').then(m => m.LogsModule), + }, + { + path: 'metrics', + loadChildren: () => import('./metrics/metrics.module').then(m => m.MetricsModule), + }, + /* jhipster-needle-add-admin-route - JHipster will add admin routes here */ + ]), + ], +}) +export class AdminRoutingModule {} diff --git a/src/main/webapp/app/admin/audits/audit-data.model.ts b/src/main/webapp/app/admin/audits/audit-data.model.ts new file mode 100644 index 0000000..7d5bef4 --- /dev/null +++ b/src/main/webapp/app/admin/audits/audit-data.model.ts @@ -0,0 +1,3 @@ +export class AuditData { + constructor(public remoteAddress?: string, public sessionId?: string, public message?: string) {} +} diff --git a/src/main/webapp/app/admin/audits/audit.model.ts b/src/main/webapp/app/admin/audits/audit.model.ts new file mode 100644 index 0000000..5c64107 --- /dev/null +++ b/src/main/webapp/app/admin/audits/audit.model.ts @@ -0,0 +1,5 @@ +import { AuditData } from './audit-data.model'; + +export class Audit { + constructor(public data: AuditData, public principal: string, public timestamp: string, public type: string) {} +} diff --git a/src/main/webapp/app/admin/audits/audits.component.html b/src/main/webapp/app/admin/audits/audits.component.html new file mode 100644 index 0000000..e2340bd --- /dev/null +++ b/src/main/webapp/app/admin/audits/audits.component.html @@ -0,0 +1,67 @@ +
+

Audits

+ + + +
+
+

Filter by date

+ +
+
+ from +
+ + +
+ To +
+ +
+
+
+ +
+ No audit found +
+ +
+ + + + + + + + + + + + + + + + + + + + + + + +
Date User State Extra data
DateUserStateExtra data
{{ audit.timestamp | date:'medium' }}{{ audit.principal }}{{ audit.type }} + {{ audit.data.message }} + Remote Address {{ audit.data.remoteAddress }} +
+
+ +
+
+ +
+ +
+ +
+
+
diff --git a/src/main/webapp/app/admin/audits/audits.component.ts b/src/main/webapp/app/admin/audits/audits.component.ts new file mode 100644 index 0000000..4d6fd90 --- /dev/null +++ b/src/main/webapp/app/admin/audits/audits.component.ts @@ -0,0 +1,115 @@ +import { Component, OnInit } from '@angular/core'; +import { HttpResponse, HttpHeaders } from '@angular/common/http'; +import { DatePipe } from '@angular/common'; +import { ActivatedRoute, ParamMap, Router, Data } from '@angular/router'; +import { combineLatest } from 'rxjs'; + +import { ITEMS_PER_PAGE } from 'app/shared/constants/pagination.constants'; +import { Audit } from './audit.model'; +import { AuditsService } from './audits.service'; + +@Component({ + selector: 'jhi-audit', + templateUrl: './audits.component.html', +}) +export class AuditsComponent implements OnInit { + audits?: Audit[]; + fromDate = ''; + itemsPerPage = ITEMS_PER_PAGE; + page!: number; + predicate!: string; + ascending!: boolean; + toDate = ''; + totalItems = 0; + + private dateFormat = 'yyyy-MM-dd'; + + constructor( + private auditsService: AuditsService, + private activatedRoute: ActivatedRoute, + private datePipe: DatePipe, + private router: Router + ) {} + + ngOnInit(): void { + this.toDate = this.today(); + this.fromDate = this.previousMonth(); + this.handleNavigation(); + } + + canLoad(): boolean { + return this.fromDate !== '' && this.toDate !== ''; + } + + transition(): void { + if (this.canLoad()) { + this.router.navigate(['/admin/audits'], { + queryParams: { + page: this.page, + sort: this.predicate + ',' + (this.ascending ? 'asc' : 'desc'), + from: this.fromDate, + to: this.toDate, + }, + }); + } + } + + private previousMonth(): string { + let date = new Date(); + if (date.getMonth() === 0) { + date = new Date(date.getFullYear() - 1, 11, date.getDate()); + } else { + date = new Date(date.getFullYear(), date.getMonth() - 1, date.getDate()); + } + return this.datePipe.transform(date, this.dateFormat)!; + } + + private today(): string { + // Today + 1 day - needed if the current day must be included + const date = new Date(); + date.setDate(date.getDate() + 1); + return this.datePipe.transform(date, this.dateFormat)!; + } + + private handleNavigation(): void { + combineLatest(this.activatedRoute.data, this.activatedRoute.queryParamMap, (data: Data, params: ParamMap) => { + const page = params.get('page'); + this.page = page !== null ? +page : 1; + const sort = (params.get('sort') ?? data['defaultSort']).split(','); + this.predicate = sort[0]; + this.ascending = sort[1] === 'asc'; + if (params.get('from')) { + this.fromDate = this.datePipe.transform(params.get('from'), this.dateFormat)!; + } + if (params.get('to')) { + this.toDate = this.datePipe.transform(params.get('to'), this.dateFormat)!; + } + this.loadData(); + }).subscribe(); + } + + private loadData(): void { + this.auditsService + .query({ + page: this.page - 1, + size: this.itemsPerPage, + sort: this.sort(), + fromDate: this.fromDate, + toDate: this.toDate, + }) + .subscribe((res: HttpResponse) => this.onSuccess(res.body, res.headers)); + } + + private sort(): string[] { + const result = [this.predicate + ',' + (this.ascending ? 'asc' : 'desc')]; + if (this.predicate !== 'id') { + result.push('id'); + } + return result; + } + + private onSuccess(audits: Audit[] | null, headers: HttpHeaders): void { + this.totalItems = Number(headers.get('X-Total-Count')); + this.audits = audits || []; + } +} diff --git a/src/main/webapp/app/admin/audits/audits.module.ts b/src/main/webapp/app/admin/audits/audits.module.ts new file mode 100644 index 0000000..cecc32d --- /dev/null +++ b/src/main/webapp/app/admin/audits/audits.module.ts @@ -0,0 +1,13 @@ +import { NgModule } from '@angular/core'; +import { RouterModule } from '@angular/router'; +import { EpasmedSharedModule } from 'app/shared/shared.module'; + +import { AuditsComponent } from './audits.component'; + +import { auditsRoute } from './audits.route'; + +@NgModule({ + imports: [EpasmedSharedModule, RouterModule.forChild([auditsRoute])], + declarations: [AuditsComponent], +}) +export class AuditsModule {} diff --git a/src/main/webapp/app/admin/audits/audits.route.ts b/src/main/webapp/app/admin/audits/audits.route.ts new file mode 100644 index 0000000..b586c4c --- /dev/null +++ b/src/main/webapp/app/admin/audits/audits.route.ts @@ -0,0 +1,12 @@ +import { Route } from '@angular/router'; + +import { AuditsComponent } from './audits.component'; + +export const auditsRoute: Route = { + path: '', + component: AuditsComponent, + data: { + pageTitle: 'Audits', + defaultSort: 'auditEventDate,desc', + }, +}; diff --git a/src/main/webapp/app/admin/audits/audits.service.ts b/src/main/webapp/app/admin/audits/audits.service.ts new file mode 100644 index 0000000..c06c5f0 --- /dev/null +++ b/src/main/webapp/app/admin/audits/audits.service.ts @@ -0,0 +1,28 @@ +import { Injectable } from '@angular/core'; +import { HttpClient, HttpParams, HttpResponse } from '@angular/common/http'; +import { Observable } from 'rxjs'; + +import { createRequestOption, Pagination } from 'app/shared/util/request-util'; +import { SERVER_API_URL } from 'app/app.constants'; +import { Audit } from './audit.model'; + +export interface AuditsQuery extends Pagination { + fromDate: string; + toDate: string; +} + +@Injectable({ providedIn: 'root' }) +export class AuditsService { + constructor(private http: HttpClient) {} + + query(req: AuditsQuery): Observable> { + const params: HttpParams = createRequestOption(req); + + const requestURL = SERVER_API_URL + 'management/audits'; + + return this.http.get(requestURL, { + params, + observe: 'response', + }); + } +} diff --git a/src/main/webapp/app/admin/configuration/configuration.component.html b/src/main/webapp/app/admin/configuration/configuration.component.html new file mode 100644 index 0000000..a90142c --- /dev/null +++ b/src/main/webapp/app/admin/configuration/configuration.component.html @@ -0,0 +1,50 @@ +
+

Configuration

+ + Filter (by prefix) + +

Spring configuration

+ + + + + + + + + + + + + + +
Prefix Properties
{{ bean.prefix }} +
+
{{ property.key }}
+
+ {{ property.value | json }} +
+
+
+ +
+

{{ propertySource.name }}

+ + + + + + + + + + + + + + +
PropertyValue
{{ property.key }} + {{ property.value.value }} +
+
+
diff --git a/src/main/webapp/app/admin/configuration/configuration.component.ts b/src/main/webapp/app/admin/configuration/configuration.component.ts new file mode 100644 index 0000000..0830cc4 --- /dev/null +++ b/src/main/webapp/app/admin/configuration/configuration.component.ts @@ -0,0 +1,32 @@ +import { Component, OnInit } from '@angular/core'; + +import { ConfigurationService, Bean, PropertySource } from './configuration.service'; + +@Component({ + selector: 'jhi-configuration', + templateUrl: './configuration.component.html', +}) +export class ConfigurationComponent implements OnInit { + allBeans!: Bean[]; + beans: Bean[] = []; + beansFilter = ''; + beansAscending = true; + propertySources: PropertySource[] = []; + + constructor(private configurationService: ConfigurationService) {} + + ngOnInit(): void { + this.configurationService.getBeans().subscribe(beans => { + this.allBeans = beans; + this.filterAndSortBeans(); + }); + + this.configurationService.getPropertySources().subscribe(propertySources => (this.propertySources = propertySources)); + } + + filterAndSortBeans(): void { + this.beans = this.allBeans + .filter(bean => !this.beansFilter || bean.prefix.toLowerCase().includes(this.beansFilter.toLowerCase())) + .sort((a, b) => (a.prefix < b.prefix ? (this.beansAscending ? -1 : 1) : this.beansAscending ? 1 : -1)); + } +} diff --git a/src/main/webapp/app/admin/configuration/configuration.module.ts b/src/main/webapp/app/admin/configuration/configuration.module.ts new file mode 100644 index 0000000..a211fae --- /dev/null +++ b/src/main/webapp/app/admin/configuration/configuration.module.ts @@ -0,0 +1,13 @@ +import { NgModule } from '@angular/core'; +import { RouterModule } from '@angular/router'; +import { EpasmedSharedModule } from 'app/shared/shared.module'; + +import { ConfigurationComponent } from './configuration.component'; + +import { configurationRoute } from './configuration.route'; + +@NgModule({ + imports: [EpasmedSharedModule, RouterModule.forChild([configurationRoute])], + declarations: [ConfigurationComponent], +}) +export class ConfigurationModule {} diff --git a/src/main/webapp/app/admin/configuration/configuration.route.ts b/src/main/webapp/app/admin/configuration/configuration.route.ts new file mode 100644 index 0000000..f3090ec --- /dev/null +++ b/src/main/webapp/app/admin/configuration/configuration.route.ts @@ -0,0 +1,11 @@ +import { Route } from '@angular/router'; + +import { ConfigurationComponent } from './configuration.component'; + +export const configurationRoute: Route = { + path: '', + component: ConfigurationComponent, + data: { + pageTitle: 'Configuration', + }, +}; diff --git a/src/main/webapp/app/admin/configuration/configuration.service.ts b/src/main/webapp/app/admin/configuration/configuration.service.ts new file mode 100644 index 0000000..69ea43a --- /dev/null +++ b/src/main/webapp/app/admin/configuration/configuration.service.ts @@ -0,0 +1,68 @@ +import { Injectable } from '@angular/core'; +import { HttpClient } from '@angular/common/http'; +import { Observable } from 'rxjs'; +import { map } from 'rxjs/operators'; + +import { SERVER_API_URL } from 'app/app.constants'; + +export interface ConfigProps { + contexts: Contexts; +} + +export interface Contexts { + [key: string]: Context; +} + +export interface Context { + beans: Beans; + parentId?: any; +} + +export interface Beans { + [key: string]: Bean; +} + +export interface Bean { + prefix: string; + properties: any; +} + +export interface Env { + activeProfiles?: string[]; + propertySources: PropertySource[]; +} + +export interface PropertySource { + name: string; + properties: Properties; +} + +export interface Properties { + [key: string]: Property; +} + +export interface Property { + value: string; + origin?: string; +} + +@Injectable({ providedIn: 'root' }) +export class ConfigurationService { + constructor(private http: HttpClient) {} + + getBeans(): Observable { + return this.http.get(SERVER_API_URL + 'management/configprops').pipe( + map(configProps => + Object.values( + Object.values(configProps.contexts) + .map(context => context.beans) + .reduce((allBeans: Beans, contextBeans: Beans) => ({ ...allBeans, ...contextBeans })) + ) + ) + ); + } + + getPropertySources(): Observable { + return this.http.get(SERVER_API_URL + 'management/env').pipe(map(env => env.propertySources)); + } +} diff --git a/src/main/webapp/app/admin/docs/docs.component.html b/src/main/webapp/app/admin/docs/docs.component.html new file mode 100644 index 0000000..9d1600b --- /dev/null +++ b/src/main/webapp/app/admin/docs/docs.component.html @@ -0,0 +1,2 @@ + diff --git a/src/main/webapp/app/admin/docs/docs.component.ts b/src/main/webapp/app/admin/docs/docs.component.ts new file mode 100644 index 0000000..0678530 --- /dev/null +++ b/src/main/webapp/app/admin/docs/docs.component.ts @@ -0,0 +1,8 @@ +import { Component } from '@angular/core'; + +@Component({ + selector: 'jhi-docs', + templateUrl: './docs.component.html', + styleUrls: ['docs.scss'], +}) +export class DocsComponent {} diff --git a/src/main/webapp/app/admin/docs/docs.module.ts b/src/main/webapp/app/admin/docs/docs.module.ts new file mode 100644 index 0000000..38864da --- /dev/null +++ b/src/main/webapp/app/admin/docs/docs.module.ts @@ -0,0 +1,13 @@ +import { NgModule } from '@angular/core'; +import { RouterModule } from '@angular/router'; +import { EpasmedSharedModule } from 'app/shared/shared.module'; + +import { DocsComponent } from './docs.component'; + +import { docsRoute } from './docs.route'; + +@NgModule({ + imports: [EpasmedSharedModule, RouterModule.forChild([docsRoute])], + declarations: [DocsComponent], +}) +export class DocsModule {} diff --git a/src/main/webapp/app/admin/docs/docs.route.ts b/src/main/webapp/app/admin/docs/docs.route.ts new file mode 100644 index 0000000..f94e488 --- /dev/null +++ b/src/main/webapp/app/admin/docs/docs.route.ts @@ -0,0 +1,11 @@ +import { Route } from '@angular/router'; + +import { DocsComponent } from './docs.component'; + +export const docsRoute: Route = { + path: '', + component: DocsComponent, + data: { + pageTitle: 'API', + }, +}; diff --git a/src/main/webapp/app/admin/docs/docs.scss b/src/main/webapp/app/admin/docs/docs.scss new file mode 100644 index 0000000..541c3d7 --- /dev/null +++ b/src/main/webapp/app/admin/docs/docs.scss @@ -0,0 +1,6 @@ +@import '~bootstrap/scss/functions'; +@import '~bootstrap/scss/variables'; + +iframe { + background: white; +} diff --git a/src/main/webapp/app/admin/health/health-modal.component.html b/src/main/webapp/app/admin/health/health-modal.component.html new file mode 100644 index 0000000..dc4425e --- /dev/null +++ b/src/main/webapp/app/admin/health/health-modal.component.html @@ -0,0 +1,36 @@ + + + + + diff --git a/src/main/webapp/app/admin/health/health-modal.component.ts b/src/main/webapp/app/admin/health/health-modal.component.ts new file mode 100644 index 0000000..ce4dd7e --- /dev/null +++ b/src/main/webapp/app/admin/health/health-modal.component.ts @@ -0,0 +1,37 @@ +import { Component } from '@angular/core'; +import { NgbActiveModal } from '@ng-bootstrap/ng-bootstrap'; + +import { HealthKey, HealthDetails } from './health.service'; + +@Component({ + selector: 'jhi-health-modal', + templateUrl: './health-modal.component.html', +}) +export class HealthModalComponent { + health?: { key: HealthKey; value: HealthDetails }; + + constructor(public activeModal: NgbActiveModal) {} + + readableValue(value: any): string { + if (this.health && this.health.key === 'diskSpace') { + // Should display storage space in an human readable unit + const val = value / 1073741824; + if (val > 1) { + // Value + return val.toFixed(2) + ' GB'; + } else { + return (value / 1048576).toFixed(2) + ' MB'; + } + } + + if (typeof value === 'object') { + return JSON.stringify(value); + } else { + return value.toString(); + } + } + + dismiss(): void { + this.activeModal.dismiss(); + } +} diff --git a/src/main/webapp/app/admin/health/health.component.html b/src/main/webapp/app/admin/health/health.component.html new file mode 100644 index 0000000..f7c7153 --- /dev/null +++ b/src/main/webapp/app/admin/health/health.component.html @@ -0,0 +1,38 @@ +
+

+ Health Checks + + +

+ +
+ + + + + + + + + + + + + + + +
Service NameStatusDetails
+ {{ componentHealth.key }} + + + {{ componentHealth.value.status }} + + + + + +
+
+
diff --git a/src/main/webapp/app/admin/health/health.component.ts b/src/main/webapp/app/admin/health/health.component.ts new file mode 100644 index 0000000..0332feb --- /dev/null +++ b/src/main/webapp/app/admin/health/health.component.ts @@ -0,0 +1,44 @@ +import { Component, OnInit } from '@angular/core'; +import { HttpErrorResponse } from '@angular/common/http'; +import { NgbModal } from '@ng-bootstrap/ng-bootstrap'; + +import { HealthService, HealthStatus, Health, HealthKey, HealthDetails } from './health.service'; +import { HealthModalComponent } from './health-modal.component'; + +@Component({ + selector: 'jhi-health', + templateUrl: './health.component.html', +}) +export class HealthComponent implements OnInit { + health?: Health; + + constructor(private modalService: NgbModal, private healthService: HealthService) {} + + ngOnInit(): void { + this.refresh(); + } + + getBadgeClass(statusState: HealthStatus): string { + if (statusState === 'UP') { + return 'badge-success'; + } else { + return 'badge-danger'; + } + } + + refresh(): void { + this.healthService.checkHealth().subscribe( + health => (this.health = health), + (error: HttpErrorResponse) => { + if (error.status === 503) { + this.health = error.error; + } + } + ); + } + + showHealth(health: { key: HealthKey; value: HealthDetails }): void { + const modalRef = this.modalService.open(HealthModalComponent); + modalRef.componentInstance.health = health; + } +} diff --git a/src/main/webapp/app/admin/health/health.module.ts b/src/main/webapp/app/admin/health/health.module.ts new file mode 100644 index 0000000..38d5d3a --- /dev/null +++ b/src/main/webapp/app/admin/health/health.module.ts @@ -0,0 +1,15 @@ +import { NgModule } from '@angular/core'; +import { RouterModule } from '@angular/router'; +import { EpasmedSharedModule } from 'app/shared/shared.module'; + +import { HealthComponent } from './health.component'; +import { HealthModalComponent } from './health-modal.component'; + +import { healthRoute } from './health.route'; + +@NgModule({ + imports: [EpasmedSharedModule, RouterModule.forChild([healthRoute])], + declarations: [HealthComponent, HealthModalComponent], + entryComponents: [HealthModalComponent], +}) +export class HealthModule {} diff --git a/src/main/webapp/app/admin/health/health.route.ts b/src/main/webapp/app/admin/health/health.route.ts new file mode 100644 index 0000000..3a8fe57 --- /dev/null +++ b/src/main/webapp/app/admin/health/health.route.ts @@ -0,0 +1,11 @@ +import { Route } from '@angular/router'; + +import { HealthComponent } from './health.component'; + +export const healthRoute: Route = { + path: '', + component: HealthComponent, + data: { + pageTitle: 'Health Checks', + }, +}; diff --git a/src/main/webapp/app/admin/health/health.service.ts b/src/main/webapp/app/admin/health/health.service.ts new file mode 100644 index 0000000..25005f6 --- /dev/null +++ b/src/main/webapp/app/admin/health/health.service.ts @@ -0,0 +1,30 @@ +import { Injectable } from '@angular/core'; +import { HttpClient } from '@angular/common/http'; +import { Observable } from 'rxjs'; + +import { SERVER_API_URL } from 'app/app.constants'; + +export type HealthStatus = 'UP' | 'DOWN' | 'UNKNOWN' | 'OUT_OF_SERVICE'; + +export type HealthKey = 'diskSpace' | 'mail' | 'ping' | 'db'; + +export interface Health { + status: HealthStatus; + components: { + [key in HealthKey]?: HealthDetails; + }; +} + +export interface HealthDetails { + status: HealthStatus; + details: any; +} + +@Injectable({ providedIn: 'root' }) +export class HealthService { + constructor(private http: HttpClient) {} + + checkHealth(): Observable { + return this.http.get(SERVER_API_URL + 'management/health'); + } +} diff --git a/src/main/webapp/app/admin/logs/log.model.ts b/src/main/webapp/app/admin/logs/log.model.ts new file mode 100644 index 0000000..83f2e15 --- /dev/null +++ b/src/main/webapp/app/admin/logs/log.model.ts @@ -0,0 +1,15 @@ +export type Level = 'TRACE' | 'DEBUG' | 'INFO' | 'WARN' | 'ERROR' | 'OFF'; + +export interface Logger { + configuredLevel: Level | null; + effectiveLevel: Level; +} + +export interface LoggersResponse { + levels: Level[]; + loggers: { [key: string]: Logger }; +} + +export class Log { + constructor(public name: string, public level: Level) {} +} diff --git a/src/main/webapp/app/admin/logs/logs.component.html b/src/main/webapp/app/admin/logs/logs.component.html new file mode 100644 index 0000000..316f6fe --- /dev/null +++ b/src/main/webapp/app/admin/logs/logs.component.html @@ -0,0 +1,28 @@ +
+

Logs

+ +

There are {{ loggers.length }} loggers.

+ + Filter + + + + + + + + + + + + + +
NameLevel
{{ logger.name | slice:0:140 }} + + + + + + +
+
diff --git a/src/main/webapp/app/admin/logs/logs.component.ts b/src/main/webapp/app/admin/logs/logs.component.ts new file mode 100644 index 0000000..7c50ff0 --- /dev/null +++ b/src/main/webapp/app/admin/logs/logs.component.ts @@ -0,0 +1,34 @@ +import { Component, OnInit } from '@angular/core'; + +import { Log, LoggersResponse, Logger, Level } from './log.model'; +import { LogsService } from './logs.service'; + +@Component({ + selector: 'jhi-logs', + templateUrl: './logs.component.html', +}) +export class LogsComponent implements OnInit { + loggers?: Log[]; + filter = ''; + orderProp = 'name'; + reverse = false; + + constructor(private logsService: LogsService) {} + + ngOnInit(): void { + this.findAndExtractLoggers(); + } + + changeLevel(name: string, level: Level): void { + this.logsService.changeLevel(name, level).subscribe(() => this.findAndExtractLoggers()); + } + + private findAndExtractLoggers(): void { + this.logsService + .findAll() + .subscribe( + (response: LoggersResponse) => + (this.loggers = Object.entries(response.loggers).map((logger: [string, Logger]) => new Log(logger[0], logger[1].effectiveLevel))) + ); + } +} diff --git a/src/main/webapp/app/admin/logs/logs.module.ts b/src/main/webapp/app/admin/logs/logs.module.ts new file mode 100644 index 0000000..02d414c --- /dev/null +++ b/src/main/webapp/app/admin/logs/logs.module.ts @@ -0,0 +1,13 @@ +import { NgModule } from '@angular/core'; +import { RouterModule } from '@angular/router'; +import { EpasmedSharedModule } from 'app/shared/shared.module'; + +import { LogsComponent } from './logs.component'; + +import { logsRoute } from './logs.route'; + +@NgModule({ + imports: [EpasmedSharedModule, RouterModule.forChild([logsRoute])], + declarations: [LogsComponent], +}) +export class LogsModule {} diff --git a/src/main/webapp/app/admin/logs/logs.route.ts b/src/main/webapp/app/admin/logs/logs.route.ts new file mode 100644 index 0000000..5a160eb --- /dev/null +++ b/src/main/webapp/app/admin/logs/logs.route.ts @@ -0,0 +1,11 @@ +import { Route } from '@angular/router'; + +import { LogsComponent } from './logs.component'; + +export const logsRoute: Route = { + path: '', + component: LogsComponent, + data: { + pageTitle: 'Logs', + }, +}; diff --git a/src/main/webapp/app/admin/logs/logs.service.ts b/src/main/webapp/app/admin/logs/logs.service.ts new file mode 100644 index 0000000..dc9956d --- /dev/null +++ b/src/main/webapp/app/admin/logs/logs.service.ts @@ -0,0 +1,19 @@ +import { Injectable } from '@angular/core'; +import { HttpClient } from '@angular/common/http'; +import { Observable } from 'rxjs'; + +import { SERVER_API_URL } from 'app/app.constants'; +import { LoggersResponse, Level } from './log.model'; + +@Injectable({ providedIn: 'root' }) +export class LogsService { + constructor(private http: HttpClient) {} + + changeLevel(name: string, configuredLevel: Level): Observable<{}> { + return this.http.post(SERVER_API_URL + 'management/loggers/' + name, { configuredLevel }); + } + + findAll(): Observable { + return this.http.get(SERVER_API_URL + 'management/loggers'); + } +} diff --git a/src/main/webapp/app/admin/metrics/metrics.component.html b/src/main/webapp/app/admin/metrics/metrics.component.html new file mode 100644 index 0000000..84bb2b7 --- /dev/null +++ b/src/main/webapp/app/admin/metrics/metrics.component.html @@ -0,0 +1,65 @@ +
+

+ Application Metrics + + +

+ +

JVM Metrics

+ +
+ + + + + + + + +
+ +
+

Garbage collector statistics

+ + + +
+ +
Updating...
+ + + + + + + + + + + + +
diff --git a/src/main/webapp/app/admin/metrics/metrics.component.ts b/src/main/webapp/app/admin/metrics/metrics.component.ts new file mode 100644 index 0000000..c2b52db --- /dev/null +++ b/src/main/webapp/app/admin/metrics/metrics.component.ts @@ -0,0 +1,47 @@ +import { Component, OnInit, ChangeDetectionStrategy, ChangeDetectorRef } from '@angular/core'; +import { flatMap } from 'rxjs/operators'; + +import { MetricsService, Metrics, MetricsKey, ThreadDump, Thread } from './metrics.service'; + +@Component({ + selector: 'jhi-metrics', + templateUrl: './metrics.component.html', + changeDetection: ChangeDetectionStrategy.OnPush, +}) +export class MetricsComponent implements OnInit { + metrics?: Metrics; + threads?: Thread[]; + updatingMetrics = true; + + constructor(private metricsService: MetricsService, private changeDetector: ChangeDetectorRef) {} + + ngOnInit(): void { + this.refresh(); + } + + refresh(): void { + this.updatingMetrics = true; + this.metricsService + .getMetrics() + .pipe( + flatMap( + () => this.metricsService.threadDump(), + (metrics: Metrics, threadDump: ThreadDump) => { + this.metrics = metrics; + this.threads = threadDump.threads; + this.updatingMetrics = false; + this.changeDetector.detectChanges(); + } + ) + ) + .subscribe(); + } + + metricsKeyExists(key: MetricsKey): boolean { + return this.metrics && this.metrics[key]; + } + + metricsKeyExistsAndObjectNotEmpty(key: MetricsKey): boolean { + return this.metrics && this.metrics[key] && JSON.stringify(this.metrics[key]) !== '{}'; + } +} diff --git a/src/main/webapp/app/admin/metrics/metrics.module.ts b/src/main/webapp/app/admin/metrics/metrics.module.ts new file mode 100644 index 0000000..704066b --- /dev/null +++ b/src/main/webapp/app/admin/metrics/metrics.module.ts @@ -0,0 +1,13 @@ +import { NgModule } from '@angular/core'; +import { RouterModule } from '@angular/router'; +import { EpasmedSharedModule } from 'app/shared/shared.module'; + +import { MetricsComponent } from './metrics.component'; + +import { metricsRoute } from './metrics.route'; + +@NgModule({ + imports: [EpasmedSharedModule, RouterModule.forChild([metricsRoute])], + declarations: [MetricsComponent], +}) +export class MetricsModule {} diff --git a/src/main/webapp/app/admin/metrics/metrics.route.ts b/src/main/webapp/app/admin/metrics/metrics.route.ts new file mode 100644 index 0000000..deee47b --- /dev/null +++ b/src/main/webapp/app/admin/metrics/metrics.route.ts @@ -0,0 +1,11 @@ +import { Route } from '@angular/router'; + +import { MetricsComponent } from './metrics.component'; + +export const metricsRoute: Route = { + path: '', + component: MetricsComponent, + data: { + pageTitle: 'Application Metrics', + }, +}; diff --git a/src/main/webapp/app/admin/metrics/metrics.service.ts b/src/main/webapp/app/admin/metrics/metrics.service.ts new file mode 100644 index 0000000..a8d0c99 --- /dev/null +++ b/src/main/webapp/app/admin/metrics/metrics.service.ts @@ -0,0 +1,23 @@ +import { Injectable } from '@angular/core'; +import { HttpClient } from '@angular/common/http'; +import { Observable } from 'rxjs'; + +import { SERVER_API_URL } from 'app/app.constants'; + +export type MetricsKey = 'jvm' | 'http.server.requests' | 'cache' | 'services' | 'databases' | 'garbageCollector' | 'processMetrics'; +export type Metrics = { [key in MetricsKey]: any }; +export type Thread = any; +export type ThreadDump = { threads: Thread[] }; + +@Injectable({ providedIn: 'root' }) +export class MetricsService { + constructor(private http: HttpClient) {} + + getMetrics(): Observable { + return this.http.get(SERVER_API_URL + 'management/jhimetrics'); + } + + threadDump(): Observable { + return this.http.get(SERVER_API_URL + 'management/threaddump'); + } +} diff --git a/src/main/webapp/app/admin/user-management/user-management-delete-dialog.component.html b/src/main/webapp/app/admin/user-management/user-management-delete-dialog.component.html new file mode 100644 index 0000000..d6f8213 --- /dev/null +++ b/src/main/webapp/app/admin/user-management/user-management-delete-dialog.component.html @@ -0,0 +1,23 @@ +
+ + + + + +
diff --git a/src/main/webapp/app/admin/user-management/user-management-delete-dialog.component.ts b/src/main/webapp/app/admin/user-management/user-management-delete-dialog.component.ts new file mode 100644 index 0000000..d3d934d --- /dev/null +++ b/src/main/webapp/app/admin/user-management/user-management-delete-dialog.component.ts @@ -0,0 +1,27 @@ +import { Component } from '@angular/core'; +import { NgbActiveModal } from '@ng-bootstrap/ng-bootstrap'; +import { JhiEventManager } from 'ng-jhipster'; + +import { User } from 'app/core/user/user.model'; +import { UserService } from 'app/core/user/user.service'; + +@Component({ + selector: 'jhi-user-mgmt-delete-dialog', + templateUrl: './user-management-delete-dialog.component.html', +}) +export class UserManagementDeleteDialogComponent { + user?: User; + + constructor(private userService: UserService, public activeModal: NgbActiveModal, private eventManager: JhiEventManager) {} + + cancel(): void { + this.activeModal.dismiss(); + } + + confirmDelete(login: string): void { + this.userService.delete(login).subscribe(() => { + this.eventManager.broadcast('userListModification'); + this.activeModal.close(); + }); + } +} diff --git a/src/main/webapp/app/admin/user-management/user-management-detail.component.html b/src/main/webapp/app/admin/user-management/user-management-detail.component.html new file mode 100644 index 0000000..b533b66 --- /dev/null +++ b/src/main/webapp/app/admin/user-management/user-management-detail.component.html @@ -0,0 +1,55 @@ +
+
+
+

+ User [{{ user.login }}] +

+ +
+
Login
+
+ {{ user.login }} + + +
+ +
First Name
+
{{ user.firstName }}
+ +
Last Name
+
{{ user.lastName }}
+ +
Email
+
{{ user.email }}
+ +
Created By
+
{{ user.createdBy }}
+ +
Created Date
+
{{ user.createdDate | date:'dd/MM/yy HH:mm' }}
+ +
Last Modified By
+
{{ user.lastModifiedBy }}
+ +
Last Modified Date
+
{{ user.lastModifiedDate | date:'dd/MM/yy HH:mm' }}
+ +
Profiles
+
+
    +
  • + {{ authority }} +
  • +
+
+
+ + +
+
+
diff --git a/src/main/webapp/app/admin/user-management/user-management-detail.component.ts b/src/main/webapp/app/admin/user-management/user-management-detail.component.ts new file mode 100644 index 0000000..1cb37c6 --- /dev/null +++ b/src/main/webapp/app/admin/user-management/user-management-detail.component.ts @@ -0,0 +1,18 @@ +import { Component, OnInit } from '@angular/core'; +import { ActivatedRoute } from '@angular/router'; + +import { User } from 'app/core/user/user.model'; + +@Component({ + selector: 'jhi-user-mgmt-detail', + templateUrl: './user-management-detail.component.html', +}) +export class UserManagementDetailComponent implements OnInit { + user: User | null = null; + + constructor(private route: ActivatedRoute) {} + + ngOnInit(): void { + this.route.data.subscribe(({ user }) => (this.user = user)); + } +} diff --git a/src/main/webapp/app/admin/user-management/user-management-update.component.html b/src/main/webapp/app/admin/user-management/user-management-update.component.html new file mode 100644 index 0000000..c9bab67 --- /dev/null +++ b/src/main/webapp/app/admin/user-management/user-management-update.component.html @@ -0,0 +1,119 @@ +
+
+
+

+ Create or edit a User +

+ +
+ + +
+ + +
+ +
+ + + +
+ + This field is required. + + + + This field cannot be longer than 50 characters. + + + + This field can only contain letters, digits and e-mail addresses. + +
+
+ +
+ + + +
+ + This field cannot be longer than 50 characters. + +
+
+ +
+ + + +
+ + This field cannot be longer than 50 characters. + +
+
+ +
+ + + +
+ + This field is required. + + + + This field cannot be longer than 100 characters. + + + + This field is required to be at least 5 characters. + + + + Your email is invalid. + +
+
+ +
+ +
+ +
+ + +
+
+ +
+ + + +
+
+
+
diff --git a/src/main/webapp/app/admin/user-management/user-management-update.component.ts b/src/main/webapp/app/admin/user-management/user-management-update.component.ts new file mode 100644 index 0000000..27bd96c --- /dev/null +++ b/src/main/webapp/app/admin/user-management/user-management-update.component.ts @@ -0,0 +1,105 @@ +import { Component, OnInit } from '@angular/core'; +import { FormBuilder, Validators } from '@angular/forms'; +import { ActivatedRoute } from '@angular/router'; + +import { User } from 'app/core/user/user.model'; +import { UserService } from 'app/core/user/user.service'; + +@Component({ + selector: 'jhi-user-mgmt-update', + templateUrl: './user-management-update.component.html', +}) +export class UserManagementUpdateComponent implements OnInit { + user!: User; + authorities: string[] = []; + isSaving = false; + + editForm = this.fb.group({ + id: [], + login: [ + '', + [ + Validators.required, + Validators.minLength(1), + Validators.maxLength(50), + Validators.pattern('^[a-zA-Z0-9!$&*+=?^_`{|}~.-]+@[a-zA-Z0-9-]+(?:\\.[a-zA-Z0-9-]+)*$|^[_.@A-Za-z0-9-]+$'), + ], + ], + firstName: ['', [Validators.maxLength(50)]], + lastName: ['', [Validators.maxLength(50)]], + email: ['', [Validators.minLength(5), Validators.maxLength(254), Validators.email]], + activated: [], + langKey: [], + authorities: [], + }); + + constructor(private userService: UserService, private route: ActivatedRoute, private fb: FormBuilder) {} + + ngOnInit(): void { + this.route.data.subscribe(({ user }) => { + if (user) { + this.user = user; + if (this.user.id === undefined) { + this.user.activated = true; + } + this.updateForm(user); + } + }); + this.userService.authorities().subscribe(authorities => { + this.authorities = authorities; + }); + } + + previousState(): void { + window.history.back(); + } + + save(): void { + this.isSaving = true; + this.updateUser(this.user); + if (this.user.id !== undefined) { + this.userService.update(this.user).subscribe( + () => this.onSaveSuccess(), + () => this.onSaveError() + ); + } else { + this.user.langKey = 'en'; + this.userService.create(this.user).subscribe( + () => this.onSaveSuccess(), + () => this.onSaveError() + ); + } + } + + private updateForm(user: User): void { + this.editForm.patchValue({ + id: user.id, + login: user.login, + firstName: user.firstName, + lastName: user.lastName, + email: user.email, + activated: user.activated, + langKey: user.langKey, + authorities: user.authorities, + }); + } + + private updateUser(user: User): void { + user.login = this.editForm.get(['login'])!.value; + user.firstName = this.editForm.get(['firstName'])!.value; + user.lastName = this.editForm.get(['lastName'])!.value; + user.email = this.editForm.get(['email'])!.value; + user.activated = this.editForm.get(['activated'])!.value; + user.langKey = this.editForm.get(['langKey'])!.value; + user.authorities = this.editForm.get(['authorities'])!.value; + } + + private onSaveSuccess(): void { + this.isSaving = false; + this.previousState(); + } + + private onSaveError(): void { + this.isSaving = false; + } +} diff --git a/src/main/webapp/app/admin/user-management/user-management.component.html b/src/main/webapp/app/admin/user-management/user-management.component.html new file mode 100644 index 0000000..5ed3284 --- /dev/null +++ b/src/main/webapp/app/admin/user-management/user-management.component.html @@ -0,0 +1,85 @@ +
+

+ Users + + +

+ + + + + +
+ + + + + + + + + + + + + + + + + + + + + + + + + + + +
ID Login Email ProfilesCreated Date Last Modified By Last Modified Date
{{ user.id }}{{ user.login }}{{ user.email }} + + + +
+ {{ authority }} +
+
{{ user.createdDate | date:'dd/MM/yy HH:mm' }}{{ user.lastModifiedBy }}{{ user.lastModifiedDate | date:'dd/MM/yy HH:mm' }} +
+ + + + + +
+
+
+ +
+
+ +
+ +
+ +
+
+
diff --git a/src/main/webapp/app/admin/user-management/user-management.component.ts b/src/main/webapp/app/admin/user-management/user-management.component.ts new file mode 100644 index 0000000..de94a91 --- /dev/null +++ b/src/main/webapp/app/admin/user-management/user-management.component.ts @@ -0,0 +1,106 @@ +import { Component, OnInit, OnDestroy } from '@angular/core'; +import { HttpResponse, HttpHeaders } from '@angular/common/http'; +import { NgbModal } from '@ng-bootstrap/ng-bootstrap'; +import { Subscription, combineLatest } from 'rxjs'; +import { ActivatedRoute, ParamMap, Router, Data } from '@angular/router'; +import { JhiEventManager } from 'ng-jhipster'; + +import { ITEMS_PER_PAGE } from 'app/shared/constants/pagination.constants'; +import { AccountService } from 'app/core/auth/account.service'; +import { Account } from 'app/core/user/account.model'; +import { UserService } from 'app/core/user/user.service'; +import { User } from 'app/core/user/user.model'; +import { UserManagementDeleteDialogComponent } from './user-management-delete-dialog.component'; + +@Component({ + selector: 'jhi-user-mgmt', + templateUrl: './user-management.component.html', +}) +export class UserManagementComponent implements OnInit, OnDestroy { + currentAccount: Account | null = null; + users: User[] | null = null; + userListSubscription?: Subscription; + totalItems = 0; + itemsPerPage = ITEMS_PER_PAGE; + page!: number; + predicate!: string; + ascending!: boolean; + + constructor( + private userService: UserService, + private accountService: AccountService, + private activatedRoute: ActivatedRoute, + private router: Router, + private eventManager: JhiEventManager, + private modalService: NgbModal + ) {} + + ngOnInit(): void { + this.accountService.identity().subscribe(account => (this.currentAccount = account)); + this.userListSubscription = this.eventManager.subscribe('userListModification', () => this.loadAll()); + this.handleNavigation(); + } + + ngOnDestroy(): void { + if (this.userListSubscription) { + this.eventManager.destroy(this.userListSubscription); + } + } + + setActive(user: User, isActivated: boolean): void { + this.userService.update({ ...user, activated: isActivated }).subscribe(() => this.loadAll()); + } + + trackIdentity(index: number, item: User): any { + return item.id; + } + + deleteUser(user: User): void { + const modalRef = this.modalService.open(UserManagementDeleteDialogComponent, { size: 'lg', backdrop: 'static' }); + modalRef.componentInstance.user = user; + } + + transition(): void { + this.router.navigate(['./'], { + relativeTo: this.activatedRoute.parent, + queryParams: { + page: this.page, + sort: this.predicate + ',' + (this.ascending ? 'asc' : 'desc'), + }, + }); + } + + private handleNavigation(): void { + combineLatest(this.activatedRoute.data, this.activatedRoute.queryParamMap, (data: Data, params: ParamMap) => { + const page = params.get('page'); + this.page = page !== null ? +page : 1; + const sort = (params.get('sort') ?? data['defaultSort']).split(','); + this.predicate = sort[0]; + this.ascending = sort[1] === 'asc'; + this.loadAll(); + }).subscribe(); + } + + private loadAll(): void { + this.userService + .query({ + page: this.page - 1, + size: this.itemsPerPage, + sort: this.sort(), + }) + .subscribe((res: HttpResponse) => this.onSuccess(res.body, res.headers)); + } + + private sort(): string[] { + const result = [this.predicate + ',' + (this.ascending ? 'asc' : 'desc')]; + if (this.predicate !== 'id') { + result.push('id'); + } + return result; + } + + private onSuccess(users: User[] | null, headers: HttpHeaders): void { + this.totalItems = Number(headers.get('X-Total-Count')); + this.users = users; + } +} diff --git a/src/main/webapp/app/admin/user-management/user-management.module.ts b/src/main/webapp/app/admin/user-management/user-management.module.ts new file mode 100644 index 0000000..464d98b --- /dev/null +++ b/src/main/webapp/app/admin/user-management/user-management.module.ts @@ -0,0 +1,21 @@ +import { NgModule } from '@angular/core'; +import { RouterModule } from '@angular/router'; + +import { EpasmedSharedModule } from 'app/shared/shared.module'; +import { UserManagementComponent } from './user-management.component'; +import { UserManagementDetailComponent } from './user-management-detail.component'; +import { UserManagementUpdateComponent } from './user-management-update.component'; +import { UserManagementDeleteDialogComponent } from './user-management-delete-dialog.component'; +import { userManagementRoute } from './user-management.route'; + +@NgModule({ + imports: [EpasmedSharedModule, RouterModule.forChild(userManagementRoute)], + declarations: [ + UserManagementComponent, + UserManagementDetailComponent, + UserManagementUpdateComponent, + UserManagementDeleteDialogComponent, + ], + entryComponents: [UserManagementDeleteDialogComponent], +}) +export class UserManagementModule {} diff --git a/src/main/webapp/app/admin/user-management/user-management.route.ts b/src/main/webapp/app/admin/user-management/user-management.route.ts new file mode 100644 index 0000000..a85ba63 --- /dev/null +++ b/src/main/webapp/app/admin/user-management/user-management.route.ts @@ -0,0 +1,53 @@ +import { Injectable } from '@angular/core'; +import { Resolve, ActivatedRouteSnapshot, Routes } from '@angular/router'; +import { Observable, of } from 'rxjs'; + +import { User, IUser } from 'app/core/user/user.model'; +import { UserService } from 'app/core/user/user.service'; +import { UserManagementComponent } from './user-management.component'; +import { UserManagementDetailComponent } from './user-management-detail.component'; +import { UserManagementUpdateComponent } from './user-management-update.component'; + +@Injectable({ providedIn: 'root' }) +export class UserManagementResolve implements Resolve { + constructor(private service: UserService) {} + + resolve(route: ActivatedRouteSnapshot): Observable { + const id = route.params['login']; + if (id) { + return this.service.find(id); + } + return of(new User()); + } +} + +export const userManagementRoute: Routes = [ + { + path: '', + component: UserManagementComponent, + data: { + defaultSort: 'id,asc', + }, + }, + { + path: ':login/view', + component: UserManagementDetailComponent, + resolve: { + user: UserManagementResolve, + }, + }, + { + path: 'new', + component: UserManagementUpdateComponent, + resolve: { + user: UserManagementResolve, + }, + }, + { + path: ':login/edit', + component: UserManagementUpdateComponent, + resolve: { + user: UserManagementResolve, + }, + }, +]; diff --git a/src/main/webapp/app/app-routing.module.ts b/src/main/webapp/app/app-routing.module.ts new file mode 100644 index 0000000..06e6829 --- /dev/null +++ b/src/main/webapp/app/app-routing.module.ts @@ -0,0 +1,40 @@ +import { NgModule } from '@angular/core'; +import { RouterModule } from '@angular/router'; +import { errorRoute } from './layouts/error/error.route'; +import { navbarRoute } from './layouts/navbar/navbar.route'; +import { DEBUG_INFO_ENABLED } from 'app/app.constants'; +import { Authority } from 'app/shared/constants/authority.constants'; + +import { UserRouteAccessService } from 'app/core/auth/user-route-access-service'; + +const LAYOUT_ROUTES = [navbarRoute, ...errorRoute]; + +@NgModule({ + imports: [ + RouterModule.forRoot( + [ + { + path: 'admin', + data: { + authorities: [Authority.ADMIN], + }, + canActivate: [UserRouteAccessService], + loadChildren: () => import('./admin/admin-routing.module').then(m => m.AdminRoutingModule), + }, + { + path: 'account', + loadChildren: () => import('./account/account.module').then(m => m.AccountModule), + }, + { + path: 'entities', + loadChildren: () => import('./entities/entity.module').then(m => m.EpasmedEntityModule), + }, + + ...LAYOUT_ROUTES, + ], + { enableTracing: DEBUG_INFO_ENABLED } + ), + ], + exports: [RouterModule], +}) +export class EpasmedAppRoutingModule {} diff --git a/src/main/webapp/app/app.constants.ts b/src/main/webapp/app/app.constants.ts new file mode 100644 index 0000000..4b34b65 --- /dev/null +++ b/src/main/webapp/app/app.constants.ts @@ -0,0 +1,8 @@ +// These constants are injected via webpack environment variables. +// You can add more variables in webpack.common.js or in profile specific webpack..js files. +// If you change the values in the webpack config files, you need to re run webpack to update the application + +export const VERSION = process.env.VERSION; +export const DEBUG_INFO_ENABLED = Boolean(process.env.DEBUG_INFO_ENABLED); +export const SERVER_API_URL = process.env.SERVER_API_URL; +export const BUILD_TIMESTAMP = process.env.BUILD_TIMESTAMP; diff --git a/src/main/webapp/app/app.main.ts b/src/main/webapp/app/app.main.ts new file mode 100644 index 0000000..f2594cf --- /dev/null +++ b/src/main/webapp/app/app.main.ts @@ -0,0 +1,16 @@ +import './polyfills'; +import { platformBrowserDynamic } from '@angular/platform-browser-dynamic'; +import { ProdConfig } from './blocks/config/prod.config'; +import { EpasmedAppModule } from './app.module'; + +ProdConfig(); + +if (module['hot']) { + module['hot'].accept(); +} + +platformBrowserDynamic() + .bootstrapModule(EpasmedAppModule, { preserveWhitespaces: true }) + // eslint-disable-next-line no-console + .then(() => console.log('Application started')) + .catch(err => console.error(err)); diff --git a/src/main/webapp/app/app.module.ts b/src/main/webapp/app/app.module.ts new file mode 100644 index 0000000..c8a0556 --- /dev/null +++ b/src/main/webapp/app/app.module.ts @@ -0,0 +1,30 @@ +import { NgModule } from '@angular/core'; +import { BrowserModule } from '@angular/platform-browser'; + +import './vendor'; +import { EpasmedSharedModule } from 'app/shared/shared.module'; +import { EpasmedCoreModule } from 'app/core/core.module'; +import { EpasmedAppRoutingModule } from './app-routing.module'; +import { EpasmedHomeModule } from './home/home.module'; +import { EpasmedEntityModule } from './entities/entity.module'; +// jhipster-needle-angular-add-module-import JHipster will add new module here +import { MainComponent } from './layouts/main/main.component'; +import { NavbarComponent } from './layouts/navbar/navbar.component'; +import { FooterComponent } from './layouts/footer/footer.component'; +import { PageRibbonComponent } from './layouts/profiles/page-ribbon.component'; +import { ErrorComponent } from './layouts/error/error.component'; + +@NgModule({ + imports: [ + BrowserModule, + EpasmedSharedModule, + EpasmedCoreModule, + EpasmedHomeModule, + // jhipster-needle-angular-add-module JHipster will add new module here + EpasmedEntityModule, + EpasmedAppRoutingModule, + ], + declarations: [MainComponent, NavbarComponent, ErrorComponent, PageRibbonComponent, FooterComponent], + bootstrap: [MainComponent], +}) +export class EpasmedAppModule {} diff --git a/src/main/webapp/app/blocks/config/prod.config.ts b/src/main/webapp/app/blocks/config/prod.config.ts new file mode 100644 index 0000000..b1ccef1 --- /dev/null +++ b/src/main/webapp/app/blocks/config/prod.config.ts @@ -0,0 +1,9 @@ +import { enableProdMode } from '@angular/core'; +import { DEBUG_INFO_ENABLED } from 'app/app.constants'; + +export function ProdConfig(): void { + // disable debug data on prod profile to improve performance + if (!DEBUG_INFO_ENABLED) { + enableProdMode(); + } +} diff --git a/src/main/webapp/app/blocks/config/uib-pagination.config.ts b/src/main/webapp/app/blocks/config/uib-pagination.config.ts new file mode 100644 index 0000000..002c5cf --- /dev/null +++ b/src/main/webapp/app/blocks/config/uib-pagination.config.ts @@ -0,0 +1,13 @@ +import { Injectable } from '@angular/core'; +import { NgbPaginationConfig } from '@ng-bootstrap/ng-bootstrap'; +import { ITEMS_PER_PAGE } from 'app/shared/constants/pagination.constants'; + +@Injectable({ providedIn: 'root' }) +export class PaginationConfig { + constructor(config: NgbPaginationConfig) { + config.boundaryLinks = true; + config.maxSize = 5; + config.pageSize = ITEMS_PER_PAGE; + config.size = 'sm'; + } +} diff --git a/src/main/webapp/app/blocks/interceptor/auth-expired.interceptor.ts b/src/main/webapp/app/blocks/interceptor/auth-expired.interceptor.ts new file mode 100644 index 0000000..b0b1767 --- /dev/null +++ b/src/main/webapp/app/blocks/interceptor/auth-expired.interceptor.ts @@ -0,0 +1,36 @@ +import { Injectable } from '@angular/core'; +import { HttpInterceptor, HttpRequest, HttpHandler, HttpEvent, HttpErrorResponse } from '@angular/common/http'; +import { Observable } from 'rxjs'; +import { tap } from 'rxjs/operators'; +import { Router } from '@angular/router'; + +import { LoginService } from 'app/core/login/login.service'; +import { LoginModalService } from 'app/core/login/login-modal.service'; +import { StateStorageService } from 'app/core/auth/state-storage.service'; + +@Injectable() +export class AuthExpiredInterceptor implements HttpInterceptor { + constructor( + private loginService: LoginService, + private loginModalService: LoginModalService, + private stateStorageService: StateStorageService, + private router: Router + ) {} + + intercept(request: HttpRequest, next: HttpHandler): Observable> { + return next.handle(request).pipe( + tap(null, (err: HttpErrorResponse) => { + if (err.status === 401 && err.url && !err.url.includes('api/account')) { + if (err.url.includes(this.loginService.logoutUrl())) { + this.loginService.logoutInClient(); + return; + } + this.stateStorageService.storeUrl(this.router.routerState.snapshot.url); + this.loginService.logout(); + this.router.navigate(['']); + this.loginModalService.open(); + } + }) + ); + } +} diff --git a/src/main/webapp/app/blocks/interceptor/errorhandler.interceptor.ts b/src/main/webapp/app/blocks/interceptor/errorhandler.interceptor.ts new file mode 100644 index 0000000..bdd6ab8 --- /dev/null +++ b/src/main/webapp/app/blocks/interceptor/errorhandler.interceptor.ts @@ -0,0 +1,20 @@ +import { Injectable } from '@angular/core'; +import { JhiEventManager, JhiEventWithContent } from 'ng-jhipster'; +import { HttpInterceptor, HttpRequest, HttpErrorResponse, HttpHandler, HttpEvent } from '@angular/common/http'; +import { Observable } from 'rxjs'; +import { tap } from 'rxjs/operators'; + +@Injectable() +export class ErrorHandlerInterceptor implements HttpInterceptor { + constructor(private eventManager: JhiEventManager) {} + + intercept(request: HttpRequest, next: HttpHandler): Observable> { + return next.handle(request).pipe( + tap(null, (err: HttpErrorResponse) => { + if (!(err.status === 401 && (err.message === '' || (err.url && err.url.includes('api/account'))))) { + this.eventManager.broadcast(new JhiEventWithContent('epasmedApp.httpError', err)); + } + }) + ); + } +} diff --git a/src/main/webapp/app/blocks/interceptor/notification.interceptor.ts b/src/main/webapp/app/blocks/interceptor/notification.interceptor.ts new file mode 100644 index 0000000..bf954bd --- /dev/null +++ b/src/main/webapp/app/blocks/interceptor/notification.interceptor.ts @@ -0,0 +1,30 @@ +import { JhiAlertService } from 'ng-jhipster'; +import { HttpInterceptor, HttpRequest, HttpResponse, HttpHandler, HttpEvent } from '@angular/common/http'; +import { Injectable } from '@angular/core'; +import { Observable } from 'rxjs'; +import { tap } from 'rxjs/operators'; + +@Injectable() +export class NotificationInterceptor implements HttpInterceptor { + constructor(private alertService: JhiAlertService) {} + + intercept(request: HttpRequest, next: HttpHandler): Observable> { + return next.handle(request).pipe( + tap((event: HttpEvent) => { + if (event instanceof HttpResponse) { + let alert: string | null = null; + + event.headers.keys().forEach(entry => { + if (entry.toLowerCase().endsWith('app-alert')) { + alert = event.headers.get(entry); + } + }); + + if (alert) { + this.alertService.success(alert); + } + } + }) + ); + } +} diff --git a/src/main/webapp/app/core/auth/account.service.ts b/src/main/webapp/app/core/auth/account.service.ts new file mode 100644 index 0000000..2ef76d2 --- /dev/null +++ b/src/main/webapp/app/core/auth/account.service.ts @@ -0,0 +1,82 @@ +import { Injectable } from '@angular/core'; +import { Router } from '@angular/router'; +import { HttpClient } from '@angular/common/http'; +import { Observable, ReplaySubject, of } from 'rxjs'; +import { shareReplay, tap, catchError } from 'rxjs/operators'; +import { StateStorageService } from 'app/core/auth/state-storage.service'; + +import { SERVER_API_URL } from 'app/app.constants'; +import { Account } from 'app/core/user/account.model'; + +@Injectable({ providedIn: 'root' }) +export class AccountService { + private userIdentity: Account | null = null; + private authenticationState = new ReplaySubject(1); + private accountCache$?: Observable; + + constructor(private http: HttpClient, private stateStorageService: StateStorageService, private router: Router) {} + + save(account: Account): Observable<{}> { + return this.http.post(SERVER_API_URL + 'api/account', account); + } + + authenticate(identity: Account | null): void { + this.userIdentity = identity; + this.authenticationState.next(this.userIdentity); + } + + hasAnyAuthority(authorities: string[] | string): boolean { + if (!this.userIdentity || !this.userIdentity.authorities) { + return false; + } + if (!Array.isArray(authorities)) { + authorities = [authorities]; + } + return this.userIdentity.authorities.some((authority: string) => authorities.includes(authority)); + } + + identity(force?: boolean): Observable { + if (!this.accountCache$ || force || !this.isAuthenticated()) { + this.accountCache$ = this.fetch().pipe( + catchError(() => { + return of(null); + }), + tap((account: Account | null) => { + this.authenticate(account); + + if (account) { + this.navigateToStoredUrl(); + } + }), + shareReplay() + ); + } + return this.accountCache$; + } + + isAuthenticated(): boolean { + return this.userIdentity !== null; + } + + getAuthenticationState(): Observable { + return this.authenticationState.asObservable(); + } + + getImageUrl(): string { + return this.userIdentity ? this.userIdentity.imageUrl : ''; + } + + private fetch(): Observable { + return this.http.get(SERVER_API_URL + 'api/account'); + } + + private navigateToStoredUrl(): void { + // previousState can be set in the authExpiredInterceptor and in the userRouteAccessService + // if login is successful, go to stored previousState and clear previousState + const previousUrl = this.stateStorageService.getUrl(); + if (previousUrl) { + this.stateStorageService.clearUrl(); + this.router.navigateByUrl(previousUrl); + } + } +} diff --git a/src/main/webapp/app/core/auth/auth-session.service.ts b/src/main/webapp/app/core/auth/auth-session.service.ts new file mode 100644 index 0000000..d013091 --- /dev/null +++ b/src/main/webapp/app/core/auth/auth-session.service.ts @@ -0,0 +1,36 @@ +import { Injectable } from '@angular/core'; +import { HttpClient, HttpHeaders } from '@angular/common/http'; +import { Observable } from 'rxjs'; +import { map } from 'rxjs/operators'; + +import { SERVER_API_URL } from 'app/app.constants'; +import { Login } from 'app/core/login/login.model'; + +export const LOGOUT_URL = SERVER_API_URL + 'api/logout'; + +@Injectable({ providedIn: 'root' }) +export class AuthServerProvider { + constructor(private http: HttpClient) {} + + login(credentials: Login): Observable<{}> { + const data = + `username=${encodeURIComponent(credentials.username)}` + + `&password=${encodeURIComponent(credentials.password)}` + + `&remember-me=${credentials.rememberMe}` + + '&submit=Login'; + + const headers = new HttpHeaders().set('Content-Type', 'application/x-www-form-urlencoded'); + + return this.http.post(SERVER_API_URL + 'api/authentication', data, { headers }); + } + + logout(): Observable { + // logout from the server + return this.http.post(LOGOUT_URL, {}).pipe( + map(() => { + // to get a new csrf token call the api + this.http.get(SERVER_API_URL + 'api/account').subscribe(); + }) + ); + } +} diff --git a/src/main/webapp/app/core/auth/csrf.service.ts b/src/main/webapp/app/core/auth/csrf.service.ts new file mode 100644 index 0000000..0a49162 --- /dev/null +++ b/src/main/webapp/app/core/auth/csrf.service.ts @@ -0,0 +1,11 @@ +import { Injectable } from '@angular/core'; +import { CookieService } from 'ngx-cookie-service'; + +@Injectable({ providedIn: 'root' }) +export class CSRFService { + constructor(private cookieService: CookieService) {} + + getCSRF(name = 'XSRF-TOKEN'): string { + return this.cookieService.get(name); + } +} diff --git a/src/main/webapp/app/core/auth/state-storage.service.ts b/src/main/webapp/app/core/auth/state-storage.service.ts new file mode 100644 index 0000000..5d8861d --- /dev/null +++ b/src/main/webapp/app/core/auth/state-storage.service.ts @@ -0,0 +1,21 @@ +import { Injectable } from '@angular/core'; +import { SessionStorageService } from 'ngx-webstorage'; + +@Injectable({ providedIn: 'root' }) +export class StateStorageService { + private previousUrlKey = 'previousUrl'; + + constructor(private $sessionStorage: SessionStorageService) {} + + storeUrl(url: string): void { + this.$sessionStorage.store(this.previousUrlKey, url); + } + + getUrl(): string | null | undefined { + return this.$sessionStorage.retrieve(this.previousUrlKey); + } + + clearUrl(): void { + this.$sessionStorage.clear(this.previousUrlKey); + } +} diff --git a/src/main/webapp/app/core/auth/user-route-access-service.ts b/src/main/webapp/app/core/auth/user-route-access-service.ts new file mode 100644 index 0000000..30d7b7e --- /dev/null +++ b/src/main/webapp/app/core/auth/user-route-access-service.ts @@ -0,0 +1,53 @@ +import { Injectable, isDevMode } from '@angular/core'; +import { ActivatedRouteSnapshot, CanActivate, Router, RouterStateSnapshot } from '@angular/router'; +import { Observable } from 'rxjs'; +import { map } from 'rxjs/operators'; + +import { AccountService } from 'app/core/auth/account.service'; +import { LoginModalService } from 'app/core/login/login-modal.service'; +import { StateStorageService } from './state-storage.service'; + +@Injectable({ providedIn: 'root' }) +export class UserRouteAccessService implements CanActivate { + constructor( + private router: Router, + private loginModalService: LoginModalService, + private accountService: AccountService, + private stateStorageService: StateStorageService + ) {} + + canActivate(route: ActivatedRouteSnapshot, state: RouterStateSnapshot): Observable { + const authorities = route.data['authorities']; + // We need to call the checkLogin / and so the accountService.identity() function, to ensure, + // that the client has a principal too, if they already logged in by the server. + // This could happen on a page refresh. + return this.checkLogin(authorities, state.url); + } + + checkLogin(authorities: string[], url: string): Observable { + return this.accountService.identity().pipe( + map(account => { + if (!authorities || authorities.length === 0) { + return true; + } + + if (account) { + const hasAnyAuthority = this.accountService.hasAnyAuthority(authorities); + if (hasAnyAuthority) { + return true; + } + if (isDevMode()) { + console.error('User has not any of required authorities: ', authorities); + } + this.router.navigate(['accessdenied']); + return false; + } + + this.stateStorageService.storeUrl(url); + this.router.navigate(['']); + this.loginModalService.open(); + return false; + }) + ); + } +} diff --git a/src/main/webapp/app/core/core.module.ts b/src/main/webapp/app/core/core.module.ts new file mode 100644 index 0000000..f219668 --- /dev/null +++ b/src/main/webapp/app/core/core.module.ts @@ -0,0 +1,63 @@ +import { NgModule, LOCALE_ID } from '@angular/core'; +import { DatePipe, registerLocaleData } from '@angular/common'; +import { HttpClientModule, HTTP_INTERCEPTORS } from '@angular/common/http'; +import { Title } from '@angular/platform-browser'; +import { FaIconLibrary } from '@fortawesome/angular-fontawesome'; +import { CookieService } from 'ngx-cookie-service'; +import { NgxWebstorageModule } from 'ngx-webstorage'; +import { NgJhipsterModule } from 'ng-jhipster'; +import locale from '@angular/common/locales/en'; + +import * as moment from 'moment'; +import { NgbDateAdapter, NgbDatepickerConfig } from '@ng-bootstrap/ng-bootstrap'; +import { NgbDateMomentAdapter } from 'app/shared/util/datepicker-adapter'; + +import { AuthExpiredInterceptor } from 'app/blocks/interceptor/auth-expired.interceptor'; +import { ErrorHandlerInterceptor } from 'app/blocks/interceptor/errorhandler.interceptor'; +import { NotificationInterceptor } from 'app/blocks/interceptor/notification.interceptor'; + +import { fontAwesomeIcons } from './icons/font-awesome-icons'; + +@NgModule({ + imports: [ + HttpClientModule, + NgxWebstorageModule.forRoot({ prefix: 'jhi', separator: '-' }), + NgJhipsterModule.forRoot({ + // set below to true to make alerts look like toast + alertAsToast: false, + alertTimeout: 5000, + }), + ], + providers: [ + Title, + CookieService, + { + provide: LOCALE_ID, + useValue: 'en', + }, + { provide: NgbDateAdapter, useClass: NgbDateMomentAdapter }, + DatePipe, + { + provide: HTTP_INTERCEPTORS, + useClass: AuthExpiredInterceptor, + multi: true, + }, + { + provide: HTTP_INTERCEPTORS, + useClass: ErrorHandlerInterceptor, + multi: true, + }, + { + provide: HTTP_INTERCEPTORS, + useClass: NotificationInterceptor, + multi: true, + }, + ], +}) +export class EpasmedCoreModule { + constructor(iconLibrary: FaIconLibrary, dpConfig: NgbDatepickerConfig) { + registerLocaleData(locale); + iconLibrary.addIcons(...fontAwesomeIcons); + dpConfig.minDate = { year: moment().year() - 100, month: 1, day: 1 }; + } +} diff --git a/src/main/webapp/app/core/icons/font-awesome-icons.ts b/src/main/webapp/app/core/icons/font-awesome-icons.ts new file mode 100644 index 0000000..6c4872c --- /dev/null +++ b/src/main/webapp/app/core/icons/font-awesome-icons.ts @@ -0,0 +1,75 @@ +import { + faUser, + faSort, + faSortUp, + faSortDown, + faSync, + faEye, + faBan, + faTimes, + faArrowLeft, + faSave, + faPlus, + faPencilAlt, + faBars, + faThList, + faUserPlus, + faRoad, + faTachometerAlt, + faHeart, + faList, + faBell, + faBook, + faHdd, + faFlag, + faWrench, + faLock, + faCloud, + faSignOutAlt, + faSignInAlt, + faCalendarAlt, + faSearch, + faTrashAlt, + faAsterisk, + faTasks, + // jhipster-needle-add-icon-import + faHome, +} from '@fortawesome/free-solid-svg-icons'; + +export const fontAwesomeIcons = [ + faUser, + faSort, + faSortUp, + faSortDown, + faSync, + faEye, + faBan, + faTimes, + faArrowLeft, + faSave, + faPlus, + faPencilAlt, + faBars, + faHome, + faThList, + faUserPlus, + faRoad, + faTachometerAlt, + faHeart, + faList, + faBell, + faTasks, + faBook, + faHdd, + faFlag, + faWrench, + faLock, + faCloud, + faSignOutAlt, + faSignInAlt, + faCalendarAlt, + faSearch, + faTrashAlt, + // jhipster-needle-add-icon-import + faAsterisk, +]; diff --git a/src/main/webapp/app/core/login/login-modal.service.ts b/src/main/webapp/app/core/login/login-modal.service.ts new file mode 100644 index 0000000..73af6ae --- /dev/null +++ b/src/main/webapp/app/core/login/login-modal.service.ts @@ -0,0 +1,20 @@ +import { Injectable } from '@angular/core'; +import { NgbModal, NgbModalRef } from '@ng-bootstrap/ng-bootstrap'; + +import { LoginModalComponent } from 'app/shared/login/login.component'; + +@Injectable({ providedIn: 'root' }) +export class LoginModalService { + private isOpen = false; + + constructor(private modalService: NgbModal) {} + + open(): void { + if (this.isOpen) { + return; + } + this.isOpen = true; + const modalRef: NgbModalRef = this.modalService.open(LoginModalComponent); + modalRef.result.finally(() => (this.isOpen = false)); + } +} diff --git a/src/main/webapp/app/core/login/login.model.ts b/src/main/webapp/app/core/login/login.model.ts new file mode 100644 index 0000000..422fce9 --- /dev/null +++ b/src/main/webapp/app/core/login/login.model.ts @@ -0,0 +1,3 @@ +export class Login { + constructor(public username: string, public password: string, public rememberMe: boolean) {} +} diff --git a/src/main/webapp/app/core/login/login.service.ts b/src/main/webapp/app/core/login/login.service.ts new file mode 100644 index 0000000..b384fac --- /dev/null +++ b/src/main/webapp/app/core/login/login.service.ts @@ -0,0 +1,29 @@ +import { Injectable } from '@angular/core'; +import { Observable } from 'rxjs'; +import { flatMap } from 'rxjs/operators'; + +import { Account } from 'app/core/user/account.model'; +import { AccountService } from 'app/core/auth/account.service'; +import { AuthServerProvider, LOGOUT_URL } from 'app/core/auth/auth-session.service'; +import { Login } from './login.model'; + +@Injectable({ providedIn: 'root' }) +export class LoginService { + constructor(private accountService: AccountService, private authServerProvider: AuthServerProvider) {} + + login(credentials: Login): Observable { + return this.authServerProvider.login(credentials).pipe(flatMap(() => this.accountService.identity(true))); + } + + logoutUrl(): string { + return LOGOUT_URL; + } + + logoutInClient(): void { + this.accountService.authenticate(null); + } + + logout(): void { + this.authServerProvider.logout().subscribe(null, null, () => this.accountService.authenticate(null)); + } +} diff --git a/src/main/webapp/app/core/user/account.model.ts b/src/main/webapp/app/core/user/account.model.ts new file mode 100644 index 0000000..4d2c833 --- /dev/null +++ b/src/main/webapp/app/core/user/account.model.ts @@ -0,0 +1,12 @@ +export class Account { + constructor( + public activated: boolean, + public authorities: string[], + public email: string, + public firstName: string, + public langKey: string, + public lastName: string, + public login: string, + public imageUrl: string + ) {} +} diff --git a/src/main/webapp/app/core/user/user.model.ts b/src/main/webapp/app/core/user/user.model.ts new file mode 100644 index 0000000..585f7ed --- /dev/null +++ b/src/main/webapp/app/core/user/user.model.ts @@ -0,0 +1,33 @@ +export interface IUser { + id?: any; + login?: string; + firstName?: string; + lastName?: string; + email?: string; + activated?: boolean; + langKey?: string; + authorities?: string[]; + createdBy?: string; + createdDate?: Date; + lastModifiedBy?: string; + lastModifiedDate?: Date; + password?: string; +} + +export class User implements IUser { + constructor( + public id?: any, + public login?: string, + public firstName?: string, + public lastName?: string, + public email?: string, + public activated?: boolean, + public langKey?: string, + public authorities?: string[], + public createdBy?: string, + public createdDate?: Date, + public lastModifiedBy?: string, + public lastModifiedDate?: Date, + public password?: string + ) {} +} diff --git a/src/main/webapp/app/core/user/user.service.ts b/src/main/webapp/app/core/user/user.service.ts new file mode 100644 index 0000000..b598769 --- /dev/null +++ b/src/main/webapp/app/core/user/user.service.ts @@ -0,0 +1,39 @@ +import { Injectable } from '@angular/core'; +import { HttpClient, HttpResponse } from '@angular/common/http'; +import { Observable } from 'rxjs'; + +import { SERVER_API_URL } from 'app/app.constants'; +import { createRequestOption, Pagination } from 'app/shared/util/request-util'; +import { IUser } from './user.model'; + +@Injectable({ providedIn: 'root' }) +export class UserService { + public resourceUrl = SERVER_API_URL + 'api/users'; + + constructor(private http: HttpClient) {} + + create(user: IUser): Observable { + return this.http.post(this.resourceUrl, user); + } + + update(user: IUser): Observable { + return this.http.put(this.resourceUrl, user); + } + + find(login: string): Observable { + return this.http.get(`${this.resourceUrl}/${login}`); + } + + query(req?: Pagination): Observable> { + const options = createRequestOption(req); + return this.http.get(this.resourceUrl, { params: options, observe: 'response' }); + } + + delete(login: string): Observable<{}> { + return this.http.delete(`${this.resourceUrl}/${login}`); + } + + authorities(): Observable { + return this.http.get(SERVER_API_URL + 'api/users/authorities'); + } +} diff --git a/src/main/webapp/app/entities/entity.module.ts b/src/main/webapp/app/entities/entity.module.ts new file mode 100644 index 0000000..261a460 --- /dev/null +++ b/src/main/webapp/app/entities/entity.module.ts @@ -0,0 +1,18 @@ +import { NgModule } from '@angular/core'; +import { RouterModule } from '@angular/router'; + +@NgModule({ + imports: [ + RouterModule.forChild([ + /* jhipster-needle-add-entity-route - JHipster will add entity modules routes here */ + { + path: 'tabssi', + loadChildren: () => import('./tabssi/tabssi.module').then(m => m.TabsSIModule), + data: { + pageTitle: 'Tables SI', + }, + }, + ]), + ], +}) +export class EpasmedEntityModule {} diff --git a/src/main/webapp/app/entities/tabssi/tabssi-delete-dialog.component.html b/src/main/webapp/app/entities/tabssi/tabssi-delete-dialog.component.html new file mode 100644 index 0000000..aa3bb25 --- /dev/null +++ b/src/main/webapp/app/entities/tabssi/tabssi-delete-dialog.component.html @@ -0,0 +1,23 @@ +
+ + + + + +
diff --git a/src/main/webapp/app/entities/tabssi/tabssi-delete-dialog.component.ts b/src/main/webapp/app/entities/tabssi/tabssi-delete-dialog.component.ts new file mode 100644 index 0000000..5f687e6 --- /dev/null +++ b/src/main/webapp/app/entities/tabssi/tabssi-delete-dialog.component.ts @@ -0,0 +1,27 @@ +import { Component } from '@angular/core'; +import { NgbActiveModal } from '@ng-bootstrap/ng-bootstrap'; +import { JhiEventManager } from 'ng-jhipster'; + +import { TabsSI } from './tabssi.model'; +import { TabsSIService } from './tabssi.service'; + +@Component({ + selector: 'jhi-tabssi-delete-dialog', + templateUrl: './tabssi-delete-dialog.component.html', +}) +export class TabsSIDeleteDialogComponent { + tabsSI?: TabsSI; + + constructor(private tabsSIService: TabsSIService, public activeModal: NgbActiveModal, private eventManager: JhiEventManager) {} + + cancel(): void { + this.activeModal.dismiss(); + } + + confirmDelete(id: string): void { + this.tabsSIService.delete(id).subscribe(() => { + this.eventManager.broadcast('tabsSIListModification'); + this.activeModal.close(); + }); + } +} diff --git a/src/main/webapp/app/entities/tabssi/tabssi-detail.component.html b/src/main/webapp/app/entities/tabssi/tabssi-detail.component.html new file mode 100644 index 0000000..03469fc --- /dev/null +++ b/src/main/webapp/app/entities/tabssi/tabssi-detail.component.html @@ -0,0 +1,30 @@ +
+
+
+

+ Table SI [{{ tabsSI.id }}] +

+ +
+
Id
+
{{ tabsSI.id }}
+ +
Nome
+
{{ tabsSI.nome }}
+ +
Operazioni
+
{{ tabsSI.operazioni }}
+ +
Id Flusso
+
{{ tabsSI.idFlusso }}
+ +
Last Updated
+
{{ tabsSI.lastUpdate }}
+
+ + +
+
+
diff --git a/src/main/webapp/app/entities/tabssi/tabssi-detail.component.ts b/src/main/webapp/app/entities/tabssi/tabssi-detail.component.ts new file mode 100644 index 0000000..276468d --- /dev/null +++ b/src/main/webapp/app/entities/tabssi/tabssi-detail.component.ts @@ -0,0 +1,18 @@ +import { Component, OnInit } from '@angular/core'; +import { ActivatedRoute } from '@angular/router'; + +import { TabsSI } from './tabssi.model'; + +@Component({ + selector: 'jhi-tabssi-detail', + templateUrl: './tabssi-detail.component.html', +}) +export class TabsSIDetailComponent implements OnInit { + tabsSI: TabsSI | null = null; + + constructor(private route: ActivatedRoute) {} + + ngOnInit(): void { + this.route.data.subscribe(({ tabsSI }) => (this.tabsSI = tabsSI)); + } +} diff --git a/src/main/webapp/app/entities/tabssi/tabssi-update.component.html b/src/main/webapp/app/entities/tabssi/tabssi-update.component.html new file mode 100644 index 0000000..5d490d1 --- /dev/null +++ b/src/main/webapp/app/entities/tabssi/tabssi-update.component.html @@ -0,0 +1,80 @@ +
+
+
+

+ Create or edit Table SI +

+ +
+ + +
+ + +
+ +
+ + + +
+ + This field is required. + +
+
+ +
+ + + +
+ + This field is required. + +
+
+ +
+ + + +
+ + This field is required. + +
+
+ +
+ + + +
+ + This field is required. + +
+
+
+ +
+ + + +
+
+
+
diff --git a/src/main/webapp/app/entities/tabssi/tabssi-update.component.ts b/src/main/webapp/app/entities/tabssi/tabssi-update.component.ts new file mode 100644 index 0000000..3fdd7a3 --- /dev/null +++ b/src/main/webapp/app/entities/tabssi/tabssi-update.component.ts @@ -0,0 +1,80 @@ +import { Component, OnInit } from '@angular/core'; +import { FormBuilder, Validators } from '@angular/forms'; +import { ActivatedRoute } from '@angular/router'; + +import { TabsSI } from './tabssi.model'; +import { TabsSIService } from './tabssi.service'; + +@Component({ + selector: 'jhi-tabssi-update', + templateUrl: './tabssi-update.component.html', +}) +export class TabsSIUpdateComponent implements OnInit { + tabsSI!: TabsSI; + isSaving = false; + + editForm = this.fb.group({ + id: [], + nome: ['', [Validators.required]], + operazioni: ['', [Validators.required]], + idFlusso: ['', [Validators.required]], + lastUpdate: ['', [Validators.required]], + }); + + constructor(private tabsSIService: TabsSIService, private route: ActivatedRoute, private fb: FormBuilder) {} + + ngOnInit(): void { + this.route.data.subscribe(({ tabsSI }) => { + if (tabsSI) { + this.tabsSI = tabsSI; + this.updateForm(tabsSI); + } + }); + } + + previousState(): void { + window.history.back(); + } + + save(): void { + this.isSaving = true; + this.updateTabsSI(this.tabsSI); + if (this.tabsSI.id !== undefined) { + this.tabsSIService.update(this.tabsSI).subscribe( + () => this.onSaveSuccess(), + () => this.onSaveError() + ); + } else { + this.tabsSIService.create(this.tabsSI).subscribe( + () => this.onSaveSuccess(), + () => this.onSaveError() + ); + } + } + + private updateForm(tabsSI: TabsSI): void { + this.editForm.patchValue({ + id: tabsSI.id, + nome: tabsSI.nome, + operazioni: tabsSI.operazioni, + idFlusso: tabsSI.idFlusso, + lastUpdate: tabsSI.lastUpdate, + }); + } + + private updateTabsSI(tabsSI: TabsSI): void { + tabsSI.nome = this.editForm.get(['nome'])!.value; + tabsSI.operazioni = this.editForm.get(['operazioni'])!.value; + tabsSI.idFlusso = this.editForm.get(['idFlusso'])!.value; + tabsSI.lastUpdate = this.editForm.get(['lastUpdate'])!.value; + } + + private onSaveSuccess(): void { + this.isSaving = false; + this.previousState(); + } + + private onSaveError(): void { + this.isSaving = false; + } +} diff --git a/src/main/webapp/app/entities/tabssi/tabssi.component.html b/src/main/webapp/app/entities/tabssi/tabssi.component.html new file mode 100644 index 0000000..905ac5a --- /dev/null +++ b/src/main/webapp/app/entities/tabssi/tabssi.component.html @@ -0,0 +1,75 @@ +
+

+ Tables SI + + +

+ + + + + +
+ + + + + + + + + + + + + + + + + + + + + + +
ID NomeOperazioniID FlussoLast Update
{{ tabssi.id }}{{ tabssi.nome }}{{ tabssi.operazioni }}{{ tabssi.idFlusso }}{{ tabssi.lastUpdate }} +
+ + + +
+
+
+ +
+
+ +
+ +
+ +
+
+
diff --git a/src/main/webapp/app/entities/tabssi/tabssi.component.ts b/src/main/webapp/app/entities/tabssi/tabssi.component.ts new file mode 100644 index 0000000..2bdbf79 --- /dev/null +++ b/src/main/webapp/app/entities/tabssi/tabssi.component.ts @@ -0,0 +1,102 @@ +import { Component, OnInit, OnDestroy } from '@angular/core'; +import { HttpResponse, HttpHeaders } from '@angular/common/http'; +import { NgbModal } from '@ng-bootstrap/ng-bootstrap'; +import { Subscription, combineLatest } from 'rxjs'; +import { ActivatedRoute, ParamMap, Router, Data } from '@angular/router'; +import { JhiEventManager } from 'ng-jhipster'; + +import { ITEMS_PER_PAGE } from 'app/shared/constants/pagination.constants'; +// import { AccountService } from 'app/core/auth/account.service'; +// import { Account } from 'app/core/user/account.model'; +import { TabsSIService } from './tabssi.service'; +import { TabsSI } from './tabssi.model'; +import { TabsSIDeleteDialogComponent } from './tabssi-delete-dialog.component'; + +@Component({ + selector: 'jhi-tabssi', + templateUrl: './tabssi.component.html', +}) +export class TabsSIComponent implements OnInit, OnDestroy { + // currentAccount: Account | null = null; + tabssis: TabsSI[] | null = null; + tabsSIListSubscription?: Subscription; + totalItems = 0; + itemsPerPage = ITEMS_PER_PAGE; + page!: number; + predicate!: string; + ascending!: boolean; + + constructor( + private tabsSIService: TabsSIService, + // private accountService: AccountService, + private activatedRoute: ActivatedRoute, + private router: Router, + private eventManager: JhiEventManager, + private modalService: NgbModal + ) {} + + ngOnInit(): void { + // this.accountService.identity().subscribe(account => (this.currentAccount = account)); + this.tabsSIListSubscription = this.eventManager.subscribe('tabsSIListModification', () => this.loadAll()); + this.handleNavigation(); + } + + ngOnDestroy(): void { + if (this.tabsSIListSubscription) { + this.eventManager.destroy(this.tabsSIListSubscription); + } + } + + trackIdentity(index: number, item: TabsSI): any { + return item.id; + } + + deleteTabsSI(tabsSI: TabsSI): void { + const modalRef = this.modalService.open(TabsSIDeleteDialogComponent, { size: 'lg', backdrop: 'static' }); + modalRef.componentInstance.tabsSI = tabsSI; + } + + transition(): void { + this.router.navigate(['./'], { + relativeTo: this.activatedRoute.parent, + queryParams: { + page: this.page, + sort: this.predicate + ',' + (this.ascending ? 'asc' : 'desc'), + }, + }); + } + + private handleNavigation(): void { + combineLatest(this.activatedRoute.data, this.activatedRoute.queryParamMap, (data: Data, params: ParamMap) => { + const page = params.get('page'); + this.page = page !== null ? +page : 1; + const sort = (params.get('sort') ?? data['defaultSort']).split(','); + this.predicate = sort[0]; + this.ascending = sort[1] === 'asc'; + this.loadAll(); + }).subscribe(); + } + + private loadAll(): void { + this.tabsSIService + .query({ + page: this.page - 1, + size: this.itemsPerPage, + sort: this.sort(), + }) + .subscribe((res: HttpResponse) => this.onSuccess(res.body, res.headers)); + } + + private sort(): string[] { + const result = [this.predicate + ',' + (this.ascending ? 'asc' : 'desc')]; + if (this.predicate !== 'id') { + result.push('id'); + } + return result; + } + + private onSuccess(tabssis: TabsSI[] | null, headers: HttpHeaders): void { + this.totalItems = Number(headers.get('X-Total-Count')); + this.tabssis = tabssis; + } +} diff --git a/src/main/webapp/app/entities/tabssi/tabssi.model.ts b/src/main/webapp/app/entities/tabssi/tabssi.model.ts new file mode 100644 index 0000000..0aca200 --- /dev/null +++ b/src/main/webapp/app/entities/tabssi/tabssi.model.ts @@ -0,0 +1,11 @@ +export interface ITabsSI { + id?: any; + nome?: string; + operazioni?: string; + idFlusso?: any; + lastUpdate?: string; +} + +export class TabsSI implements ITabsSI { + constructor(public id?: any, public nome?: string, public operazioni?: string, public idFlusso?: any, public lastUpdate?: string) {} +} diff --git a/src/main/webapp/app/entities/tabssi/tabssi.module.ts b/src/main/webapp/app/entities/tabssi/tabssi.module.ts new file mode 100644 index 0000000..8b0b2cd --- /dev/null +++ b/src/main/webapp/app/entities/tabssi/tabssi.module.ts @@ -0,0 +1,16 @@ +import { NgModule } from '@angular/core'; +import { RouterModule } from '@angular/router'; + +import { EpasmedSharedModule } from 'app/shared/shared.module'; +import { TabsSIComponent } from './tabssi.component'; +import { TabsSIDetailComponent } from './tabssi-detail.component'; +import { TabsSIUpdateComponent } from './tabssi-update.component'; +import { TabsSIDeleteDialogComponent } from './tabssi-delete-dialog.component'; +import { tabsSIRoute } from './tabssi.route'; + +@NgModule({ + imports: [EpasmedSharedModule, RouterModule.forChild(tabsSIRoute)], + declarations: [TabsSIComponent, TabsSIDetailComponent, TabsSIUpdateComponent, TabsSIDeleteDialogComponent], + entryComponents: [TabsSIComponent], +}) +export class TabsSIModule {} diff --git a/src/main/webapp/app/entities/tabssi/tabssi.route.ts b/src/main/webapp/app/entities/tabssi/tabssi.route.ts new file mode 100644 index 0000000..520696c --- /dev/null +++ b/src/main/webapp/app/entities/tabssi/tabssi.route.ts @@ -0,0 +1,53 @@ +import { Injectable } from '@angular/core'; +import { Resolve, ActivatedRouteSnapshot, Routes } from '@angular/router'; +import { Observable, of } from 'rxjs'; + +import { TabsSI, ITabsSI } from './tabssi.model'; +import { TabsSIService } from './tabssi.service'; +import { TabsSIComponent } from './tabssi.component'; +import { TabsSIDetailComponent } from './tabssi-detail.component'; +import { TabsSIUpdateComponent } from './tabssi-update.component'; + +@Injectable({ providedIn: 'root' }) +export class TabsSIResolve implements Resolve { + constructor(private service: TabsSIService) {} + + resolve(route: ActivatedRouteSnapshot): Observable { + const id = route.params['id']; + if (id) { + return this.service.find(id); + } + return of(new TabsSI()); + } +} + +export const tabsSIRoute: Routes = [ + { + path: '', + component: TabsSIComponent, + data: { + defaultSort: 'id,asc', + }, + }, + { + path: ':id/view', + component: TabsSIDetailComponent, + resolve: { + tabsSI: TabsSIResolve, + }, + }, + { + path: 'new', + component: TabsSIUpdateComponent, + resolve: { + tabsSI: TabsSIResolve, + }, + }, + { + path: ':id/edit', + component: TabsSIUpdateComponent, + resolve: { + tabsSI: TabsSIResolve, + }, + }, +]; diff --git a/src/main/webapp/app/entities/tabssi/tabssi.service.ts b/src/main/webapp/app/entities/tabssi/tabssi.service.ts new file mode 100644 index 0000000..355d8c9 --- /dev/null +++ b/src/main/webapp/app/entities/tabssi/tabssi.service.ts @@ -0,0 +1,35 @@ +import { Injectable } from '@angular/core'; +import { HttpClient, HttpResponse } from '@angular/common/http'; +import { Observable } from 'rxjs'; + +import { SERVER_API_URL } from 'app/app.constants'; +import { createRequestOption, Pagination } from 'app/shared/util/request-util'; +import { ITabsSI } from './tabssi.model'; + +@Injectable({ providedIn: 'root' }) +export class TabsSIService { + public resourceUrl = SERVER_API_URL + 'api/tabssi'; + + constructor(private http: HttpClient) {} + + query(req?: Pagination): Observable> { + const options = createRequestOption(req); + return this.http.get(this.resourceUrl, { params: options, observe: 'response' }); + } + + create(tabssi: ITabsSI): Observable { + return this.http.post(this.resourceUrl, tabssi); + } + + update(tabssi: ITabsSI): Observable { + return this.http.put(this.resourceUrl, tabssi); + } + + find(id: string): Observable { + return this.http.get(`${this.resourceUrl}/${id}`); + } + + delete(id: string): Observable<{}> { + return this.http.delete(`${this.resourceUrl}/${id}`); + } +} diff --git a/src/main/webapp/app/home/home.component.html b/src/main/webapp/app/home/home.component.html new file mode 100644 index 0000000..0f19ad6 --- /dev/null +++ b/src/main/webapp/app/home/home.component.html @@ -0,0 +1,32 @@ +
+
+ +
+ +
+

Welcome in ePAS Mediator!

+ + + + + +
+
diff --git a/src/main/webapp/app/home/home.component.ts b/src/main/webapp/app/home/home.component.ts new file mode 100644 index 0000000..ec2f76f --- /dev/null +++ b/src/main/webapp/app/home/home.component.ts @@ -0,0 +1,36 @@ +import { Component, OnInit, OnDestroy } from '@angular/core'; +import { Subscription } from 'rxjs'; + +import { LoginModalService } from 'app/core/login/login-modal.service'; +import { AccountService } from 'app/core/auth/account.service'; +import { Account } from 'app/core/user/account.model'; + +@Component({ + selector: 'jhi-home', + templateUrl: './home.component.html', + styleUrls: ['home.scss'], +}) +export class HomeComponent implements OnInit, OnDestroy { + account: Account | null = null; + authSubscription?: Subscription; + + constructor(private accountService: AccountService, private loginModalService: LoginModalService) {} + + ngOnInit(): void { + this.authSubscription = this.accountService.getAuthenticationState().subscribe(account => (this.account = account)); + } + + isAuthenticated(): boolean { + return this.accountService.isAuthenticated(); + } + + login(): void { + this.loginModalService.open(); + } + + ngOnDestroy(): void { + if (this.authSubscription) { + this.authSubscription.unsubscribe(); + } + } +} diff --git a/src/main/webapp/app/home/home.module.ts b/src/main/webapp/app/home/home.module.ts new file mode 100644 index 0000000..2bcac6b --- /dev/null +++ b/src/main/webapp/app/home/home.module.ts @@ -0,0 +1,12 @@ +import { NgModule } from '@angular/core'; +import { RouterModule } from '@angular/router'; + +import { EpasmedSharedModule } from 'app/shared/shared.module'; +import { HOME_ROUTE } from './home.route'; +import { HomeComponent } from './home.component'; + +@NgModule({ + imports: [EpasmedSharedModule, RouterModule.forChild([HOME_ROUTE])], + declarations: [HomeComponent], +}) +export class EpasmedHomeModule {} diff --git a/src/main/webapp/app/home/home.route.ts b/src/main/webapp/app/home/home.route.ts new file mode 100644 index 0000000..83d6d6e --- /dev/null +++ b/src/main/webapp/app/home/home.route.ts @@ -0,0 +1,12 @@ +import { Route } from '@angular/router'; + +import { HomeComponent } from './home.component'; + +export const HOME_ROUTE: Route = { + path: '', + component: HomeComponent, + data: { + authorities: [], + pageTitle: 'Welcome, to ePAS Mediator!', + }, +}; diff --git a/src/main/webapp/app/home/home.scss b/src/main/webapp/app/home/home.scss new file mode 100644 index 0000000..5e45ae5 --- /dev/null +++ b/src/main/webapp/app/home/home.scss @@ -0,0 +1,23 @@ +/* ========================================================================== +Main page styles +========================================================================== */ + +.hipster { + display: inline-block; + width: 347px; + height: 497px; + background: url('../../content/images/plumber_0.svg') no-repeat center top; + background-size: contain; +} + +/* wait autoprefixer update to allow simple generation of high pixel density media query */ +@media only screen and (-webkit-min-device-pixel-ratio: 2), + only screen and (-moz-min-device-pixel-ratio: 2), + only screen and (-o-min-device-pixel-ratio: 2/1), + only screen and (min-resolution: 192dpi), + only screen and (min-resolution: 2dppx) { + .hipster { + background: url('../../content/images/plumber_0.svg') no-repeat center top; + background-size: contain; + } +} diff --git a/src/main/webapp/app/layouts/error/error.component.html b/src/main/webapp/app/layouts/error/error.component.html new file mode 100644 index 0000000..d5f293f --- /dev/null +++ b/src/main/webapp/app/layouts/error/error.component.html @@ -0,0 +1,15 @@ +
+
+
+ +
+ +
+

Error Page!

+ +
+
{{ errorMessage }}
+
+
+
+
diff --git a/src/main/webapp/app/layouts/error/error.component.ts b/src/main/webapp/app/layouts/error/error.component.ts new file mode 100644 index 0000000..011feae --- /dev/null +++ b/src/main/webapp/app/layouts/error/error.component.ts @@ -0,0 +1,20 @@ +import { Component, OnInit } from '@angular/core'; +import { ActivatedRoute } from '@angular/router'; + +@Component({ + selector: 'jhi-error', + templateUrl: './error.component.html', +}) +export class ErrorComponent implements OnInit { + errorMessage?: string; + + constructor(private route: ActivatedRoute) {} + + ngOnInit(): void { + this.route.data.subscribe(routeData => { + if (routeData.errorMessage) { + this.errorMessage = routeData.errorMessage; + } + }); + } +} diff --git a/src/main/webapp/app/layouts/error/error.route.ts b/src/main/webapp/app/layouts/error/error.route.ts new file mode 100644 index 0000000..a5a1243 --- /dev/null +++ b/src/main/webapp/app/layouts/error/error.route.ts @@ -0,0 +1,36 @@ +import { Routes } from '@angular/router'; + +import { ErrorComponent } from './error.component'; + +export const errorRoute: Routes = [ + { + path: 'error', + component: ErrorComponent, + data: { + authorities: [], + pageTitle: 'Error page!', + }, + }, + { + path: 'accessdenied', + component: ErrorComponent, + data: { + authorities: [], + pageTitle: 'Error page!', + errorMessage: 'You are not authorized to access this page.', + }, + }, + { + path: '404', + component: ErrorComponent, + data: { + authorities: [], + pageTitle: 'Error page!', + errorMessage: 'The page does not exist.', + }, + }, + { + path: '**', + redirectTo: '/404', + }, +]; diff --git a/src/main/webapp/app/layouts/footer/footer.component.html b/src/main/webapp/app/layouts/footer/footer.component.html new file mode 100644 index 0000000..3b3d751 --- /dev/null +++ b/src/main/webapp/app/layouts/footer/footer.component.html @@ -0,0 +1,3 @@ + diff --git a/src/main/webapp/app/layouts/footer/footer.component.ts b/src/main/webapp/app/layouts/footer/footer.component.ts new file mode 100644 index 0000000..7c640ec --- /dev/null +++ b/src/main/webapp/app/layouts/footer/footer.component.ts @@ -0,0 +1,7 @@ +import { Component } from '@angular/core'; + +@Component({ + selector: 'jhi-footer', + templateUrl: './footer.component.html', +}) +export class FooterComponent {} diff --git a/src/main/webapp/app/layouts/main/main.component.html b/src/main/webapp/app/layouts/main/main.component.html new file mode 100644 index 0000000..d8bcc22 --- /dev/null +++ b/src/main/webapp/app/layouts/main/main.component.html @@ -0,0 +1,13 @@ + + +
+ +
+ +
+
+ +
+ + +
diff --git a/src/main/webapp/app/layouts/main/main.component.ts b/src/main/webapp/app/layouts/main/main.component.ts new file mode 100644 index 0000000..08cde79 --- /dev/null +++ b/src/main/webapp/app/layouts/main/main.component.ts @@ -0,0 +1,43 @@ +import { Component, OnInit } from '@angular/core'; +import { Title } from '@angular/platform-browser'; +import { Router, ActivatedRouteSnapshot, NavigationEnd, NavigationError } from '@angular/router'; + +import { AccountService } from 'app/core/auth/account.service'; + +@Component({ + selector: 'jhi-main', + templateUrl: './main.component.html', +}) +export class MainComponent implements OnInit { + constructor(private accountService: AccountService, private titleService: Title, private router: Router) {} + + ngOnInit(): void { + // try to log in automatically + this.accountService.identity().subscribe(); + + this.router.events.subscribe(event => { + if (event instanceof NavigationEnd) { + this.updateTitle(); + } + if (event instanceof NavigationError && event.error.status === 404) { + this.router.navigate(['/404']); + } + }); + } + + private getPageTitle(routeSnapshot: ActivatedRouteSnapshot): string { + let title: string = routeSnapshot.data && routeSnapshot.data['pageTitle'] ? routeSnapshot.data['pageTitle'] : ''; + if (routeSnapshot.firstChild) { + title = this.getPageTitle(routeSnapshot.firstChild) || title; + } + return title; + } + + private updateTitle(): void { + let pageTitle = this.getPageTitle(this.router.routerState.snapshot.root); + if (!pageTitle) { + pageTitle = 'ePASMed'; + } + this.titleService.setTitle(pageTitle); + } +} diff --git a/src/main/webapp/app/layouts/navbar/navbar.component.html b/src/main/webapp/app/layouts/navbar/navbar.component.html new file mode 100644 index 0000000..c9a1061 --- /dev/null +++ b/src/main/webapp/app/layouts/navbar/navbar.component.html @@ -0,0 +1,146 @@ + diff --git a/src/main/webapp/app/layouts/navbar/navbar.component.ts b/src/main/webapp/app/layouts/navbar/navbar.component.ts new file mode 100644 index 0000000..47a61d6 --- /dev/null +++ b/src/main/webapp/app/layouts/navbar/navbar.component.ts @@ -0,0 +1,63 @@ +import { Component, OnInit } from '@angular/core'; +import { Router } from '@angular/router'; + +import { VERSION } from 'app/app.constants'; +import { AccountService } from 'app/core/auth/account.service'; +import { LoginModalService } from 'app/core/login/login-modal.service'; +import { LoginService } from 'app/core/login/login.service'; +import { ProfileService } from 'app/layouts/profiles/profile.service'; + +@Component({ + selector: 'jhi-navbar', + templateUrl: './navbar.component.html', + styleUrls: ['navbar.scss'], +}) +export class NavbarComponent implements OnInit { + inProduction?: boolean; + isNavbarCollapsed = true; + swaggerEnabled?: boolean; + version: string; + + constructor( + private loginService: LoginService, + private accountService: AccountService, + private loginModalService: LoginModalService, + private profileService: ProfileService, + private router: Router + ) { + this.version = VERSION ? (VERSION.toLowerCase().startsWith('v') ? VERSION : 'v' + VERSION) : ''; + } + + ngOnInit(): void { + this.profileService.getProfileInfo().subscribe(profileInfo => { + this.inProduction = profileInfo.inProduction; + this.swaggerEnabled = profileInfo.swaggerEnabled; + }); + } + + collapseNavbar(): void { + this.isNavbarCollapsed = true; + } + + isAuthenticated(): boolean { + return this.accountService.isAuthenticated(); + } + + login(): void { + this.loginModalService.open(); + } + + logout(): void { + this.collapseNavbar(); + this.loginService.logout(); + this.router.navigate(['']); + } + + toggleNavbar(): void { + this.isNavbarCollapsed = !this.isNavbarCollapsed; + } + + getImageUrl(): string { + return this.isAuthenticated() ? this.accountService.getImageUrl() : ''; + } +} diff --git a/src/main/webapp/app/layouts/navbar/navbar.route.ts b/src/main/webapp/app/layouts/navbar/navbar.route.ts new file mode 100644 index 0000000..82c591f --- /dev/null +++ b/src/main/webapp/app/layouts/navbar/navbar.route.ts @@ -0,0 +1,9 @@ +import { Route } from '@angular/router'; + +import { NavbarComponent } from './navbar.component'; + +export const navbarRoute: Route = { + path: '', + component: NavbarComponent, + outlet: 'navbar', +}; diff --git a/src/main/webapp/app/layouts/navbar/navbar.scss b/src/main/webapp/app/layouts/navbar/navbar.scss new file mode 100644 index 0000000..aea859f --- /dev/null +++ b/src/main/webapp/app/layouts/navbar/navbar.scss @@ -0,0 +1,42 @@ +@import '~bootstrap/scss/functions'; +@import '~bootstrap/scss/variables'; + +/* ========================================================================== +Navbar +========================================================================== */ + +.navbar-version { + font-size: 0.65em; + color: $navbar-dark-color; +} + +.profile-image { + height: 1.75em; + width: 1.75em; +} + +.navbar { + padding: 0.2rem 1rem; + + ul.navbar-nav { + .nav-item { + margin-left: 0.5em; + } + } + + a.nav-link { + font-weight: 400; + } +} + +/* ========================================================================== +Logo styles +========================================================================== */ +.logo-img { + height: 45px; + width: 45px; + display: inline-block; + vertical-align: middle; + background: url('../../../content/images/ePASMedLogo.png') no-repeat center center; + background-size: contain; +} diff --git a/src/main/webapp/app/layouts/profiles/page-ribbon.component.ts b/src/main/webapp/app/layouts/profiles/page-ribbon.component.ts new file mode 100644 index 0000000..d263ad5 --- /dev/null +++ b/src/main/webapp/app/layouts/profiles/page-ribbon.component.ts @@ -0,0 +1,23 @@ +import { Component, OnInit } from '@angular/core'; +import { Observable } from 'rxjs'; +import { map } from 'rxjs/operators'; +import { ProfileService } from './profile.service'; + +@Component({ + selector: 'jhi-page-ribbon', + template: ` + + `, + styleUrls: ['page-ribbon.scss'], +}) +export class PageRibbonComponent implements OnInit { + ribbonEnv$?: Observable; + + constructor(private profileService: ProfileService) {} + + ngOnInit(): void { + this.ribbonEnv$ = this.profileService.getProfileInfo().pipe(map(profileInfo => profileInfo.ribbonEnv)); + } +} diff --git a/src/main/webapp/app/layouts/profiles/page-ribbon.scss b/src/main/webapp/app/layouts/profiles/page-ribbon.scss new file mode 100644 index 0000000..a78f268 --- /dev/null +++ b/src/main/webapp/app/layouts/profiles/page-ribbon.scss @@ -0,0 +1,31 @@ +/* ========================================================================== +Developement Ribbon +========================================================================== */ +.ribbon { + background-color: rgba(170, 0, 0, 0.5); + left: -3.5em; + -moz-transform: rotate(-45deg); + -ms-transform: rotate(-45deg); + -o-transform: rotate(-45deg); + -webkit-transform: rotate(-45deg); + transform: rotate(-45deg); + overflow: hidden; + position: absolute; + top: 40px; + white-space: nowrap; + width: 15em; + z-index: 9999; + pointer-events: none; + opacity: 0.75; + a { + color: #fff; + display: block; + font-weight: 400; + margin: 1px 0; + padding: 10px 50px; + text-align: center; + text-decoration: none; + text-shadow: 0 0 5px #444; + pointer-events: none; + } +} diff --git a/src/main/webapp/app/layouts/profiles/profile-info.model.ts b/src/main/webapp/app/layouts/profiles/profile-info.model.ts new file mode 100644 index 0000000..e7763ee --- /dev/null +++ b/src/main/webapp/app/layouts/profiles/profile-info.model.ts @@ -0,0 +1,15 @@ +export interface InfoResponse { + 'display-ribbon-on-profiles'?: string; + git?: any; + build?: any; + activeProfiles?: string[]; +} + +export class ProfileInfo { + constructor( + public activeProfiles?: string[], + public ribbonEnv?: string, + public inProduction?: boolean, + public swaggerEnabled?: boolean + ) {} +} diff --git a/src/main/webapp/app/layouts/profiles/profile.service.ts b/src/main/webapp/app/layouts/profiles/profile.service.ts new file mode 100644 index 0000000..9d2740e --- /dev/null +++ b/src/main/webapp/app/layouts/profiles/profile.service.ts @@ -0,0 +1,43 @@ +import { Injectable } from '@angular/core'; +import { HttpClient } from '@angular/common/http'; +import { map, shareReplay } from 'rxjs/operators'; +import { Observable } from 'rxjs'; + +import { SERVER_API_URL } from 'app/app.constants'; +import { ProfileInfo, InfoResponse } from './profile-info.model'; + +@Injectable({ providedIn: 'root' }) +export class ProfileService { + private infoUrl = SERVER_API_URL + 'management/info'; + private profileInfo$!: Observable; + + constructor(private http: HttpClient) {} + + getProfileInfo(): Observable { + if (this.profileInfo$) { + return this.profileInfo$; + } + + this.profileInfo$ = this.http.get(this.infoUrl).pipe( + map((response: InfoResponse) => { + const profileInfo: ProfileInfo = { + activeProfiles: response.activeProfiles, + inProduction: response.activeProfiles && response.activeProfiles.includes('prod'), + swaggerEnabled: response.activeProfiles && response.activeProfiles.includes('swagger'), + }; + if (response.activeProfiles && response['display-ribbon-on-profiles']) { + const displayRibbonOnProfiles = response['display-ribbon-on-profiles'].split(','); + const ribbonProfiles = displayRibbonOnProfiles.filter( + profile => response.activeProfiles && response.activeProfiles.includes(profile) + ); + if (ribbonProfiles.length > 0) { + profileInfo.ribbonEnv = ribbonProfiles[0]; + } + } + return profileInfo; + }), + shareReplay() + ); + return this.profileInfo$; + } +} diff --git a/src/main/webapp/app/polyfills.ts b/src/main/webapp/app/polyfills.ts new file mode 100644 index 0000000..f8a8d65 --- /dev/null +++ b/src/main/webapp/app/polyfills.ts @@ -0,0 +1,3 @@ +import 'zone.js/dist/zone'; +import '@angular/localize/init'; +require('../manifest.webapp'); diff --git a/src/main/webapp/app/shared/alert/alert-error.component.ts b/src/main/webapp/app/shared/alert/alert-error.component.ts new file mode 100644 index 0000000..149b4b2 --- /dev/null +++ b/src/main/webapp/app/shared/alert/alert-error.component.ts @@ -0,0 +1,113 @@ +import { Component, OnDestroy } from '@angular/core'; +import { HttpErrorResponse } from '@angular/common/http'; +import { JhiEventManager, JhiAlert, JhiAlertService, JhiEventWithContent } from 'ng-jhipster'; +import { Subscription } from 'rxjs'; + +import { AlertError } from './alert-error.model'; + +@Component({ + selector: 'jhi-alert-error', + template: ` `, +}) +export class AlertErrorComponent implements OnDestroy { + alerts: JhiAlert[] = []; + errorListener: Subscription; + httpErrorListener: Subscription; + + constructor(private alertService: JhiAlertService, private eventManager: JhiEventManager) { + this.errorListener = eventManager.subscribe('epasmedApp.error', (response: JhiEventWithContent) => { + const errorResponse = response.content; + this.addErrorAlert(errorResponse.message); + }); + + this.httpErrorListener = eventManager.subscribe('epasmedApp.httpError', (response: JhiEventWithContent) => { + const httpErrorResponse = response.content; + switch (httpErrorResponse.status) { + // connection refused, server not reachable + case 0: + this.addErrorAlert('Server not reachable'); + break; + + case 400: { + const arr = httpErrorResponse.headers.keys(); + let errorHeader = null; + arr.forEach(entry => { + if (entry.toLowerCase().endsWith('app-error')) { + errorHeader = httpErrorResponse.headers.get(entry); + } + }); + if (errorHeader) { + this.addErrorAlert(errorHeader); + } else if (httpErrorResponse.error !== '' && httpErrorResponse.error.fieldErrors) { + const fieldErrors = httpErrorResponse.error.fieldErrors; + for (const fieldError of fieldErrors) { + if (['Min', 'Max', 'DecimalMin', 'DecimalMax'].includes(fieldError.message)) { + fieldError.message = 'Size'; + } + // convert 'something[14].other[4].id' to 'something[].other[].id' so translations can be written to it + const convertedField = fieldError.field.replace(/\[\d*\]/g, '[]'); + const fieldName = convertedField.charAt(0).toUpperCase() + convertedField.slice(1); + this.addErrorAlert('Error on field "' + fieldName + '"'); + } + } else if (httpErrorResponse.error !== '' && httpErrorResponse.error.message) { + this.addErrorAlert(httpErrorResponse.error.message); + } else { + this.addErrorAlert(httpErrorResponse.error); + } + break; + } + + case 404: + this.addErrorAlert('Not found'); + break; + + default: + if (httpErrorResponse.error !== '' && httpErrorResponse.error.message) { + this.addErrorAlert(httpErrorResponse.error.message); + } else { + this.addErrorAlert(httpErrorResponse.error); + } + } + }); + } + + setClasses(alert: JhiAlert): { [key: string]: boolean } { + const classes = { 'jhi-toast': Boolean(alert.toast) }; + if (alert.position) { + return { ...classes, [alert.position]: true }; + } + return classes; + } + + ngOnDestroy(): void { + if (this.errorListener) { + this.eventManager.destroy(this.errorListener); + } + if (this.httpErrorListener) { + this.eventManager.destroy(this.httpErrorListener); + } + } + + addErrorAlert(message: string): void { + const newAlert: JhiAlert = { + type: 'danger', + msg: message, + timeout: 5000, + toast: this.alertService.isToast(), + scoped: true, + }; + + this.alerts.push(this.alertService.addAlert(newAlert, this.alerts)); + } + + close(alert: JhiAlert): void { + // NOSONAR can be removed after https://github.com/SonarSource/SonarJS/issues/1930 is resolved + alert.close?.(this.alerts); // NOSONAR + } +} diff --git a/src/main/webapp/app/shared/alert/alert-error.model.ts b/src/main/webapp/app/shared/alert/alert-error.model.ts new file mode 100644 index 0000000..2b8cb8f --- /dev/null +++ b/src/main/webapp/app/shared/alert/alert-error.model.ts @@ -0,0 +1,3 @@ +export class AlertError { + constructor(public message: string) {} +} diff --git a/src/main/webapp/app/shared/alert/alert.component.ts b/src/main/webapp/app/shared/alert/alert.component.ts new file mode 100644 index 0000000..794de59 --- /dev/null +++ b/src/main/webapp/app/shared/alert/alert.component.ts @@ -0,0 +1,39 @@ +import { Component, OnDestroy, OnInit } from '@angular/core'; +import { JhiAlertService, JhiAlert } from 'ng-jhipster'; + +@Component({ + selector: 'jhi-alert', + template: ` `, +}) +export class AlertComponent implements OnInit, OnDestroy { + alerts: JhiAlert[] = []; + + constructor(private alertService: JhiAlertService) {} + + ngOnInit(): void { + this.alerts = this.alertService.get(); + } + + setClasses(alert: JhiAlert): { [key: string]: boolean } { + const classes = { 'jhi-toast': Boolean(alert.toast) }; + if (alert.position) { + return { ...classes, [alert.position]: true }; + } + return classes; + } + + ngOnDestroy(): void { + this.alertService.clear(); + } + + close(alert: JhiAlert): void { + // NOSONAR can be removed after https://github.com/SonarSource/SonarJS/issues/1930 is resolved + alert.close?.(this.alerts); // NOSONAR + } +} diff --git a/src/main/webapp/app/shared/auth/has-any-authority.directive.ts b/src/main/webapp/app/shared/auth/has-any-authority.directive.ts new file mode 100644 index 0000000..5b10e1d --- /dev/null +++ b/src/main/webapp/app/shared/auth/has-any-authority.directive.ts @@ -0,0 +1,47 @@ +import { Directive, Input, TemplateRef, ViewContainerRef, OnDestroy } from '@angular/core'; +import { Subscription } from 'rxjs'; + +import { AccountService } from 'app/core/auth/account.service'; + +/** + * @whatItDoes Conditionally includes an HTML element if current user has any + * of the authorities passed as the `expression`. + * + * @howToUse + * ``` + * ... + * + * ... + * ``` + */ +@Directive({ + selector: '[jhiHasAnyAuthority]', +}) +export class HasAnyAuthorityDirective implements OnDestroy { + private authorities: string[] = []; + private authenticationSubscription?: Subscription; + + constructor(private accountService: AccountService, private templateRef: TemplateRef, private viewContainerRef: ViewContainerRef) {} + + @Input() + set jhiHasAnyAuthority(value: string | string[]) { + this.authorities = typeof value === 'string' ? [value] : value; + this.updateView(); + // Get notified each time authentication state changes. + this.authenticationSubscription = this.accountService.getAuthenticationState().subscribe(() => this.updateView()); + } + + ngOnDestroy(): void { + if (this.authenticationSubscription) { + this.authenticationSubscription.unsubscribe(); + } + } + + private updateView(): void { + const hasAnyAuthority = this.accountService.hasAnyAuthority(this.authorities); + this.viewContainerRef.clear(); + if (hasAnyAuthority) { + this.viewContainerRef.createEmbeddedView(this.templateRef); + } + } +} diff --git a/src/main/webapp/app/shared/constants/authority.constants.ts b/src/main/webapp/app/shared/constants/authority.constants.ts new file mode 100644 index 0000000..1501bcf --- /dev/null +++ b/src/main/webapp/app/shared/constants/authority.constants.ts @@ -0,0 +1,4 @@ +export enum Authority { + ADMIN = 'ROLE_ADMIN', + USER = 'ROLE_USER', +} diff --git a/src/main/webapp/app/shared/constants/error.constants.ts b/src/main/webapp/app/shared/constants/error.constants.ts new file mode 100644 index 0000000..42d91d9 --- /dev/null +++ b/src/main/webapp/app/shared/constants/error.constants.ts @@ -0,0 +1,3 @@ +export const PROBLEM_BASE_URL = 'https://www.jhipster.tech/problem'; +export const EMAIL_ALREADY_USED_TYPE = PROBLEM_BASE_URL + '/email-already-used'; +export const LOGIN_ALREADY_USED_TYPE = PROBLEM_BASE_URL + '/login-already-used'; diff --git a/src/main/webapp/app/shared/constants/input.constants.ts b/src/main/webapp/app/shared/constants/input.constants.ts new file mode 100644 index 0000000..1e3978a --- /dev/null +++ b/src/main/webapp/app/shared/constants/input.constants.ts @@ -0,0 +1,2 @@ +export const DATE_FORMAT = 'YYYY-MM-DD'; +export const DATE_TIME_FORMAT = 'YYYY-MM-DDTHH:mm'; diff --git a/src/main/webapp/app/shared/constants/pagination.constants.ts b/src/main/webapp/app/shared/constants/pagination.constants.ts new file mode 100644 index 0000000..a148d45 --- /dev/null +++ b/src/main/webapp/app/shared/constants/pagination.constants.ts @@ -0,0 +1 @@ +export const ITEMS_PER_PAGE = 20; diff --git a/src/main/webapp/app/shared/login/login.component.html b/src/main/webapp/app/shared/login/login.component.html new file mode 100644 index 0000000..1c7e9e5 --- /dev/null +++ b/src/main/webapp/app/shared/login/login.component.html @@ -0,0 +1,47 @@ + + + diff --git a/src/main/webapp/app/shared/login/login.component.ts b/src/main/webapp/app/shared/login/login.component.ts new file mode 100644 index 0000000..2659222 --- /dev/null +++ b/src/main/webapp/app/shared/login/login.component.ts @@ -0,0 +1,73 @@ +import { Component, AfterViewInit, ElementRef, ViewChild } from '@angular/core'; +import { FormBuilder } from '@angular/forms'; +import { NgbActiveModal } from '@ng-bootstrap/ng-bootstrap'; +import { Router } from '@angular/router'; + +import { LoginService } from 'app/core/login/login.service'; + +@Component({ + selector: 'jhi-login-modal', + templateUrl: './login.component.html', +}) +export class LoginModalComponent implements AfterViewInit { + @ViewChild('username', { static: false }) + username?: ElementRef; + + authenticationError = false; + + loginForm = this.fb.group({ + username: [''], + password: [''], + rememberMe: [false], + }); + + constructor(private loginService: LoginService, private router: Router, public activeModal: NgbActiveModal, private fb: FormBuilder) {} + + ngAfterViewInit(): void { + if (this.username) { + this.username.nativeElement.focus(); + } + } + + cancel(): void { + this.authenticationError = false; + this.loginForm.patchValue({ + username: '', + password: '', + }); + this.activeModal.dismiss('cancel'); + } + + login(): void { + this.loginService + .login({ + username: this.loginForm.get('username')!.value, + password: this.loginForm.get('password')!.value, + rememberMe: this.loginForm.get('rememberMe')!.value, + }) + .subscribe( + () => { + this.authenticationError = false; + this.activeModal.close(); + if ( + this.router.url === '/account/register' || + this.router.url.startsWith('/account/activate') || + this.router.url.startsWith('/account/reset/') + ) { + this.router.navigate(['']); + } + }, + () => (this.authenticationError = true) + ); + } + + register(): void { + this.activeModal.dismiss('to state register'); + this.router.navigate(['/account/register']); + } + + requestResetPassword(): void { + this.activeModal.dismiss('to state requestReset'); + this.router.navigate(['/account/reset', 'request']); + } +} diff --git a/src/main/webapp/app/shared/shared-libs.module.ts b/src/main/webapp/app/shared/shared-libs.module.ts new file mode 100644 index 0000000..d2d77f7 --- /dev/null +++ b/src/main/webapp/app/shared/shared-libs.module.ts @@ -0,0 +1,12 @@ +import { NgModule } from '@angular/core'; +import { CommonModule } from '@angular/common'; +import { FormsModule, ReactiveFormsModule } from '@angular/forms'; +import { NgbModule } from '@ng-bootstrap/ng-bootstrap'; +import { NgJhipsterModule } from 'ng-jhipster'; +import { InfiniteScrollModule } from 'ngx-infinite-scroll'; +import { FontAwesomeModule } from '@fortawesome/angular-fontawesome'; + +@NgModule({ + exports: [FormsModule, CommonModule, NgbModule, NgJhipsterModule, InfiniteScrollModule, FontAwesomeModule, ReactiveFormsModule], +}) +export class EpasmedSharedLibsModule {} diff --git a/src/main/webapp/app/shared/shared.module.ts b/src/main/webapp/app/shared/shared.module.ts new file mode 100644 index 0000000..f91522e --- /dev/null +++ b/src/main/webapp/app/shared/shared.module.ts @@ -0,0 +1,14 @@ +import { NgModule } from '@angular/core'; +import { EpasmedSharedLibsModule } from './shared-libs.module'; +import { AlertComponent } from './alert/alert.component'; +import { AlertErrorComponent } from './alert/alert-error.component'; +import { LoginModalComponent } from './login/login.component'; +import { HasAnyAuthorityDirective } from './auth/has-any-authority.directive'; + +@NgModule({ + imports: [EpasmedSharedLibsModule], + declarations: [AlertComponent, AlertErrorComponent, LoginModalComponent, HasAnyAuthorityDirective], + entryComponents: [LoginModalComponent], + exports: [EpasmedSharedLibsModule, AlertComponent, AlertErrorComponent, LoginModalComponent, HasAnyAuthorityDirective], +}) +export class EpasmedSharedModule {} diff --git a/src/main/webapp/app/shared/util/datepicker-adapter.ts b/src/main/webapp/app/shared/util/datepicker-adapter.ts new file mode 100644 index 0000000..027f123 --- /dev/null +++ b/src/main/webapp/app/shared/util/datepicker-adapter.ts @@ -0,0 +1,23 @@ +/** + * Angular bootstrap Date adapter + */ +import { Injectable } from '@angular/core'; +import { NgbDateAdapter, NgbDateStruct } from '@ng-bootstrap/ng-bootstrap'; +import { Moment } from 'moment'; +import * as moment from 'moment'; + +@Injectable() +export class NgbDateMomentAdapter extends NgbDateAdapter { + fromModel(date: Moment): NgbDateStruct { + if (date && moment.isMoment(date) && date.isValid()) { + return { year: date.year(), month: date.month() + 1, day: date.date() }; + } + // ! can be removed after https://github.com/ng-bootstrap/ng-bootstrap/issues/1544 is resolved + return null!; + } + + toModel(date: NgbDateStruct): Moment { + // ! after null can be removed after https://github.com/ng-bootstrap/ng-bootstrap/issues/1544 is resolved + return date ? moment(date.year + '-' + date.month + '-' + date.day, 'YYYY-MM-DD') : null!; + } +} diff --git a/src/main/webapp/app/shared/util/request-util.ts b/src/main/webapp/app/shared/util/request-util.ts new file mode 100644 index 0000000..f839fd6 --- /dev/null +++ b/src/main/webapp/app/shared/util/request-util.ts @@ -0,0 +1,33 @@ +import { HttpParams } from '@angular/common/http'; + +export interface Pagination { + page: number; + size: number; + sort: string[]; +} + +export interface Search { + query: string; +} + +export interface SearchWithPagination extends Search, Pagination {} + +export const createRequestOption = (req?: any): HttpParams => { + let options: HttpParams = new HttpParams(); + + if (req) { + Object.keys(req).forEach(key => { + if (key !== 'sort') { + options = options.set(key, req[key]); + } + }); + + if (req.sort) { + req.sort.forEach((val: string) => { + options = options.append('sort', val); + }); + } + } + + return options; +}; diff --git a/src/main/webapp/app/vendor.ts b/src/main/webapp/app/vendor.ts new file mode 100644 index 0000000..a89c6da --- /dev/null +++ b/src/main/webapp/app/vendor.ts @@ -0,0 +1,2 @@ +/* after changing this file run 'npm run webpack:build' */ +import '../content/scss/vendor.scss'; diff --git a/src/main/webapp/content/css/loading.css b/src/main/webapp/content/css/loading.css new file mode 100644 index 0000000..ba83452 --- /dev/null +++ b/src/main/webapp/content/css/loading.css @@ -0,0 +1,152 @@ +@keyframes lds-pacman-1 { + 0% { + -webkit-transform: rotate(0deg); + transform: rotate(0deg); + } + 50% { + -webkit-transform: rotate(-45deg); + transform: rotate(-45deg); + } + 100% { + -webkit-transform: rotate(0deg); + transform: rotate(0deg); + } +} +@-webkit-keyframes lds-pacman-1 { + 0% { + -webkit-transform: rotate(0deg); + transform: rotate(0deg); + } + 50% { + -webkit-transform: rotate(-45deg); + transform: rotate(-45deg); + } + 100% { + -webkit-transform: rotate(0deg); + transform: rotate(0deg); + } +} +@keyframes lds-pacman-2 { + 0% { + -webkit-transform: rotate(180deg); + transform: rotate(180deg); + } + 50% { + -webkit-transform: rotate(225deg); + transform: rotate(225deg); + } + 100% { + -webkit-transform: rotate(180deg); + transform: rotate(180deg); + } +} +@-webkit-keyframes lds-pacman-2 { + 0% { + -webkit-transform: rotate(180deg); + transform: rotate(180deg); + } + 50% { + -webkit-transform: rotate(225deg); + transform: rotate(225deg); + } + 100% { + -webkit-transform: rotate(180deg); + transform: rotate(180deg); + } +} +@keyframes lds-pacman-3 { + 0% { + -webkit-transform: translate(190px, 0); + transform: translate(190px, 0); + opacity: 0; + } + 20% { + opacity: 1; + } + 100% { + -webkit-transform: translate(70px, 0); + transform: translate(70px, 0); + opacity: 1; + } +} +@-webkit-keyframes lds-pacman-3 { + 0% { + -webkit-transform: translate(190px, 0); + transform: translate(190px, 0); + opacity: 0; + } + 20% { + opacity: 1; + } + 100% { + -webkit-transform: translate(70px, 0); + transform: translate(70px, 0); + opacity: 1; + } +} + +.app-loading { + font-family: 'Helvetica Neue', Helvetica, Arial, sans-serif; + position: relative; + display: flex; + flex-direction: column; + align-items: center; + justify-content: center; + top: 10em; +} +.app-loading p { + display: block; + font-size: 1.17em; + margin-inline-start: 0px; + margin-inline-end: 0px; + font-weight: normal; +} + +.app-loading .lds-pacman { + position: relative; + margin: auto; + width: 200px !important; + height: 200px !important; + -webkit-transform: translate(-100px, -100px) scale(1) translate(100px, 100px); + transform: translate(-100px, -100px) scale(1) translate(100px, 100px); +} +.app-loading .lds-pacman > div:nth-child(2) div { + position: absolute; + top: 40px; + left: 40px; + width: 120px; + height: 60px; + border-radius: 120px 120px 0 0; + background: #bbcedd; + -webkit-animation: lds-pacman-1 1s linear infinite; + animation: lds-pacman-1 1s linear infinite; + -webkit-transform-origin: 60px 60px; + transform-origin: 60px 60px; +} +.app-loading .lds-pacman > div:nth-child(2) div:nth-child(2) { + -webkit-animation: lds-pacman-2 1s linear infinite; + animation: lds-pacman-2 1s linear infinite; +} +.app-loading .lds-pacman > div:nth-child(1) div { + position: absolute; + top: 97px; + left: -8px; + width: 24px; + height: 10px; + background-image: url('../images/ePASMedLogo.png'); + background-size: contain; + -webkit-animation: lds-pacman-3 1s linear infinite; + animation: lds-pacman-3 1.5s linear infinite; +} +.app-loading .lds-pacman > div:nth-child(1) div:nth-child(1) { + -webkit-animation-delay: -0.67s; + animation-delay: -1s; +} +.app-loading .lds-pacman > div:nth-child(1) div:nth-child(2) { + -webkit-animation-delay: -0.33s; + animation-delay: -0.5s; +} +.app-loading .lds-pacman > div:nth-child(1) div:nth-child(3) { + -webkit-animation-delay: 0s; + animation-delay: 0s; +} diff --git a/src/main/webapp/content/images/ePASMedLogo.png b/src/main/webapp/content/images/ePASMedLogo.png new file mode 100644 index 0000000000000000000000000000000000000000..50e35683b84e9a7528add11d688fbb20a3d42cf5 GIT binary patch literal 7181 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 + + + + + + + + + image/svg+xml + + + + + + + + + diff --git a/src/main/webapp/content/images/plumber_0_head_192.png b/src/main/webapp/content/images/plumber_0_head_192.png new file mode 100644 index 0000000000000000000000000000000000000000..be4e9066937943a67f4e0fbb2c68841ffa427eb5 GIT binary patch literal 20533 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�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 literal 0 HcmV?d00001 diff --git a/src/main/webapp/content/images/plumber_0_head_256.png b/src/main/webapp/content/images/plumber_0_head_256.png new file mode 100644 index 0000000000000000000000000000000000000000..5886696f75be365ab9fe7e2a0a995d285198696e GIT binary patch literal 31122 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 literal 0 HcmV?d00001 diff --git a/src/main/webapp/content/images/plumber_0_head_384.png b/src/main/webapp/content/images/plumber_0_head_384.png new file mode 100644 index 0000000000000000000000000000000000000000..e27221bfb8528a88a4fabe06812af23518c98fdc GIT binary patch literal 59863 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

CNLwCa5KMp(GodXV)DkWF=2~p6vc#U5ME3a= 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

pXQg&cRp($sx>`~n?4 zUnmKL(%IK(w~D&^i=WCwB0I_N(Nk<{k^zrLIKm=gi>4gSJdiUkyet>nArp9yi!lGHRuVi(ZtV*xNC4P?1wXel?Q+?}0dQqP{WsPC|MNin z*X8_eEpi-dh|FSbN*Zc&Pk)FdTSG%5myzi!XatnO`1_;e!(z!tM>YZGW;;`*U2Z_k zCU!!&w-!J;SO@eOA|pCm%%nN?gz6jml2_Feefukkca1Bl!s=*sQsP48B0t4>1)YFQ z=sNb#o|mUr3!GOh@!?h0GY_Hk%lFSNEd9K`o>gQI?Q-NKt)h+p^oC%2>hN9{?Yr7^ zY9=7@A68OH7KtQOS}FqbXDHiwRjZ10IKuS_i(M6}I)d%oM=XT5=%`+6doRWseIV5| zZ4(w8_ps8Bh);lY|v949Iu%L;>xDJJ{=;LO$ucTod%mzN-5P7 z@--H`W%PZaAlXv(?enGUy2R_Po|R`vlLq>qp{N%xHV`)aO}7S&J+rq{C&=uC3TSG) ztQCr15sp~%WB426LA5EW+#+7~)jAd3?5}>ru2nR1JYK|Fk;;X*D;_s6mCaLs{dxx~ za&^46^AGSoyTm5KUvGe0w zgMCSfVQ!TY0#dl~-c_>jPD$v2#el<-JbJCR|jSkD>q`(gRZj-7Da(BW^x8dV7EYy!4 zv^ppTvd2tSE5R(%Q69$D$twHG`5Fuac! z`*yCuvYJiX6*hZsgK>f}{W}vfA-Y zpyfxIUozN}SzP4$scSnsP>egjud)4~-PbWEVaP#(Sb2I*p0GsDJ~K{AItYDaiGQK< z(VuS%ZXDs~A(BkXO2jf)CuZ6V4!T6hk>o%ipy~3YWBot+iTg2bMjSrmNvL`yp|R;1 zmdvI{!X#tLB=of|)^gMn&8A`zGiN#{H>$Ku;@Xad3gP|*D$2fl_g8f?eU0;@I4NvCGw%-Htzslf}sFPlAwIzRW?YvG5SS<{nrN7Pq~)> z&SKE%_jGQF4Bmr$kJ1ho3O71n8K@8jrhW0b7HLMH!+HghLZT_M#xWY5X;4UO4*k?5CJb1(D;^<5l zooqii6&9fBUGxnP02{^ex1j|2qK~Dz>{2&P^bd~03lD;%c$@}H=Vor;ZjdULgd*!(K{co>a;&*Jfarq1 zQyK4a%AGjZ%`Y~e)m2C6v7iUlN`EgbV=NJ?MCj3s3rS9G$-YlRrSm1sdKFRf{@!m~ zcOBn+qDN(u9)eh^)^q)`+}y^svVwN-_^e^H6I}@SUp=K<>R~;<3;Bfl!If{#_wg*X z-g+b)0DhT`dy*^{2OG)*!q7Ps2pFF|jwS-!Q*-O`7eV>@5l;W1Xsf+21xA~xr%Ltq zP5fhRA!2{9^+&i^NT(yxc_B<@>0*oV5#ypMr!5h+>H_uqw4k^!iXl?HfC%tU99}?W zljzMx;4{4*2@g;;Te-+tcznY&`=RtdKw&fhMzi#=e0(WjdQSupLFhbXOug`p<%^ki zQQ)D?h5gxxS1!v4ByW@$oMa0pZjO(R(O zp{1Xobm_u`gVy~`JJF~S%67t^ws=sdecZ_o>BqY}L1EEePwoIaCL@tk(P6Xi!Cd5h zR=#awx1gfG3KgLc6!><`U{7Q+Hm=cG+H2$H|^`v1n0IJCpzQ>!$)lw zL@1rQ5~pti)t-uQ-o8c7v!%hcF$}pcS6@$FC;+B|4Ec1weDX`bP_p0N z%^ZSd;$BYf-bfd7E#&HG z@#pdw&ad6yLgG@N=>wWlP*FJzbPIxoJj=SK^CgU4QT9LMDU-+F9bEdN$~cgPv30Ap zC0lV1Gs_~QrG>ux80kwP!yvt?Y{%t|`xHJwOWy4$fLY&*DHt81?Q!!MNtO6N`IV3K z*X>Y4SO6@TN<%NKkF{>R3QRJa6iojV_({K6dicEIEO%p2_xGU4n3^z+;73FoYu2i( z3k`BbzsAZZ6dgIubrYlakwOoX&%d25ja1em@$R*!< z^Rr}Wjbg&}z{*oSM)2+C7p`WS`ujrZtOQ(GH%N+<9*Um#J(7?%O1f83tLPjROq*d> zzr-QQW;iPNBQ%FN|KU^qZ)K8>M2FwQXAmS-JsrQPsEiAeka^(0S3iChmud?b8}yw_ za{XfcXg#ySWmLfp934Nsx1{+bV#3aqIdK#CHve%-J-s43vZgWqXgbCq;#1z>|8iSPi3}x3dnpIVkuxPj{(dVGU zNa0GP_>FMbtPvJiUv|NLWB?#IJvLMsFoF&6!{-bI7GG85S7V6+)SW=M1!N%$G|p<0ERBYdTtDM28cLSf;0EGTH%E6vXd|Kh5AE z4)zjli)H?t{ork^_b=0-rAgVK{q2wmB=Umx8V?IU2Dts}%F*Ony2^R+bb{4-TLQ1g z0e~6AFZEQw-gAJZ+OgwafucGRQ9>lZW&`Ie8~Z{OS1mW%|Aawo@i!$nkwn+x7@C zEEsk~N9H`txvRQf7&x&D-$sP}C$W$W8Fo~kE_pX+QhDiXCKq?R68%i3AP>=ku!@hr^T2v0gV9ag1< z>2Eh0GBl+>j4VjwUYKa1bSjepy|+-1N^N4Lv6Ss$`h36xPuJ>RMS$TGo1tdf_&ok_ z&kZC!MBxUU;(Z&$eAAPFCn1EdLE*#le4POZ*wK1b2I!Z^P$#R0<_8S_cmU>G9eA-7Q0vum z4ERqH^R&8Jk@nxDAM3M`57LoL!uQ1aTC4PU=izISL~Y24Dh-y=jE#e{fy}{W3}J$u zKB6Jhw^2#>w9n1hiZ{4soAIO^p7HmhcQXe#K8yA$Tl)GrpZ%?n{oBtCi7>oQ z1Hb;g({cFZ%AsSMD@_nmu6)v<#CKM1XyCVlj}cwP?udrk)@-0C+4lJ7Agw?9YLcX~ zJGdV;q=E$1VBSaOqU5yWXZ^{TdcLagnbabdOKZ`4&iL?CBx0D=hQ{izjhVK}a7w7ZKg}T*du>z<=%vK;$;Ho{0+<;5 z?Rh;>tM_x${^DP#Vv%27~dmSgl*ciVlg-3;HyQl1=K}i0- zO7jiXf;2cRScViWlD_fnn8AY9muf|TsW1M7pxD+Af@+boa1C%uw^unh&>*C9?Q8Doc`86#$CtRjQ*YoiJ%6-1 zk8vtPO zKqW{gI^qxWbOl!yb0#R7&PGu|3btgSj%(hrz#xnI>SgK2eMgv}2ocz3j&Oy*oO zB5&q9bpZfO?hiP8TDG2Ehb&gb7x3*$T*Vjk72@a){qnkyYsKRyTw!VOC(4FfL})1H z(TYU((K46ZC&+uvZtho#!OY+q{)5$_&%lPp+u6k(T1e=)DOH zOc;TRB!IoiJO&5c{}@&oFv#TabaqmC6)zNq$Pj!BR&%Em1QPRlJ9y# zEHmwMQ&6O_N+-(Dr)5C!1Dv(4Ho)&hfUXJia4pMAV>uMgs~=|A=(IbKp*3ZijJ{BK zG5j{#?3~zecpQG(B{Q z|EeraMdJd2Y~HQVjk>3Z%qxD0z+)xgpJ{H7Po`H{g}$_Qk-M@$WX4Y3QnJjA}-W#zOL zyokJ-uKI93+I5j|aIGHkgb5_l{%0IhS1XT(0wV}R8Xk<}s=C3g%s_&R88TG#@BD&- znX8wFVv8+-kp+{4BM_E~zdXR+I`R(wSgkLDf?rqFJcAf;YLMNb)z`{2{NjXKbg7tO zP^gyh2LS$8wLYXqFvlYb z(d502(^p{GI2`1(pPgT{)qBGzm2RBD_4%x^$zPkU4q_zch9@J2tdR4D4@MuZ+7T2o z=k1R&*AQgP{Zxq^#eClOLN72PtVudfx%j!E49MzzR92Sd!C@R%`d?-xR9Z}#eD;Ng z+dE^%T>d-jx18C31;y3e{4?ni@y&B>M(sfpm`Mp3Ta>F7GO98#qTH!{ktEVG=uf}d zg$1wCU?#5(-MTvxH&ylzD*;YNAAVa$4cb90Aw#vjeOrtTRgU!0>w8tJ(f?i_W@FFe z$v9&sa-kbI=s9F0_gg>rS7V%~+c}@?xP4)jk|H2+6@vz#Y&}y2blz6bT+!hD$E4io z33T}jkN6KJ?^u0x8=6K;Xt$2a-Pq7y=}_%Vy^U|V!ub}+8O-0HuO+Qq;3wKR{rP2q zNec0XVth{li?CZ5Oyil`KUF|U%i}u-Mz(wUU}!0zc1E7CAK4r$;xYpiGgm02ll&4U z@lg0pM}7FM?UrS@Hi(MC&Iwl7JKra~Zg&Y9E$m0-rzD5xDL*S1EusJ#efFRQNQYD6 zHh6a<`nqNwZ&nC$u75nv`}Y5rzQdspJ1bk~6^~yc_@>h<>%AO_<7Wmx*q8~kxw>r6 z{&^Ub){&Mp%dXz!PjF&JX^JL2Guqf;{&`yyvzDuEjjIv^ns`nJY?aJ#wZI8840B}B z*2>Cy*$4)iM|QbJehKe1|3B%eKuJ4CDuh}|_pwm3U2Z*!EpqXz1!*{xVX>n4CZ{m9 z$K{iHA1i8ZeMA|!bF(nwAe|*#hD+%F?4->b;g+o4uBQ_3A3pMg+8%PQONVK1$yOG= z2At{o8QFl0M`W#>W~|_{Wo*S$lpAi*t;`vSWA&>~S2%}@q#itQ0_*5~l9i^zok>xvx>vik$+N^TnUaN{L~c}?q_4nH$VUDz z&%I)BIUyoc?M~D_1et?C`GQg|H4*SpC&=`4ZecooWUr(lUG9UAq#&B4Q~F`3<6&mV z-{)=*rSCXjq2nd55rr+)GtUM5SP5J!aBKF92{OH6=zLmfY|^jpw;{;aQl)Ff)cF24 z%m6R{8?pYIeCF?lBfq#;ZE=++zXyCXpS4khTDBt|MLqq_Fjv^BrQD^C*Q@Rt_UJV3 zLN>3{z8U$g{XqQ^EKS!8qlLW+=Zycaeo;1$eXDb;<|S7mh)8Ee9+|aya(@2CiTx`) z*%Kq*z}KjjcPCzogLcp0%6gV&$ZG*iUhujU1@Z9zq5p~$-*W2fB110PDASoA;zC zfnRa*Ad_E`8smgq&fpIYqxsBqkHgl_T08IjW*6tXZ~GYYXWtV3o{1#De-#4?p6j+_ z0=E8c^+|()K$v=#S9ODvG$gXOcOu94a{>}KGmmPR0Mp0Ajs*4lgTSM>j3@Nj4i)oB z$m0oifAMvJcFUih8n25q=50Ft@Q~7uHP^hTyv~$Q0tVlm*n)c}Xi{lvJk#c`T(Rlw zB~~aaN0RmPTBCyBDvfk^ zteUytZcKvpQ(wpVmE`bMFX~#+TCdVQ!e5eBIl*>4({}>O29JS;*lA1mftrNOG45|S zPd3dD)J%~gefL<F$I_s04?_D(iG7q~ukNng6VDckBSxZvx zq_pv7UF~?UqCd1k0PWu+DM0-8>ZxYQ9E%bP8LfWxQ{czf)=o#ik%nIz8TILIJk_dy zzOLk(tO!wgSx*!aM$(5%4)#Swm7it29F}JaPaluG+kVddD$WN7%$e%ulevx<<2qUv z={-6EBT*sW_2TGuQi+9OP`sfr!e_IH&C%tGYo}l152f*Xb#B_9RMhQ34|dZEY=cvZjH z=Xvsi;nPKrxXE*fB8#g#uS$5+j(6S&ju(UlU zJ#KFa74c5U%D31Z;*bVN4@}q134?o2CPDh?QcB&^(@^tV*#1u7PwAh&7!&JiH$oqs zm-hEOg^?2VNIt2_Vs@cV4GXj&&n$WGhpJ$69M(`3T952;k<`q*UBstyE3Hb zB9R3(Zrnyrl?B@Qvr~)5?h;+KG4C^bL(-)cUlK^7NCT$cMrA#4IWGx!-Zfq#8CD*v z)93Z7sYOxCH=-UpeVo5^Hm;d}A`sxbWYY8e^1-o&D!8vaaTr2Pc-Z$|0}bf=&~CVg zDoyy4G%nuABex#T>t6(v0u`a0W1u=Zx9C4I$4fDkyK7e5BOHoy7#2qxm5O(hQgkp z+ThRm%&sSGxMR|fsSnHL^Wr=jghHpH9w?F|%=Dsn44qzYu!y7v>VXS; z`HByI|08HOb-oF434fj*D>7V6cIf&3JG!iPPTq^=-(=_ zQQpf-xh5|MjJrE}J!odt=wQ4V`yRZUUAD|kbSSad`t0hkZathz#5OLoOqgDi?_9Ag zw;P~Ixq--P`MUS5Vtm0~IB%+J`jUTIV0v=bv%I)Nf6z=*(C)-*m7vi8)cD-!;2D}fT1;A zVYM6FcT#B+XhKZ_Fw`4XtFJLgM9pU^7ts3xuWRicI|5%F(X6ey#`3;|S&6_$Z3Cp4 z1~X~Lhr@TW)$~|A<9)s>9b(hsF`-B+mH;)$nTryOwe>8Kp5e)f+bu?9@rbA^DVI!1ps8Jbi%Ab^?M0xT+g?O47#m z-@!GSlptDp%=#KL$)9n`R}*D}uJ?V>o2p|b&M4pgZXAT{TgDQ z#sFqOT5@ZVB9O0QBz%;*%9zSbFmlPI!r|xZsGxMFvh3r`cO9t%ZgE!5sjVX(iO|S( zoUr7>XBY`Ah@2I>ZLLSL<&L&czbl;F`nOdvU#L%l94Z0^>_R=$3z&zdS7shicxPC> z@oLwXiMI!OV7>FMNOqaGu)n_C*23AByKrdn!2>&V4JrlYw+};dg15dbG%Tec{Pc`l4s zWlXldHTrUP`KaJin1+^_&BDus$3*e8ZO|#%8;6~I8U4RdE$Myye66;Eej$tr$2E`B z!B1`x1sj9~iounfW{hBAx5t&;-Vl&ZF02s1Gs*$>m9Z5hn0ND|E!1)@Q>QFgRz2<` zDsw|H#ci>PZ!0ncRm=q%6C<${*;)7S9peVH>YSv1`1O@d0SmN5YxP@*{IK;dgJ0xU z6@PuekmJ92j0afta}=-!1j}6OlhT_#gZ*}V*QU?14~>iae52>J`F1)C*9pS-RKZ2CzRayh$J8v*9?M!XLl0vJ%ah4Xt+splmqsjiZ!nfq{>mqD!!+PXiJq1tQB8YCYuqW{zX~jQ7jMcuqi3mC7Ma||FYKUO zVD(^7^U#-|m?sSg9H-!XdbzRq(BMe?pr8WsgvJAygOMrLX`<`%p?E|n$PHgg325mvUDp0P+31{X~d4%Dy@Jz?m#N^YUTN~QfN+dsd!|5^#Q8wP$=Z}F3z zdf01p?-m96=j!|I6<3x-i7XU)8?>B*&7 zf?|7DZC{93YZRBb+wfsdvW{m8;Rhk|@_j9Zh zh^bpr%HGD@m2?qQ0tYvo9(~sMx+1a zObV~Rrex}$5ett_Sus%|f1FyHtXBm+clZ+eHH)lBWWHEZ!H3DxZ4tF3sSn?XBT2Mq@pz=-F9d9ayK5) z6!1!NJcSpLFv<#MgtO@4EhFmHe$J1F3l1RC@71gB*Ylr{Eab11lV zKKuRiA7fQ&yFY(l;@AAGvKh;vDB-#sThG8D4cfA1C(!dT)oe5fLQ$ zt>HoKS^Z2Vv1(ry5H@tZPN!TsN^&o4GX09}*E{XvY6mp`l6g+z50+j1Q3kSDJN(__ znnV6kKyxY;=7j1BD&K`i@!T?+vvC2-k%?GZXdZa{ zJ5M2T!v`R6Y5T9YnA7ukEcGBsbf$$O^uF3EDX0rCDq##4G{z$4GgVZ-dtq3K_tAyIjKVl$_7l?XpJe1-}R zDck`U(ORfW1GL0zv|(K^KiW4VvdK8Ll$w4kv!ex3NU(l;;pOOfNn7+a3dnZh=IY?P zvFny!L5{p-KtF;V%j#`3-Ef|=vFzaZ9ckx*hXp(w7QTi}^KC`~fkx0P2E12acKvTM zDByRJ1P}I3kJ_be$0ceeRY%{*!U!ulMl&=@EFMkBzxFW00|M_05we|!5^j~7PQ=}O z<3e_gNzjJ!U8x@52w2FC0K$T_fI3jYpUwSQk1obEcv4uwl8PXuCa%unFe$i#>y1YG z(0?t_yl&UoE+nG#m}Ar${1AyLAFuH}W>ExG4!Net28*gL-I_WOAH{4F9q7^Gbz42z z5@34)UOpT2rU#X#4(X5n|7%`_n>jydCctk9W!rMhDwZ;EtRHO<{>?=%^ik(i z&{?{viQcOjDEelMk>2-%Au1SjT#l{mK|ILZP@OmeI6c$SRLj@dXF#&V zP{~d9TH73su5d^dt)6gZ)XY&@#5owmg)W;-uc~clY!aewy+qMd<0#3HRB0@0`(Thw zeS|P@Ioo`kGGzY`$6Opcu3E;LS}%D301>2Pcy67$V@P#t)S9o(w*Ny%x>U|}Lu+%JRm95jia*M6r1$eH?N ztntWC0}~-IolZcYHrm>8gkoR`;>tnd`h7KG`>BEWla|4xwhzIpcjTL)GtB3rVgDTPBc*&)0wfV(}z>MM4d~dFA-ft~7&v;$w zGJxAJDbk-BQUIy-K_Q~-L(NEOq`!f^?-Qg2qkkXHN{Nx>n>rg+eGe|a@LA)w~uwEuqb ze}o95pbpyUK)n8pM{LJ6ty$kwD=xTbx;22u#`~Aj492gpIXx42ukK&6=*3ab9`@h2 zWzT60Wm`NtzPyP-`L&oLy>VU9kIh0`qhIvcp}V%jFGns9tJ_S9 zE=xZ9oBs`^#2%cm*&3pAG&8r7J_p@{!tPNj^(~D7(yV3mY;X@+`^c{HpnI#U$`y4bYjqlnkTW_zVM3OYEJTlYoDiCRBOO@|oqO~q!f zo3bDRU95P>Pg@VSGqM8KaA0k!WoMJCPm{S;jusDpkR z^8A;yYz{Xe9c0nE*4n;mBt7C;RgF#Ok0$Noe|zm=u>wJ zJ=U@&6=|z55$xpq<7^?+hgnBcO~7Py3bz`H$8TC2E|^ZtHWp+&0f2&s7C;c9kI!ZH zKgL0|EAWU0fZbf1dXNP@ctaG>8!{stl`WFDD{*I(ym)istM3jC=FZb_0+Hf~S=?K? zcogcCS>i2G(@M-trn@glnKN{TO&A%tM00m2$$|*bmvOmo`O=0xy)9qbUm}!Be}0eP6@>bdJM-F1yNj zS__zKT1+*Xv|$`T%te|ij#Zhkx;qR9C$Y{>PrC+uPKCv^*8f-fsKUtD$`8Y3!s zLio0wD=kdo-=qE?$I4z6xes5wQwbYca#l9&KgJsCkR=?c%AU<+dYMTIJ%~na;CZ(dGSJoC-zhSjq z8A~MS4AJ-m$=@jLK%|00s@u-#9}zjIojJ%Mr6GC5f8s8s|8)R_4Dn298i8(g?Z)00 zU5REqYdeyV!j~~333HG8Aoqp24$|;D&eP?N%j<&y7)TTW-avXGrC)4c?<=$~{alVe z+cr@g&i|DR3{c`dCb@lSlLQ;RY#&djP0x8-eG2bzA>XlK2`<@D8sCGFTncd)40x!> z(f`$-ZZ655Q#W?F+jLZ^TlS(gfTtT>z)PDtT24+A{0mwh)~Qy%00AeL6w5*X5hJxT zkH?JMmoyjv!n@;I$S0=^>elWC(x3(WA0(DQrs)X^H=1V!OR*9@_4rv({1;$6Ij(%c z2pzQynwVM{lwPcyHlhjrG^ecK)M^04(S*g zX+Izx0wPFAcMKujBHbV*-97U@`u)9Y!N0TabMLw5?z_*~`*MfwzZ*s;mWydQvl$J>SxPzmbG1v(DtuI`8aKAY{ zOd8~eYCFD~{>BVa*q~B8*zS9*Fq>r`HDgKHvlI zsh|EFklszmocL>6Fm<|o$8|fdzalan)U!8^E4+;#zLx?rLuEF?iRUFjG-dr-WPW=U zMrRy!CTp=exTwlL#;ptDWHpisA-fPJJeEyIWSU*1=-$6|p9ge3=%H!*irpDFVn((uy zo5x!Ef2n29;XA_MVt3pzA9G}CT&R3HThBgdFHjizSQ_G5`CA{*$~vXLJDq`oVoE>2 zmzZ6>_E(=hQ=ztYiJS=E`%y=ZJ<%jiFi(~7lLwxeI52n-XnCEVmm-o*zKAHB(*O3A z>rXo^OSni%LwS?g(;4FS_F5!(g8E-xqYK5K1Uaoz)@`Imgb7HW%6TW$k{|X|(P3O+I{fxqQ7eCc{HCS@u--9xI(ILbF?0`3e`Zc`&`QbV97PQ8~K} zJprHk9`bEHQS@E1y*hKRHg(h+5&n9I^gKfUYIe?}ry-%PWU=4h@^ul{-q~T1Hnbvu z_BE4_Jd^=kDU_I$#7~!?H9iARcg^@zlsD61G9qtnnR^*MLa^kIhi&s3V~|f#Vp$-& z3bA+DUlQ!?aYP)_;tmfvTS7#`(gV=|RJX~2=Z*Zd3Nw=9@5D34Sspidhhkf5C=r6k zJS6cN$-$BiMp(O{cd`0s99-64=w>C{y|r+GSZ<}?Bf%Twh(l6}j;2UZ(zrW``1vc` zLsuS!@yF=CG*2eTOyc9hS_%fi`{+HF)my|uz%+x}_k&l5FK>3v&yT8Rzne|sLP6l< zLS@3jW)7CAdif7U&UQ2fZpw|l|FuWZ;vqal4>2A}Xf5xeY4b8OGG_D~zI+@oYN;Y3 zB~fulZMI~N=>T6E>AH@Lw$UJ(Gw%e}%KvnPnjJ@Tc)hK~|GEMS{(W{jcx;nj#Hgl=fMKh?(hqVo@y4CnSp^al*m z1dTDY&YqY7AL>0p-t8sR#aNuEy-6Pls;fUiV)`pO|l6`zvWkLSzFaNV=PPt~KvS z|4I29SZT=3y&gvQ=x*%&&&;a*@*C{X#TCRn$8}CX&HV`p>2GMugvAQOe>m&G=A4&D z>fm%f%dHit2oGlp>ttKc(XVf8F2w4G9n&WCGfasYzQ~g(IIv=dF_Z?f#$f1CJ5pU~FEu*99LAZA>)=9RN6@dC zAjy;#yy0s*wSYgawGY@$2UzN~6Z|PCE__Ou9^wr^Ql7M8mg>4lj}`MN5~}}+p|Ji2 z@$$v(Tm%j`g*De4Y~KdS+N%MNmps{V#tZN)=yr&ix))b;q>kD0)v^V38$}6yWkOnL zPulkivkAcmGV$9iS?>4YA=r8p?pHu5EHr*;?_@c8{Qm5zdIh04c`Sv;zzyrQ{~tu5 z*SW{|bx2RNwUTojT~Swf<7z4DIfm=A9m+=kn^6<%o@Dh(RtCfHg-^j$PTwI_R|KHA zCuDfk9L*(r%v5D8(r)$p_-VopfSwN-6D52K7xcOw1NEvb?t?vF)t}-0tkAo z@jIW{oU#xeJ?Q?Exi<_v2)+h`dMcY*aSjes3dY#wQjUBobPIp7>=qB%v_{E+%vM+5 z8RGHwx7iRdC4=iz6^dqMNlGP0}+08AM|82Re=@!K39h1&Eea@^T ze#+F~%b)ms-tOW5yS4F>@rK4?P^NI{$WoN50O&VXS!Cek4#F;DUGG^q){5_)7WC=x#5LR|*qEv2u}Y zF8>b~iVb&}+bE~P*Lm+JFY<=i-8uWiS&4scOLZP_htB?f>t}h++)EIut#LH=EE}IM zerA#@dRQCY)a7+^{m+r#imi> z9!~^`^8JA_nsHUlRQ^ZEgP@lEi|6zk$p7xCb4@D(>~r>@!6aD>A z94OwL^Y5H5XQ=y}c%j$TDHUA2;roL!f2rjNZwtf=vDRcWfe=8qFjTo6-sIGI)E#%t z%9gi#JYX|#RfLYi@MIfjuKErzsq+N2OIBDXKysU#w7Fx{dIH@*HdTzt;KC!T0c?Oa#@I z8J*{tiaPGjDfa1%j%^zA{27ltLsj5aq-Dg_mAaFYGqXzrRapoige&`)oMn2eQ)NYp zBx^}w^#YJn2;{u<=}czsOx2L7A$xoGK5n*G8f3|u9MqkXu1#GU00pE zy>CBy$X~hHckByxX-U!MtKhKrtze;azIO$(-jnfH>K>@_&PAlMOtFBr zm2+n@hX)fxs>nGjgjJSmS*e&DJ0>?CGq^w1v-!AnF$JqYR~{TuXb28pn!I$#M6&8;bbG|;rroMAC)l1uZE5>$4);R7>!ty+`xP$*Al)(ln zs>1L^CeQYH!1QcKU!m40*~@1&4z%8Y^%nM3A_1@<~YE#>zR7odld$w;jfMt0maU4BvE@_Z4`Lud-&)W zy_2NhBe2gMNUJhigNa7S$jF>1^{6Q-2`IxkTta&$38%Qs&D*^R&;VO%ij$_;HNrTx z*I%A}g8S7ZE>PJKrCy2<+{}K#DRQmM^?k^%ls+BD4vzHwYat<5AV78>$?I)6melq8 zrf1Likhb;W7ejpPjubo57f;M0U0rz9aQW|X@7({gQQzVUP|2bb$jYeh_v2tAF?@OU z0!3bkOYa)#_Bm7}9OairItK8jfg>s-tC5N}s=riMtIx(Bx-#d~u@_*RHg$9}rD{YV zi_}*G>H##7nv6`v9$~ZN+8GO71_$z`)uj|#6op$fYh=dWL zFS#&~l9P2a7aURBYiz1I`!OOIaFXkn42;c{+NRy#fJZ0&@G5h4s`y(cBi+7AT5x>{ zo~v%G`S4|$GZkJ&fsF|*_F^s(h{!-+Um%GhrRe*CX}l-lDgGEHqb;M{V!??@Ng$8$ zMV6l2mBJZc9WpMShXlOCO zMWi~Jg?Pg-Ol>TJNQ`*MXVX)WMZJ?AvJ`eBZe`jq6iBku@6&CA&jjDISibwl z+URO<#}NO>oeKU%C$X|33OBHSlApP}X*7pdJ@%abwVYV|D0QXz*WLlMcRD7TAcv^!~}AKFDW26RYUx#SS* zPuTI`7&~|etqWifGyowb8aIC+D1F_W_fo?pkQbnE#d9O&Wh!N5)VRN_T2CyeVmzP2 z;~}daX+?!Z5n0mo)IP_Fa&k@}h@S-tS!kiuG6v0PI#ouQ22)l#)prk&X!i3+kly?s zjU~qcGv%mkUKi=hH}*ckhvA&3#=%S%a^Uw+)^6WSP{<;S$!QNS_fp;a? z&oKsT#rQ&YbK2=*9HN3+Qzl&8?8CO(Dvk>YShH(yigKV_2QZIM zhlSdpC>$pFqk_c6@RFC*1MtSC_kV`Q`-dn&?YCEe)x+`KkoomNcQ4>7bD(rx1X~>p zxzwjGoLy(GsI#s%GKro#$);na56?&;(X}lZj1WwT@0=I*X_OZP>svX;w9#3W#*QqWey#oic%7e! zy05xm65O4)Rj-zPXDmvu4PJwfS+Hb;tPJ9#XHUeQ3!ST!C;q2Kie`xvqzv1 z%(2PD;vu&4f|=`Ai>cI}lGT4og5nhV{W_K=XIpWFo%kScJO*2AZ3#J;;b665Cb%PV{qX*96OGY;(w$-LF z*g^t+g{oe$5g_u|>225eSt@|5H_U3Oic#2y84_~f6FrkJPguPUQdU&D_E%avf0`O< zlFWbYod7fZ>pZ#4$5Im9*h1dFBD8l%3!<~j?XO;>EL;Z1`iy|HtF-%*+#bduz9wAo z3sx6jMphdhC->l!(dy*nBguo?bDbg2`PymB7rzXk*KM z^r}cT5^`gS-L|=4^Jf`LtCi}zwn`K#rflk^7lltgZsBkI|GH=Cv%Kz2nHF6?p0aEF zi@&Y2%_i6oK759EC&k=%1#1(mgJlrMbooG`?>vX?ZWy9<%z{=u*n?>@gxMD3Xc-@> zN7X;y@1w%W9Vr9p(Zv}=^|OBQ^N!|XpTubUcuxjx#6O&41nl&}raYm0+}JC?>vE>K z?Yj8Q3B18IW>)s-E#BZMaCR1)6k_ZRqBS91<*KJq@=_FSwaHneycm5?z&+H*Ko zuFiS_K&u~1XMTU6#a!3kncI681v5Z`x)me%a2ShGK9TXqra5|@JX}S-r@2z_(O=r%^CO3a@~M| z?e=qYu7;ob!hHBiFT*CleGF!dVYprUaEGFTS{$ws%lL~#H>}X_2huTC5$@R9XfwPq zsL1_H*%LKBXH0;6yrKb>zjS)Pu;G^xiWYWm0x=M2<3r6)=tq?)^d~MWp5tq@fANEh z=s9^O7AQuea}M>&OB(6JP0t_vFg~}1km=c9y<2$dPg5p)k!UcXYE_8sH|~wP%K56~ zP;9tm1Oo;{R%dOZpJ~Da=>B@eX8q-&MG&)05hWX@&Vn++kk{aUu)E}_1Y_C!ZS6Qf zf9L-|?ia%wL4JlV^G1imG*HZbHE2hQvg3c?I>3l36t0cf!ZDuLl zzu%&L0xSp$M3>PqL-`U3nut-f>Z@;ffG$s=4E2GG_#6J$!08VjHE5jTI)sYFynI9H zlPBPJf5LL=XLHe+5vSoXBm9Ju{B^ghz$; z5sG82ing?y(tVwxjb`rGMz;0^|A~FZbz$Eg1_p`xBZ}a{{|l&s2}xGYKkdi@_l8XF z_{OyE{4OTn3T9rplZ~mtHM_Pniohs#*hQpXX^d*SlYVCLk1wg$`C=zk7ZW@Y!|YuG z^~3zx8a4Va5R{cGt~fkqkqcoM7=>ho*D_bauOA(bekFylJH3BB{SdP9#_8Y6(%lwU zg=sWnbsf&QGX8NobZbO_s2j|@WktLhceqY3oE_1BZ@BxyCjjGoIdXfrUm9e$#c`Dx zOo@h3MBxt9hpfpT+-)_mbC73Kgw;vcx^!cGn6pca5gfU+aX%crSHq#R4CCtm(l1HBu z3j!r5es!`LL_-E@`RU-Z#{MnYz9T;9r0@@f;6Ir#hO|KpbCsDeca{Ho6qG;rKbV(x z)1nwZP-f58i)6*G;D*9JK89y7`Ev)>h+2}{z1qw_pPV}0Bd5wL!>$`7=9-&Dy5XVl z$zcR&L|Pn-sF;;0|GyWY7=$u&@y$rtB&IC2Kaz<-9$ZvV2Lrdzdm*?b@qRgX}vH36KHDlc{*u$XB#Cgl=#|I5$#5C!d8vZ{L$CdkWYIkMl{aSQ#`_Y{5{YuDOXjWB z?EMWIjDtt1J!_du_v^dK{`JjclB$l*xF;2?Vq8Fu^+kq(HNuTm-{;mS%hIHTA^{yE$O=s&`GNXLU*KU#jD<;+Tt!H^~B z(PT>Agc2ym8G#6qI4512CN(xmUHy`J1rcc9cn}akOu~$f`?;__X=aO~Fj6d8Pt2~4 zb3FfLblmQQb-PH^ux`WmmAID&7L@%l;&NWXiDPI3tM@tr^Y>PWTe&q8>9fhq^A>ve zV935}+P_uC_gaf4BZaa{b6t3aV6#aNu{p}w3n|rS%OjOizUWqXzOz`(pKU7|;vRNE z&X!@1rQJMcNV2dvY-e`z{kbo=W~sfUR3Od% zJt-O$p-Ugv*Wu}?41!$v7Psb@5TnWZnXJD+`kKP;&1=@xB3Na8v-!NDW^9+^ga#9U zg-2J`hOQ64tc~$Ici%h}x!xT#3kFhx@(}mzAW%v0IWs)!Z!$ka<4JCAdaHXqopz^h zh+-V?w_)QQjw#bIZfc`zqHw^3nF16=7_g8dw2>I9Z7B@+O2v_@XpBbex@)~$cv8nL z>zz}5(m8vvz@No;4M{fi&#yLHV!mwJ8l5zD!{^6)FsZ*pL z-4#4c8URSYXWqOfGFh!I2*rjd%2_HhOL=zznSiIg)xEr?km9!Y=Wi_AZa$Ps`cADe z{*C0-c3gp_go0_}FBiEY&qJJqD?TLzvtVee{&4>-uqPw#IOFfOqaZCr$hhzI5im69 zCHN8lOs(LP5sJT2@bQ*N%<6)lQa` zm07oBL7VNTA9wq%IESf@8-^YEk(3DnuMku%dmgrBAFs(fz^H}|v$EWg5);YeyTHk7 z?dnS2M*iGNobn|Sd*9|LO!OpOE3W5Dv`w?5V-@Yw&iaquke0jk07Ti!ku);MKBtn` zS@M?7yz}fYxsaf6{ijJ_5Hbw1Nac=PjaN^{UyC9y)=f|CbC2-`EWvFuQRa}u^&whV zp$%i|hdHg}E8SE*d{M)1N*48OoUJ%dJ(&;Kp_M8AV*)_^?O7uJZ|>~^S;~1HW19;A z-}B&D=^0)5euKADh>g?&#>B|8vthm}&nOgM9vD_O1O)pef*~?h*%pr0P?r+ajbZxn z%8%`af@yywzx!{w6q+X9n!KEL4J<0MmTm6DsTAtp3OYPFvU0s)vP;1P5UmlXGUq!b zS%aKACCp7Y+JD_UW+A()ybVfA^bK537v4>X`)$z=(f{@(=^eIt5ckA5BB$lFsL4Y( zc>1QGmMFPimgNYefCP6dH7b;AuC=U}qA0=na!$BwR%esP05j6dVrMtP{9N@Qh&84v zb#~;{3eyCMP9FKn#L&`%_(WEmtomZY1 zu0JC|B)Bz~?EXpkKfS%)CsM-D+W&fH&a{rczHlKIw5SZ7#Zu+`-I8)edvat6xZ(-t zGEE2=6vcJ%maUtAS0U`jFs7aUQ`wP*Bisr6BLclP3t%h$r^vdeAsgBG|98B&YIaFK z(Sl^WE@_Xk#Y1@IPt6$|BnS8DRIUHZZbY_j=#F)`;{Z9Z5{e&DKp0HWCADf?J)Khk zmui0cB>qlc!X*Op(6_+ZySGVSY5Va7V<$EJa&IOjKk99Z9{IDevj~4ZQV-FOL)@&+gDjj8OH=} z?~dRC6VjA%`DTEvn75>pkxYi(o9NICvk5xlV01Z!(ARZHR!T)3OOsaMp zOw%wsV>}w5h1yKYPX1xocMXKim?N?oTo8t?2e(bfewT9t!2|y`sJh&zJLKxop+|0n z4JbF#Nw94E$66|iGk=c~O8q9r5zBWfu%~lZ8hHnjjL?l#{_F1D@%3fLDvPHE>pY0f!K`bm4vx65nbwd;nmWs?&SGqmt&NL>l{dV?gGPMU2LM;y# zRVk|bWgHWp@!msXJMpxLW>sgBsAo_?1hW5OFr!q`6@Ix11o#CL;4oqDy*^dDk+?tb z1oh? z4XT5WRsAC2FP8TgGYu_|wb^+fib}WR%eP)_((-L9kfyb0MaC54BB85IUsmkVXxLic z7P--*PDrhXc9na*W`NTfvZ=OSC|MP2+*X{IaG&oD6YxDcd5p>6m>ee4pRxQ@-jhf5 zIR=xqA&D=gFGWgt$!1c|<&49f@?^I)*4EyI-vuVVm?*%?HX9gNjwx>FV{vkay|t~) z=SCWwg~kvxD8;;F{_&J>qOdYWZbN}sf{%Z5%?rGo`^C7;_zxMmNd0`W2ZJi%Pax3K z;?_Zq`YKrXCQNL@s`P1AE5#nApv!Ht-(s@gDUG>lu>lsIwe7+P8kvW=(b<~?6B-XB z7KqS5P%NjmVO~<^sgz}cPyT;K)gCS_ha`~u-%%8;7??ek7R;jVNo=p!Ko+x9Lc~!8<+#)AZMX=+~cq7A*^>H3cXkX=ATL? z%NRf=Y4@M9t3gyfuZw4dOnWY3D^ZEDlwc^8-Lv(Q66a38XoK2?=I@>C58x+%Hzf%u z@M-OF0tLEuhmSyqQU1dSJF@1tmh_6b&@5+=un|S7+0d+iiij^-g)NGmZLR9Cu{d0DlI8gU! zDPD}D?b`YDGxd+-_L?nWLSF|d9Q(jw-0Tx31I=1!X}PI{Z>B*i+mz=o!pBK#y%LAu z&%dSw*<4V{6LcX(9=b+o!4j^t=#RV)6Gfde6U_48^d2cAy^G9HV7BvnrBlZH4*5;a zMjKj}|5a3uG>WHOwaT-%NPM(8?ZhZOdi0fW#l@E%Ya)v5Yu5`58t})}o5#f)-m^5x ze<8i4X$J=-3$X9Bg|tTow&JGTXr`2Q*b4i4Yd7@X1NJ|JXzJCXP`#jbgdgb-o2IGZvsqzx=DA;!aEc z?=$yFzCxp_k3wmIg2fK%M%mRyn~)`sPMT~n<$OkKt@hDFRx~EcJA(SMtT%+XB?6QU z_c3kEEJeAV05V*2z-^LRU66oB<~9YdM{oNCT_hpMk-ARc35Y3I*?;%87QRuQBHlv- z!W;|Xm(^bn1Z?7rRWuSa8Bq#W5~ogD+?&pP@vNVjnA2M-jOSDMrEQFk@LD2zP=6fZ zh4G}zSu927{HFNo_odW3Uo~|S3ezG@nx`BW!G zCY)^4Bk3wsG0XLSsaOheDC9cm(@?# zC*WTv9{=%FLJYsm$fs8xoag<2pq+(ROf!0!UZzp8)w|Mx4ZQ9L+@Twv!-rS~;BcHQ z^{KkoYIsm2C(5_tN_qClTR{XL7R6x1$NaO;!w}lS8hzFyPPrHyY*-`Q%@||~i!HxK z&3BpO257w$^j}1k7N99d;)k(>#UlYnkCO||%e}S%7@)ZKdqUCY`=t&d|T+eei`@g16HN{x`AC}#1FUE8J%^h(~{e($O!oE zF>cJ)h8(6@xOQUk;?>mAG2?`gm&?1{rSfc1`ctojQ3TPJ>2p!fqh!jsi}juK>?+Us zQ~5KKD^9RWyS|yh-3NdJrZ94bY2Ij0B6?hK z_e4{a!-I3hB;Mv+8?}VR!ih&&H6oX;(3{@f$C<9N)g0ewMLNNsIzMF16>(C9+*RWf z$cGNy?R>PCy7-64$d;Jqj$P3jo16OC0#M!s(qs~S|5!n%b4&IO)$Nm^8t2qeJs*u1 zAC*_~IO+9FyC&N2EO*{YjK&vZcUjKo=|}lRS*>(4yVu#79HyL@28=jDs9l?NwIR6+ zOau|Zu*j>1=ZZzbt-148FZgtoh>BY#%M7P-JFDM*!v}r366{ssOhwkc z;^bkh|8J?yxS6Vp<{LTe9<)21U-RyfU_#~xnfy64{twyE@%&pvjAr3FojgV_7AGMfHZd@j8-8D`OCM3DkGShXG~eZ&zl z$6xA&IpQWstkTi?eQkPw;v;e#vg`Zy#wT~Wk;8A{IQi4U)9&ft;9mU zjQ&bt>6ulepMD#)68DC}I@JHVi6s7x=;aI3F-P-WnCXQex&0D^V`~4${34Bobw`pO z4J6rZ@ls@CWtq|m6t@Fgro1z#)Zfn1H=)1`!yrM$=#?te6FI6ll9;1&4yhAby!&;U zM;~xF*n1K}`k7yln6z{3P{R4$>k+0WZqkBHYfd8G5adBb(17snjuX{5;g z!OhD);@PyQj|)z*kU+6TCZqB*Z3x+nC_S`Cp=trWt1El)qU%!}SXUeg>M5BD_&6n2 z*UWa0j}uT+KlKn!-3ytpt8!>)#R_r9m|W*?YhL`qlxC}f=Vfmdo>)?o#+}-nhy7%8Wou_}H$b=D)RQ_&mxJ2Mj zoSN{07+=X&B#4W)3Wn97*^eKI_0r2$3g)DjQ3gVDPqOpdXX4=RJ%|(-8=KHXy`6BM zg3&E0#2X+xZDJF!utKGR%?w-3i+a%qa{(-_M05%i%Em6DZ>bv#)hjPkDx6`nhSJ`R zA(x@-Mo*Qmff4@x7H;P4eas4y3D}B&q5ms`u_-UUg4j^#h5s&;&_G$p}Mh;-XZF^mAoJkD88{ASg)&vC%Jr&am@u6H-E z_m=~~#|j|d4afW+LpP8A!RG%B>zbNP@IvZvYocezoGnF4b=KusW1m0vV&xQOSLs20 z+l?JqzJ?fWaq$pfEMn_+`!hxb_=qDM%Jye^k1B~C^-Do*H7_VtD=Oc+J?>#pA4qyF z@+%Wi%-X}bdAPR^@y8f;sww<>z5HQ2aw*~~ zMf=&4fC1m;i$TY2wbX7zwD}`5sLMF`TKe*T6l~L!l;6FvOPs_#F$2?=wTF=W@R^Vi z@uN_Ogmp!bi;$Y3l+A=aNE@R*g@F3NbkvU7!TKXNaS$?2dBf z8!96P3UHBrxgY_0$*k$SJ(i!pS~rL4}&~ zq5{Q4%6jYFQ?u39AeS2GOo8BnfjfX@TMK z;*InjHt${B-rphxEvEQAaRMyvTG2(V9?i;Ou^UYlwYROw~ezJ*~kF-?qod;c%Re2U2fZ-Ft_;j|Tz$Vy zRQ7R~@NRA%N6)8yXDM%V$|nbTA-TNT(2oc5eYS9Sn$K&m-E0D4wg!Z&Yo-)RXc?1r z=4aB_>Sa=-*2!Ktb+$kdpSvbZ%IgfU%U^h<$=L#?P3s;nlC>`MIgW(^=>X%_a~lDa zk84!_fIc<>-P4)fcX5gLX}k8F>TOjyvdaNb!5@>J%lEa=X!Ljnf;}&*AMQ}^mx>h1 z05@Yl@tQC5>!eXHm2m>`EbTUz{#`$Nv;1k5ZKe+P<12I>PJn|W8~`i!3MV>U)zsBR zF&oPPl)WsC_N{?`!_w)jiTA1l{?~fkZFC6$!;qSenSq*I zAsWfnD6^(WhGc>U=yogoPPhYIgzdB{^#()=8`iV((31Vk!RT0$Pt$1q0uc-m zmQa=te0!{aV{wph_fUkQo-m5YsimJW?I`w7)dG%#4^vEOR=j9D%Cqkcl{@f@U%K9S zbE-y@AjbxNu7FRW?BUmY^}63q9{-`&V~9lH=N3`4fe?rOs$iWM3n<@$WiD86#f~vL z_D6Pl);)#$Ge8n86OwEZ)DewM&L$cV9A9UD15{+^g)oUWJlGFy6P%R0M0xg`V~>(Yn6q(%P--5v%ZTWp2F#sxAY zd}?7{A6hZNPL(xWOS&l+6Qy1^BQZVpr7IW5!T*lkj`fWpeP?k$oQ|XX;?gk*Ni~VJ zjxF87?P7^pUZ0TP<=2m*j${Eo|fZn%A&*>TXnZ{uBQ*eA@ zo)Fl37CrM-*sJE(T~oWbvaBLzL^4a`F?m<|BX8%wEWrNNJ|7+ACH%So7OW$9k@DwE zdQ3i}J@Nxx^+GWE;-DR#Y7>O!o%Vl<)U>>+Jz*e5OiK<3Ny~-(h~^^0giz{F&0!f% zWb5S`(A*Bn2MkLF0tN8q{HL2b~jf$Ta6ZXh8j97#Nso z{L{>LLV7wxF1}2vOa%W2y(a&R3HZP4c>s|Y4ml?-iQJQPtUL4MTR@{G4Hu%a6Spnw zYnuJiRcv&NcS+3kfcmQL!Hj@0QSOIUHC0U7lkJ}`;UGuK*>jDKcY-vL3W!Z$m9 zAFzP|MNvVWm{(qCC1*d|{?GKd7Ev|=0(WXA7v5)V|CqOC>Zt_@RM23^DW-fT?Mgv58t+OcJq)OU&?_}$-MUm&JO2K~TC2!vUm_y^yA zLLA^IbP!|s4Jt~a>}crPY%gg&yM0GIsi1ko=I%;^T8i~dhdJZ7BAYJ%MU_CweYBhx zH6ANiR>mJ_wj%EtG4(ScHG_BS@1$)?+s3Sw@QEo|QXksYx74=YJH561|GfZ}mtH~8 z%Vt6YQ9d1>uaSXOLTLNG@lN7kYR{R==`iqr1X|Qg$8!CWAuW=Ds9n89qmKu^2Pcn= z$RW3vw+D#V#=}7C=yZycXY*87>)Yo6}meg_uqjc$6ECtFLJS zieLr_?CY`g{6(qwNxftmB|cG5UUx1uJ^K?R=BW4AttP+TNur)MPq9GbIY2 z0@Spj_FPMUi?8J?5t0($pXr)&APkEEN_22X(7@TmlR=0+h;?FUlz~xU@=p}Ro@0Rn z=KD9)PD_h$9`|n$(jchB{WK+Xz|Q<(j-1e27#!Pme^l)-qo4=;-pk9mPgz|+&0<%q z_72A{tF^zQ91mWcDu_&=7okGe=s)~tS>(I#ve2nIlx{9^_S;QTQErnYB~Fqg?~-F$ z&kyHHCo*IH8DW#upG~DgSOS5)>!xVz{l1@@C(hA_e+Fya>l*?g<7tXe z;>SaaZ=aV&-py}`*WNp}9E?^gGQZv({3+(;em%j$McQ9(%P5beSWJEy_-IRbLuWrS z{J%CfiSIVptXX^))~`HNha6t9A#`hZdM7=rHMo}~%!ur4J-&R-nPW2tHMe2N)QRfs ztDOJ$oQeGB@b^+13y;XaJiX$Oo?3=0-AND=h=?t>Q!3~#fPN`(%~fLOB>r!vp-W%t z0z~&m1w{`s@{2q7lqSBiOI~x|%H|&l2`WPkLB-`kK>>ERcHM0mqI8=0qMC|jIUJVZ z66zn-*)~P;(AQ2}QwL~q*N}m(Lo#iWz4gH) z$s+apbiLt_VHPQL!UPqR8M>9hqu8kDA&((9L?A9TaB&Pvdb9$zuFCGRgJpxFc&?Q& zUGHzd4Uteb_w~&!EHwgoB&K*qW7?^gcUwbP%xC{M|H7wJK_KX3G>|goUGE4$pC85% zdwox?Pvs_Tj?4z0SPQGYIqGP-)w}}YR#GqMFPld)K)fT59=6*fqkO;$3(HC6L&|CR z$~`&)XGR8f0eUUmX8x8u|1JP%jM#8TRKrbhG-LX|?upYm^y=6@5NY4cjz^{sD%4}0sG(@Yt;;}*c&hjXi3_>N8pFh9p z`nUI}QTba>Q1kMhkm62R8WdgY0(iqq>|>=sTGUJD??QdJ{}cu5?1KB)^&CrIC`jcq zzTO?*L+2<%4pc7+i6iIb0zptH|9Uq3^;r=rde}P=hSPaBjvq85GFZzGGw1u3%qM+={&-jeJza zJ;TT?p@t7?xmXt!)Hccq&ugZ5O;WOXTi}M;Lpmc4laTzevSE@-Nh+H$`&Dv@r$J?| ziEGe%KAy#=Oe!S|{ZE6y7g!B4oQcf-o7)407ouR0@`QLUbap2wMlMU>LZ44ka#jFG z>`=b{kF&5JxfAQWwzaYaAw3==SgnwsTWd(4)Wz;pIzH8OM%VSr0okm9LveDkC-5|L z?GTAmecXe5d1!|d8&4q|!UDzfU5neFw!-0yVJvKfwfseLzRS-DrcfK}PE@QBKMXEN z*sAEGV^%fNqy{UPtG%fOKUjshmH`@^Z1QYr5LtY9jGQFP9a-$Slll#l@_iiZ`H0*mJ z`Ce0%2K{gzC=3eCgpr{PdHUa6ivRwt#9~9P2P0Nw9aR*yq-aaEru$B^Th<-;8Ntv^ zPrsazgZKj|{JcL<^JiqzeyqcB#Et){X=wD~p_q>=x}h=oL#KBxNB@Qr(#$(1FqU;p zbLCUVzl6WFV;y$A=#;a+dq1?U``^a%gG=(y(@6%oE=EGA@saD1#58mNkEXA1i}HKk z-XNBeZjdEJx;vK;P>>J_LAtxUmhO@+Nd@VUZjg{J>68ZPuI1g&_jkS5_51_RoO5RG zxo7T~S}%~sAE2={Uug4%p}sI7a!ZQ{6QDv<{hiP10gKk(xX~IhbN$3eGi~j{{BtUn z_lcuB6Q9t^_>o+Kl=erT5>3%29l9>`5j3{dL?3-xa0J&^z$pj({|v35`T+6tp7mQ@ zKtWEd{g%9>c)jg_sL#+FoHxaOcAkF{i4Bi< z2~)TP+olNi{J!Ox*CDeptPj(~L{cgjR4l{!7ua%#1+VhB5Wux%Q8^R{ux!eYU--su z;lLlGYh=wSzSTb9It|%%BZml%2LTG?{ZRpiJFU7C3qGUASh!iy<1695_x1w!IhyZZ zm1P`~aQ+LSC`&!;?$xFJbo{L7r{YZXPOl}2o zZ;v(2Ypu;dtpIGtTfHg@dA*c9FTgV-PX+9`v+;Z%&l524KNPTu6Jm)o2EG-|ip|Y? z(n5obYB80mv*cX_gQgx{w%ZaA^^K&ybtNUHFQ9>hm2ap_;2F$Qbfe_sU zM!@=*a)&SZV}^{9gMPcD-Law;63wUoB?~OGN0f4n3#BW49`>oiBvgE22*rZpr^o*O zR4m*^l<>dEAxLAkQ^nb#YHDH%Xw&zF8$7iZdrA1~$aJ8~wTJJL(ij?smq!}eK?>60 zcyItOd+r&>L$A;Bpa`svt}C@nIiNapawz8{ZTo2%)vu7hyuQRKEX&)2f-qmP?%PK^ z044(cJ%Fo-t?uRHN!**81cT9CI6n znayRt?Mpf%_>uSR0H-UZ-9h*~;2X0DKKo-R)(OT;DakJsTIpgC*{eMFwKdEC5(=@^ zi9SJ*h1*vD`^`SIuUv4eGbc1;f9RXXMcmw9m)pgj*Y_@5mrQJun2PfG#PeM9poUpv z1z9X7GK2N@FM|uVRffBR=r-LW_G06ibf-3Cc7`d=g}yA|V=iPS%h-~f8b+4vdn6xXfg4Vyr+L#s~l=t9T;`6`459`B?vCj^UjWlrRMAlPQF1{o{$)!Uu zO#hn+@gItCI5ze@J>h2sSv(VT?^J;Rvupbyp<5yjw4JBZM5B@le3CtM!3iX`KqCNS ztZiy<5t=E6v_b?|2$COomw?wNE0wWvPTO6L%{a<-o|oaOfA`ne_N>~eSw?+BCgr9{ z#JWmwRFC_8N$taqAR$=;-nLO8sni--+NjNqFeTW8Pw9;8#skZ|m0?5{ogfB`CdfhQ zTqVxH$^S$gu-y{!69SXM#&^eOI7Fn4|1Uu30<-;>P$OqsNeq^rJ+V`7Ox(|&Y+rX` zhz4+SIV6rqO}XU-0~0B=6Or2I^o+m+V31}E7bZSRKA}Ip@nobX+UJ6A+>IKOK%M*L zmjufs!Zz(mPXuU9X|I57(p;Gz3+_a&qGa2bIw*=w+_%s_#~%FQ^_lxwY#8gd&{=P2 zV)z_zy~?Lh$tH^XFxGz3K9Z(*#PknV2)yiYn67iZ!L~3NcMRJ~nDJy9qWG}ksH$(O zh#6+7m~6_;fci$afr!p5i^Ohj_d8L7-6Va%Fy_2(U-y&FC*rZ3UX($605jAU ztM+eKspfoTN*!QG0|a+Yv9SPw>_w4sPAbG-6{#Bu#Zou2Dfv1!q9T#mciRs=E6O?g#=Wp90 ztt{+Z1UnTs29;ymP?j+={?RG?ffi5bp2?|1kj0Pi*L%6~cT%C4EF-2DPI?A5j>&&y zT#T27KYeGov)RRDjWofxEV@-0LBp;lXn;>AJ7)fJAH^y=erJXHxsm{oT=rRzf1)v$ zX5_J#_x-W&g+8XPij+rg*zpSZ7`%uMohnaN0uYgO|Ippjf@Jc4#0!#EmKz#^pro%m z^w!!zhJzBO9`SpUF%)14T;1BL_H@+52TR{^JMezgeR_mbzTddRmmU1HM?n6~ujgU+ znK~o0)v7PL&c%-CCJ}5v@FEv)SwuP3ZA)ko6l; zm&|@nNG?GZJ_ehCil!iZj3NQ0IL$?1#lJ-eLGXVPj8e68>@?#sq(|zxR=ityq3NRENPUK@1;-5 zcn4b~DKhD;oQE6Jo0`Bsz2dhe{5Wa?4I%-tK%mC?c|>wBCX^5&FLM}~IXn_RG9JNn z8}e$9>7X$te=^`hra1k(n3)@97?+eJ5KH93Wy7Ij^uh#?h;_cuX&F(Sh&Hv){wpRj z`~1uFjr~hFU#>hPXvI~P+Cwd?vQR`nv>aqwh|zJPD7N}f*rRnCbA?!^};yyBz>b zs73Tf8ti*eg032)^H^>ig2Tn1KSvpDW=mB)1jz%aw%)M$(&P8RdO=(7of92`H!HF) zYoSnrYqPIq3#QjjeA5^jxghw49yrt)6=!UQa_|hrEfzzf-q4-es;0t;Zsaf?(&`q`R+a} zFqc-ycL6^=KiqN-HRJ9O?hV(pF5IJ!Jr{Q|6xo7ZYxu^&f2AoPm%47RRE=}TKP9~B z8v|iuTP)%)1Yj3M7-}$-CUGqPpeWtsV2Q;nqQ`3m6R^S@;3evTjsw8#jggUH=am00 zjki|iiC?eMvd?8CvRGsDcO*9hL%S+(tzN`0b*f=wo}$M9^^xE57$g#6h=0`+y6#9w z#Q`)TmHxp*UGxujNBhb2GQBZ>LU*R=O8%zcrYCs)iYCZD0L@!QxBRDqkafO}$j_dD zQw;X%#;{l8)>wv2Du>MT49QaU_998y~F5>ajqL>S)1fe;hL(71l z{6Yi+90+Zm@38@9=_Mz@%ko2`MW0^*nC2d=o7ZCt^GT1*-_R$#gVlvy$J~Df!@>d4 zOb;(cX(|Aka$WtC%8$zU!5G!0wAMaEkIfGY6wg$b{YQ{AhML^AaEu7G4-ym7@?ZyM zTZ~UN&!Lpyw_T@Moh?C(vz8(AjT?Wem)2jw+4A8PV3(xBc8T(KsSfUP2P9l z{U#b1ep_b$qbR}@uZN)8Mu5eHxbQmH9;O*`hM+OjpX$Mp4W*^Di #cfL$h^Eq%k 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ΝWQSb8Sq^D}>j8l8 zT$t+>Tn z)MhPe6S}goaq{wH)Xs`l+U1>9Iy1t>(*e&u)#CE`7K=*;67trQ_o$tchxpmNER{l< zATDmJ4PQX`R-!u{*8ZvJTR-$m^4y<~<*x&O{T~qii(h(wBLL0L?Bv!zeqO`1-;L;J zG}-=2wc%`9~R+H1%nY3fz|hPzqjZyO(P(pj%G0x78kl+N1- zF(jd~abv*djXrTd^)XmbmB-mfdmt~psO*av859**ivwQoceoX`7`UQsvsgNDQZVsN zW-YR$OR@koLrXjM3`a`p-_Op3&g$UZOLpTIzgfez?&#>w9o_Cu{yoyy(v3vYB}t6a+9o3uji~(rMrQUq?Qt-T#O3LN!&uqr@Yq&JFvK zK-bGCdFLLm8PIxlGP?=!TfW;rz3nN6#ZcF(JWl&U?~96z$Q7qCgjwbv4}I`fBl*!I zd2-EOqC)3BW4_Tu8Fo{)Z-?xzN7%T6nwu%39C-|aNOs=o_4+KJ>^ zAA`#lVYVZjT?zQ`vmIV~y2VVVqVzYlIpywgDd?burkBBe8nq`h*_HI|ANd74f92ZD z`QMXzfB2vI-+cLp-)9IwzTx-2aFw+GJ!-?x=-}o@HQJiPMj3Q#vQd%)=D%6Ju^fBS zq9=72!|ffowXKBlE1zzH)QSUn+}UKUnQ2z2l>K(*VOq5%7!xiPJun(qZqRzBG_ z%DUWZHH-Imlp?MNwu`npKrqrIRqqGLI{lMd5WkB=Z$Y{aV!^cQtYuVsV&u5+`d9e- zsmiYAZZ5~u9byxR`siRM>-Kkb805(5ud6IL*+XrG5_EN(}(q<#^(7&lP)TzpY1sjhD5GimCCv;lE$_c~Ob1j}eYdEoN zJnZkLW@ywXi6`Q6A(Et*u)`%<%clgn6!hQt+rZ8D6#}5Qea%BIe04nl*!kQFcJMnH z-24R5dRx*Sn%8X~!Vr%mauAd#5w5SpjSV4Ad92WXZUY>!Fef~H+HiWw_=cL1tVyOM zzz_25HFG#+*OG4pbPgn#;M&PO997OG-UHV~bY&k^;If9ScGENZMb|yakKOhog+M7T zEBiRFV>4-8Lg?bB59(~44r~Nr9}RnG8Y0eC0y?R*3h0sC?+AKRX%4d1OZ|2?BL=%k z#`sr$QfYj0sRb}OH$fSwiZnP)Md>C1H=+(3af{G-*u5pn_(>$4@Uts%ki;qrv%-s) zTfFjYo7EG6H=gz=?&ZB`iL}rWP))(HgzbG?G)3s#(bs<9pUX?%|D`r;Qh?}v@jEdkgwZF|Y z;GJQI?POHwrzQ+FVVk9uQBJRGpe!v2uRPP@m1kPa&*sj5TAJJ!m8@DLO~DQpaLLCc zSm@%+oWAs~M4crTj0x{-3|t;)P3XE}m)8={&CT%zG^fsRnE*po=*99Z0%DFuS5}{Dq&B7k==U_sw74 z>j6NX`FrIqL5G`*3~yetTmR-W8gKnRO}ih~ct>!NuWdFmw8OG!=RN@-5D1-@{k^s! z><%EYo@4)Ul>qWYEilk-!O2D8!YV8;P$3eTX8Q*}but!Wk)vz`lyQ)N^>f0m=l=}K z@jYMfkHGDr@s}Wa9Z5I+&xwwiQI&gVUXPvFPaC&Wt*JXd=O6cd$A3J`8m5}4g+2yb z*c_#lEP+>sy_CJp$ix48NyZ$uPpz@Y`eaNrFiH?4DumXsp0rp`TI{3&k&{~Q(&NGa zRiPaSPn`;Q{$h(uX98AF1O%;V(tl6mF5`BuQ&a_I8F8NknMK{4zWUidt%cX*`5%bo z<-g`#!yA7ddEtkiCV*G>f%lD#4SNN>Z5XKwX$k{kZ1QM zP*MliHVkjy6^4lt1k{_TCjx;|gYmR!BNaL=IJpETmp$=M%b0q9kI>Knn8I!Miu$y{ z6<p?-@>?$vx_X8bfJ(}ccFA&TvW5a&}{LFjh6W%uf zNY;K9fcDvM${{7SkysMY{ivqh1#Fo8hPm&CJ!l+XHiP-`zq{LTYZKz6 zQfZiA2v~?pHXM4crR^Ndw&CoupZlNdh%Zy?xL4FtiE4W7gZYdzM`ZwjDiab<_s?6j zk@-~`1I3{(L|i{Z0%DLEUlxe3L2}JIf4k#L1gaI)OW9wV?MPF9-ACfJvZ=>qx}h?Yz8ChfDFq3)jNjXG)LHL1GDquc|J!#%qO9@n-uxuZazL5O*YO4aU5n7OhP!c# zoz!5pUi5h6sp)vUqV`qJClLg&I4@i|7x3IuElwPmW|D;k^j~jQg@kaV|+T>t`}<#SRxzW!PNTQB~VWAo`AxRu(fimJBF!s+KFUHcwv_dnEV>xv7vz$RYgrs!$X zTC30qpZEcmU9*f#)v-5FHg;jVFZ5%KpvQbw?dh?5R#oSA#k%#?q(^9XeVEOX}*xKxcisGDNP|lwe zUcA)e>9eh@_HVL19w2Hm`zBMH^iYN)0T*7;Wczn%y!kySzgOS*uP+|wyS^s^5dYQ> z(E0H9`;Qmj6g#~8oQ7Lpr^(*8Yuf#grhP~UU|Ni0?JF^W!@+(Ct9wtq$ROFhFSqus8>&mtkeWPbG|((2}Y1KMVwyEA*)z0UMz1i&)jnxr;ra zkKKWE6KoeWQ@bOmZK31U2Ac2T3Q3PhL-mf)Y}ls0%RlUS=s%4S+%b_MzW2vC z-OYqJa+TCs5560!ERltDw9vPP+fkeC#7iBUiEm07ZRjj03XNC$!ygwDc-Fr!d#26G zqWE?Vmz6P0Ry0tpB|X#YkoHRfQgECH{WQVMEINNi-~RLempuDJZ#Uwz;~4aV62cKX8luF#KsRPV_xCj3nI zOWed~q=A8jSz&oWSX~h2JDG7k3el7I%b}9Wtg>?xo>?9vfm`heSLO|OHvBrV1OCp|(&WeiU0>~%JQ;T}X??+nGD=_1T7g0%31)fCxu zIl&^wXhp|;SsivQoxh}!Z}=PTW!?T`p3f2T+CSJw%vi$|;-RItoiOMo#G%Dmt+Zt5 zfim7=Cur6SVwBhk>q)>y++t`2t3qpn&c7N26zg447R>+EI>oeRi;{{kLGAeR`$zSL#l8EHYSR{V)C$@b+=Jt0u?TvGJN=Ii*qM~j6yrRe=TLMO9n)pA^{KUFRJmgcM?D{ ziyM4`)Xj?N%;@X?`9G&~^4I0+XJY{S#m@?$!OKTK9Z(AZuK(rls(+mA-+oEso$tU6 zzFpJaE2@dl!l=;vHi*BfW5kcjD8wmrL(lxz4L$o`6aJs*zRjSu+6=IQs-Ydg%7U4A-Q^d&fhC@DmGFE zJ2AWK5yO6hbFKhm4jZym+xk)!dTGF&sLf{F!fD?4H{p+{4W-OV#+3&E#=u-#c=~jU zmoK+CxnvkeY+H{_r$NA@i9NL(KP~9P!&R`K0zCodG~w+C_zi&I<-g*8^4S^m&1WP1 zg@-yf`oKR{?%{Dtm1I2*SAn^#)b1> z>@d6}B_e2@C;iz@0(R4Yp)(oZ;US*Ofdl1|f1v*TjGp2@F}T5pFzLLS*>?KspZggb-uk(jkN%zTDDv2S1Aye( zPXN%q{Co1v5a>I9|Jii-&c`+FUDjk5Y(G0Vtp{?p@<`Ux=VEaRy&>%NDJB5qaaf$D z0TN}l4X2ibl?5=BXz0k9&z{`#RKe781i+2qoPWU7w|E>n1_sVxdv##b%=r(HsORB- zw;u^n&XDI}b40w6HbJ9TCclCE>2rD!Mk`%lBH)R8s(sof`FINtjv zNwVg@Zo;sa;&S4jhYG+;njMSKO?~5E&l)0?n$6cEI~_W@%~e>O6`ne2xNs`q)N;T~ zM;gtk)9Xn5MGkA`YH=iOoW2NRjL4i>d|^l9jejeh(?hxP$NedN$Cm=+>SvD@0VwYI zOF#Z0O?$sj!}X85@Fv(0$sK~85uT7A_p6_u;V{m?&z4;!GOLlvNH@`Y>oCj7}`+YRD z>}^F1y9qYc%5px87C{~eITD4gHSDDUp%c=|*l$Bbc|s>dM*=fNvbld{XXy0 zWtqf%XpQ=_`Boxc1K`XOoLqoFbUH^nAq=2E09>UR@PS6TYVB&(D0TGRT@nNcs@p4%rq&OoLfB2&x;3P~o9=HP^bqTdkg71WhK;zzM&cF!oaTHbdeh1>YoS2gW@mnOU4?82KLCGDAU3SU~~#IK2m zJX+(g>TT2RKv>($vj03wt7yiTmU9v_w>l=4_s$>?FZ^?AQCL}+A0q4%mzXk{S*Am^Gfr87a|@LeNKrRv?_170#bDT)kjeUCQ_WG%-0= zNsT&iks;lF0Zmqr)+(~I>fZeuKVh!?$9n+5pqH_YI6mn5&yr$Yzege%AU~ zlp5=rmID?dxY!Z(QU-eoVL!np7BTrqCM6Cea^R0983K@~(07L2v_;njIX+rJd?$>W zh?GfKHiqdX8I(>SEX@l~oeFs7Lcr>YtowgzmX4X7%ln2RP|kq)T3j-#GtbN>cYfxd zx6gm`7snXC$pirLTR%3hE_&H@?|hPI?OD=r*^6bXG|^3A@`~Qo7A31$37`v6F8^EH z)ns?IG}6(sAmAAQPt3vcJj~9BU&-Y7K&DS#R3H_{e-Ji6Ewjjx)(lrR?hU&>>rdhg zVUUqB!T8ZJ9fL+xnii8d>p?Jn_sq=uAfWi&N8Gp42m+)zDU&;Y=X{`(bNeTu4_F%E z*D+Bq*yltwN;(2bWa(~24EAEueD1#h%;iB}c>nuQrEgp8r2&0wh*fAEF%QD17_s;X z1XX6*hEvOib0-aFRs!0yro#C@y^dX`lPB5f-bo0)M$G1{LEF0hOFOdq)wi?EUv>Rw z{BNE4=1Cv`-SY(P|47>NU+=>Bl4!IjRSk5ih`ZEeFekG%RsC^K0<@PC02$wH%I&s53NYtbT(%KjxR~+G{n} zE8o7h|5X{0$%XNwW%+bK*hQ`N)Y-GGQk8F{TVLzc!w;BD>=z6`PW+R-e15Kz$UFY9 z3rU#qdr&m}NxsH%Jods^G@`gSQ^bhL(6YN8GuTTr@u*DrMjA56a`4$71#blQRMLz%Yt9lNWX7%^EQ5`u1@5Vc<= z;y_*sG}-y5(q8$(B=4h@^lyEwTmQLF(tGXei8en#PI4mkd6r3a+K*qH60%G!^g`I_ zc?6(rfb_ zQ{2W`_M7edX2Y4*B;u8QRr9Dy0BhM~rs|subpXSb|9WOFLjoRco?XlrNPWJ$xKn3+ zhkv-|v;WlB{7Z;@pBU`1?}7US`wCb`I7sPj#f1H|XpF0U72tg3+BQNe<$ezYb` zl-|G-03`}DZEumw2x`K7+m$>mcK}X>z`)8PtS&;!kkS9@bn5@%A3O#8F3laMsJhJX zJvjMItyjdRND{(u*9QQxhy1nCS0dFt&}i;zPQ@3_wvh`;v$M~pzS+*k2q$4ikRYgG zOe%-q)YE%7f34_hsDmU%q8>Wf@g)xoh-pw3=RuD=wvoSkhvQXU4R4>ERnpio%XmhXZNlER>JGFvR;2`iVRcbhT@<_xpU%@hx|6Yga3HFg zW79E^YsDH?lmvYvekJQKh;9Ki6U*$9tlnQm8tSF5f;T|Mqf~VVvfs==W&YkeNI=6q zAIQus6i7gA2Ngu^4Va@w+gcB9WSqa03aeq7_^dzdB57P%CLKW#jyWDwtV_NWE>*&w zW!Oy_4ieJ1G0C6}de&l+h_Sr5LY>UbDGZ!p#|Cs$A+psn=%@yOjHBFr#U^2aqZR+1BZ54ga5t{6Z$ofmjTQfTIl-k}KC$xR`*MphmwMQBN z=+?jK;F181T%L{y!MyCy&dZ)j#=w*Nd z!}Op{!ps+=UHP*ZBMvl@#f@V|^OU|44Kj+`VP!;bOx?Z+0-92*V_of-Eb?9>iWyHp z)blW|80dfof#1)_n3o*K;HLYAtJMJm$z~1JdVT}n^f%0=dCdI=D37u8r_@&vaH(V1 zO&DgXe>QcapFwqBBFB3L0Y(&u&`k|ni6{PXsu{I#wZTwE_TnU3f@=+c**2YR3nxzq zr=cgES>NzB?4n`Mi`-aSj`K;wW2)YFfA0s2 z-xDsc1{%(xng9a|^}V?h5E$Q`Ukd`NnPfx4wVnr;?xw-(uFgW#^9_$lh-kX+CzNAt zGAn9@jhjwWN7zpY`zdK`aTc%5ai3#3QZ}i8oViQSxe)hK!$uqsT2p)8uDCP3f6(#> z*Fm!#?-W+%;ObKW=T8}rNFE>ksp)mUB!@~Mpnk^*K-vXO0rRZb&=D;B#G}587A|d% z!q<;?lr1m+hB0jSJrYplU0OGJN4mwc_ihEAZEIo169E}B-Ofk1{`nZ78<|q6UCx3_ z{CHb5@Z>(3^DiB8m1g*8oM5ghDb8kG`#G2uKzj~?*2pK)`=_j9LgI{aIEgb_W3Oi* z&hp;Zr>HmdsddD|l>Sc4u$$o0hhr#J86|s8ov@n*Y$XA)6F=0cu2e-m!WFIJk(xFC zEzTL9yI^?gw7~?^tp7ca4uH2`Y7yB$U35j^Q^{+8@=fh5-t0W{-wdjm|KR4|2Vnl< zA2?V5Ah`h?Ct>#(KfaLl01TBVk%IlNA_9oRRp@#r_aL+qorwsu9c6LW!*1;$BPW_3 zxFb6kbv8h)%QX9x7t#=o`^B8Q*nGOB3bX5oD%!+yH*I{3yeTgY(V;_~L*^{Zp2&iL zQI{f(K(yc^Y7*oWj?u!f2ZLQCttc73$DDs@nzT~3@#(E|4x3m8y@cJ3m_av1-Gj?0 zNCUkuv>2pBQ>+T16LwR>P8yJ?$%iGWO6wKBI^p%zCN;Uj(wy+z*?^~34Xt2mDvrM` zYmkZ6_8k>l1*F@E-7^|2`~hjrUlkYpsH*)1ksy9(1fT{u9xFnH-Odx-43Tuy@r`H$ zOw#!`(W;tCPlb#~3~Bjf`Zu(Kt=WZaN@}0Zf0^KJ@{FV~-!Ys# zVK}{PI5BT9rVMhYQ%T2lPv1xEtJZp!0yYqBJtMBwLW5ns_G2facmK{W&0P5p)((^T z7Xd)wK97m>FChRSjA6I$O94okylI67wl3BqkNiH$Icl%7Z8HAb3-7c@GtC%{#rCOWuFCY1&1 z(f0(5eH9ubuu^k~*tFz0NQOY}{NKafsyYS+$tpNDQsSW_8aUF}VT>GDO<61Q(0Ly~ zF^%L5y)~)Fujho9oz4FHL(!y z_{1fGL>JX1-;V8`G{Iur-TL`ov1Vp>W#y|ZfNuRO|LZ5e_Q3)GIeHF@2SNY@C5$Pi z0P2n5Qp@0S*zQs5U1&8D(^BepwgdCCUU;KEoiL>Vo_u5o8Rle}VcLXyl zbMqumTQ~f>!0<4et|j+7Xpbw}SBO)kA9{&(8I$iEh2g1!tg4C;d;l;5?VvG|Jq-em z@Ni_!Y=wkx&gb7RC49%ypf>2Oc^o`$O`}}S>XSs&i_*q72#%z|4V?<-yqs}7^z^>b z5V48xkSL~mp0s#SNK>V^9n;@UuvQQG6f$D*G2%}SFa5ih8ivl0s7C@tYPb${3E(3- z|0>M1gi|Mk)62q%c`yCzre(<|%-&Hjp`t0+UVxZOB%7-uLC|iWeN$C`1sF({$t17- z835WJ_{;YW0CMThjyC8`W+#v&#Wz zmksl?e*WL@{A_Y+YLcFWNEa%-uc{7g59z(Sz|789vU286GkE(SGjrkZ0{w%$a%)Tv z@ndsj1&V257!2t9A zu~ti1m=n&f2q%{eomNvoYuW<%*s;Ds+qsu{-2Qx z-*>M7AZwDEw8Gy3PGVxAANeLgTj5Ktvzux$8aU=8KjL`Xlg2ppb-|eq%*_a`mNcki zk6}Z_6BY;;Mb;!nlGw9m71?s088@>q8H4QTg!j$NMas$kXAZ&thT zOB1F?WV@U6p1$F*69_XcXql=k%+&cGSp;BH5L?pq>b({EI0l5b`=e|3z^gOnL*hZn zMIojf5!8(6XxN1)C;8bYb9fTUQ`K7N6BYHpY2yFHS-tUEBY@f#)-6NuG1HzC)9Rd- z#Y_JP*m{uHZpl8?|Hrw1R6Svvb!Ec0Ur0UKd#yM2f( z@`MInd8F^qH1eO>1lb0h^FI>jdg3^LRgxYjQTh)6i;o@lNJKp0}^VZJR=&P#fFBcx2@~{ z;QwRhBe!Jf1 z4Au($os{0Tr~WN^KE{zd+|zdE&3`#1nnA>2P(mkcB>_Dfcsq3Dx&HXeX0~u)HehuD zRu%*^BVcogIBPUjj**=6^g8_hubVurAVC0X039N6G*A1~H({4vChGm~%<#Vgx9^Dn z9GkSia-|N2F~q50U6y%276+iz279tEgTUAS%{0iKUX$E?)sZxH*?42-?8W%(KaHRK zg=h+LD%eyRbW?`iv=RV3tc9(=-Ow6#l7OKz9GCH4Kb@eo0ywo~IJ0b+?Fa-})-Q7W z4Eppsf&jsSCW7rR5O2PO@@<4~tZDbReuc$X-UfOiG9r9p%Gc0Jz5OvBbOEbEY=zh= zE@ynprBX#ujGx6Vx&mZ^o4vv{UsY^!2FdS9%O>S%ygj`hW!SGsQJ?ndpY=8W;hskY z-WN8$lGOq@NE0ROCq%;(PxdGex6UHI$1vyhSCh^Ew}_p|m{O0y30F`jbOOVL)qr!W z0YM;nFL#z1YQ|(ZwMI`&09 z8@REtB_5_Ep~c3oF@Y-wf6=-EdNFAks$n}dgigi~em%a+st^P|`#-yEXtl=OoKE`y zo+M+CGgek8sKY#Z(+B7@#eNwg`ckjE6v`M{asO`s{z?{EyD@7t*!w7y9XJJTHwL z(ZCXi4x6}|PRK#eVNT&Al|aBMM9z2q_pOkqWVO=cvBv6Pwgs2Y7%rU&Fos(2Qp7`s zXy{Evn$Q`G`NPt?$}hR`0J3zf0L&(&1K2LOSAg+;^369FxQ&yXrjlDSN@ z=AY@nn8@@xt_VPmduzjKmtQJXra<2?8zJftGxrtPy{F)On|NsG%zYl%ZQU>OkjHS= zt$I3OYdss67IeInl%S(g%EXplRdN>6@pr6K(T(Xf8K+R^=hXM>LKFtcrAi!HqG5tf z?TC|;!yoh_Ac&pk|BqZL0sh!BzC1WkKMrk-u)JVcT@u?tMilVS;^ z;_$WE56}n#v}dQ-wj3Z2T`8q(p-hwJtM^^3dyCg!>}IDs*i? z=nQ(q>#zHKQ}9|OMugQ9!s#Wci2pPY+DbpbRGs`uvjp16`uG1G@6z+8z%JM@74&&r z|C37a3&5RLF<>WExTz9VR=u6%&QKb+_GIobohtlIujzF>E(gvjX>195DPiB@Tqehz zb8#Fh7$}D4i$=iE8Mc#vfitD8n=l-j>kx`cA^uX{qVeO8P6+}yzXF#|33Hu#08_XK z;enjQ^m>L}eVsoI=PzRXH(m7qkA>ksgFFK>rx^nSgBSoyDtvT%DJVGr0000` element. + +$body-bg: #e4e5e6; + +// Typography: +// Font, line-height, and color for body text, headings, and more. + +$font-size-base: 1rem; + +$dropdown-link-hover-color: white; +$dropdown-link-hover-bg: #343a40; diff --git a/src/main/webapp/content/scss/global.scss b/src/main/webapp/content/scss/global.scss new file mode 100644 index 0000000..543a62c --- /dev/null +++ b/src/main/webapp/content/scss/global.scss @@ -0,0 +1,195 @@ +@import 'bootstrap-variables'; +@import '~bootstrap/scss/functions'; +@import '~bootstrap/scss/variables'; + +/* ============================================================== +Bootstrap tweaks +===============================================================*/ + +body, +h1, +h2, +h3, +h4 { + font-weight: 300; +} + +/* Increase contrast of links to get 100% on Lighthouse Accessability Audit. Override this color if you want to change the link color, or use a Bootswatch theme */ +a { + color: #533f03; + font-weight: bold; +} + +a:hover { + color: #533f03; +} + +/* override hover color for dropdown-item forced by bootstrap to all a:not([href]):not([tabindex]) elements in _reboot.scss */ +a:not([href]):not([tabindex]):hover.dropdown-item { + color: $dropdown-link-hover-color; +} + +/* override .dropdown-item.active background-color on hover */ +.dropdown-item.active:hover { + background-color: mix($dropdown-link-hover-bg, $dropdown-link-active-bg, 50%); +} + +a:hover { + /* make sure browsers use the pointer cursor for anchors, even with no href */ + cursor: pointer; +} + +.dropdown-item:hover { + color: $dropdown-link-hover-color; +} + +/* ========================================================================== +Browser Upgrade Prompt +========================================================================== */ +.browserupgrade { + margin: 0.2em 0; + background: #ccc; + color: #000; + padding: 0.2em 0; +} + +/* ========================================================================== +Generic styles +========================================================================== */ + +/* Error highlight on input fields */ +.ng-valid[required], +.ng-valid.required { + border-left: 5px solid green; +} + +.ng-invalid:not(form) { + border-left: 5px solid red; +} + +/* other generic styles */ + +.jh-card { + padding: 1.5%; + margin-top: 20px; + border: none; +} + +.error { + color: white; + background-color: red; +} + +.pad { + padding: 10px; +} + +.w-40 { + width: 40% !important; +} + +.w-60 { + width: 60% !important; +} + +.break { + white-space: normal; + word-break: break-all; +} + +.readonly { + background-color: #eee; + opacity: 1; +} + +.footer { + border-top: 1px solid rgba(0, 0, 0, 0.125); +} + +.hand, +[jhisortby] { + cursor: pointer; +} + +/* ========================================================================== +Custom alerts for notification +========================================================================== */ +.alerts { + .alert { + text-overflow: ellipsis; + pre { + background: none; + border: none; + font: inherit; + color: inherit; + padding: 0; + margin: 0; + } + .popover pre { + font-size: 10px; + } + } + .jhi-toast { + position: fixed; + width: 100%; + &.left { + left: 5px; + } + &.right { + right: 5px; + } + &.top { + top: 55px; + } + &.bottom { + bottom: 55px; + } + } +} + +@media screen and (min-width: 480px) { + .alerts .jhi-toast { + width: 50%; + } +} + +/* ========================================================================== +entity detail page css +========================================================================== */ +.row.jh-entity-details > { + dd { + margin-bottom: 15px; + } +} + +@media screen and (min-width: 768px) { + .row.jh-entity-details > { + dt { + margin-bottom: 15px; + } + dd { + border-bottom: 1px solid #eee; + padding-left: 180px; + margin-left: 0; + } + } +} + +/* ========================================================================== +ui bootstrap tweaks +========================================================================== */ +.nav, +.pagination, +.carousel, +.panel-title a { + cursor: pointer; +} + +.thread-dump-modal-lock { + max-width: 450px; + overflow: hidden; + text-overflow: ellipsis; + white-space: nowrap; +} + +/* jhipster-needle-scss-add-main JHipster will add new css style */ diff --git a/src/main/webapp/content/scss/vendor.scss b/src/main/webapp/content/scss/vendor.scss new file mode 100644 index 0000000..31688d0 --- /dev/null +++ b/src/main/webapp/content/scss/vendor.scss @@ -0,0 +1,12 @@ +/* after changing this file run 'npm run webpack:build' */ + +/*************************** +put Sass variables here: +eg $input-color: red; +****************************/ +// Override Bootstrap variables +@import 'bootstrap-variables'; +// Import Bootstrap source files from node_modules +@import '~bootstrap/scss/bootstrap'; + +/* jhipster-needle-scss-add-vendor JHipster will add new css style */ diff --git a/src/main/webapp/favicon.ico b/src/main/webapp/favicon.ico new file mode 100644 index 0000000000000000000000000000000000000000..6f6fe68f5f6f013e196bffdc707770f2dc1e7c73 GIT binary patch literal 16958 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 qR09=ZeO~SP3!!`$oOA8Ct#&<}u + + + + + epasmed + + + + + + + + + + + +

+ + + + + + + + diff --git a/src/main/webapp/manifest.webapp b/src/main/webapp/manifest.webapp new file mode 100644 index 0000000..804c286 --- /dev/null +++ b/src/main/webapp/manifest.webapp @@ -0,0 +1,31 @@ +{ + "name": "ePASMed", + "short_name": "ePASMed", + "icons": [ + { + "src": "./content/images/plumber_0_head_192.png", + "sizes": "192x192", + "type": "image/png" + }, + { + "src": "./content/images/plumber_0_head_head_256.png", + "sizes": "256x256", + "type": "image/png" + }, + { + "src": "./content/images/plumber_0_head_head_384.png", + "sizes": "384x384", + "type": "image/png" + }, + { + "src": "./content/images/plumber_0_head_head_512.png", + "sizes": "512x512", + "type": "image/png" + } + ], + "theme_color": "#000000", + "background_color": "#e0e0e0", + "start_url": ".", + "display": "standalone", + "orientation": "portrait" +} diff --git a/src/main/webapp/robots.txt b/src/main/webapp/robots.txt new file mode 100644 index 0000000..7cda274 --- /dev/null +++ b/src/main/webapp/robots.txt @@ -0,0 +1,11 @@ +# robotstxt.org/ + +User-agent: * +Disallow: /api/account +Disallow: /api/account/change-password +Disallow: /api/account/sessions +Disallow: /api/audits/ +Disallow: /api/logs/ +Disallow: /api/users/ +Disallow: /management/ +Disallow: /v2/api-docs/ diff --git a/src/main/webapp/swagger-ui/dist/images/throbber.gif b/src/main/webapp/swagger-ui/dist/images/throbber.gif new file mode 100644 index 0000000000000000000000000000000000000000..06393889242fb3ea9e0205fa84369ec7bb66d15a GIT binary patch literal 9257 zcmd^^X;@R|x`tQg5wbE8AV3mAn1TjmQ&en2CK8~ENEH<+P_)pZ24y2E+7O0>K^a6u zQ3;5MiU^7p6*M3qDk!2=YEcHMQ>nzEYP;R`e2C@r+U+?#XaC*&gKPcB#k$`o&;7mu zYNhYYXe|Uo84#4ZIko#rcU5K8*yFL{qT47O&^5fZH$ zVZ@%(l~vVHjnm;H@KL8@r%yUHoo;rbHI_4lIH(_nsTT>S2`DFOD~uCb9_dF4`#QgI zy7ldMcLs+A_s%|e1pRPrbX-tpeNP!9(IpMFTce`t_5U%lP99z%&i6`1d~ zWeM!Rxc50<+d$e^9LT`?B+aMK~apR zHm?q;p<7{wN2g|I^aGlSws;VP84j(z%aQwvAWv83Z$}p(% zZ^?2;gxg(ey_`V5J7{;!o;o;KslW@z5EP~JGs|U)J7dF&(ff#A=6vU?cGQ$-4+;Jf z-ggJEa!yStn`_EWvl)#yhm6XVs}UUbsi;+agri;mCfjH^Uy;lH+Zw^h)4N?oZgZz4 zJk(fTZ|Bi^;+s_M=~+d#vyoxEPzTlOS=mX@sbl*uRj>=MaMr}cFIY8i?UM61>86uB zV$DlOUCiUJwbzJMP@D$urzK|lL2-PC!p1l47V-ZG<5Ev0Z5h~Kx?`KOp7gkAjV93A z-Gc7MrlxTf?wF;CbNc@tCHJH{TB3c;#{SVu%97}tyAM2n&|9W_?qv}$*Jt*%7Yxb# zV0;d;7|lDEltJYS+U)#aiJO};?_Jyy_4%syQ(uy?-J-Yx-9O5nKRk@@XSS~X<(2u~ zV-LamWm~!iqtH9wkpf8mAXZhOD&L#aA_%)4h2M;1M5jt zIR>Us+%W-GXa_f^opKg=DSrAs)AXeRa;Hp0aC1OgbxQ%Qr_QvTleM1jkR!2mkcX$3 ztsR8~G9iqh(-FJ@F_rQBIYDXV_6s7G9SxaVF^laZqcx$!D97m|7t16j6@Jt6UdDRy49Qyvs|c>RuA|@b%}`*wU}2^7q;&Vtc6@lb zcXl)T!6nYDzmMJ~%n$KNXyNlCG)GkJ4!82;v6@d3>s5r~E+3!O?049JDr14Y^PeMI02R`0lJ^=oJ zYd|*u9|SU(j7hY?+<=(?fP*mtV*zFhOrz6%{VA?ozdm&(Jf^V zMfPZ?>l`mS3{Uq8IM;e!+1YjJy2!mzK$O|wPeU{*QSbs9m+@`f5KxO3PBnQ=%RsZg%go*fJ`*w9TL{-WgZVIA$!YV}3BRcfeXaR$x#b zW)Tpd#8E4)^MyYdkH;4_;ChJuw%n+Be7Ko4;w-nHvyo$d_0e-YiF78Df&)_)(}fcr_r0mPH(4RRYWIu+d@t0&Ss@O^s! zOKyX&13)%N@83r^;QsgN{rl(!0|RF1FA)b1{CRXAy&1ySz@>olPiR4r$aMdq&_=nK zq|cFs8phWJ1@%dZ-gXd{zDbTILD>)qEvH-NU*Rf1b2J1Ri79`rBFl@ z8E^0I)OqEi{pH(a24b9YPG;Kz@t-qZW;3Mpe`MRlmYx{7bH-XZ&`RQ7Rb^%}gc&X| zd}Q-FZf|RWxHU?PR!(C?80zu(^l>*h{#ulSiid(O!J(8P-41bNM3tnX@U6NS5yo0? zdcF)~xFE&+&|gZ$23dV5t~?$$&ymZ;F8j7GGMncGSsDo%>J`26=&l=X#rSKv_64;0 zr;k6no@=gV`P)K!=kaHl>q?!`X>(A;84tg^Md<`zA%qbRLby1Z=fn*ZRdNqs%Tq|3 zOt}lZu0q9oKJhgz&+^7PCt$=UFW=R*w?a1)ePoL*`R$Gxj?TU@12tTHsT$giHQU+sqf;fS0FpT!< z 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 + + + + ePASMed - Swagger UI + + + + + + +
+ + + + + + + + + diff --git a/src/test/java/it/cnr/isti/epasmed/ArchTest.java b/src/test/java/it/cnr/isti/epasmed/ArchTest.java new file mode 100644 index 0000000..d5db4ac --- /dev/null +++ b/src/test/java/it/cnr/isti/epasmed/ArchTest.java @@ -0,0 +1,29 @@ +package it.cnr.isti.epasmed; + +import com.tngtech.archunit.core.domain.JavaClasses; +import com.tngtech.archunit.core.importer.ClassFileImporter; +import com.tngtech.archunit.core.importer.ImportOption; +import org.junit.jupiter.api.Test; + +import static com.tngtech.archunit.lang.syntax.ArchRuleDefinition.noClasses; + +class ArchTest { + + @Test + void servicesAndRepositoriesShouldNotDependOnWebLayer() { + + JavaClasses importedClasses = new ClassFileImporter() + .withImportOption(ImportOption.Predefined.DO_NOT_INCLUDE_TESTS) + .importPackages("it.cnr.isti.epasmed"); + + noClasses() + .that() + .resideInAnyPackage("it.cnr.isti.epasmed.service..") + .or() + .resideInAnyPackage("it.cnr.isti.epasmed.repository..") + .should().dependOnClassesThat() + .resideInAnyPackage("..it.cnr.isti.epasmed.web..") + .because("Services and repositories should not depend on web layer") + .check(importedClasses); + } +} diff --git a/src/test/java/it/cnr/isti/epasmed/config/NoOpMailConfiguration.java b/src/test/java/it/cnr/isti/epasmed/config/NoOpMailConfiguration.java new file mode 100644 index 0000000..86f6de8 --- /dev/null +++ b/src/test/java/it/cnr/isti/epasmed/config/NoOpMailConfiguration.java @@ -0,0 +1,24 @@ +package it.cnr.isti.epasmed.config; + +import it.cnr.isti.epasmed.service.MailService; +import org.springframework.context.annotation.Bean; +import org.springframework.context.annotation.Configuration; + +import static org.mockito.ArgumentMatchers.any; +import static org.mockito.Mockito.doNothing; +import static org.mockito.Mockito.mock; + +@Configuration +public class NoOpMailConfiguration { + private final MailService mockMailService; + + public NoOpMailConfiguration() { + mockMailService = mock(MailService.class); + doNothing().when(mockMailService).sendActivationEmail(any()); + } + + @Bean + public MailService mailService() { + return mockMailService; + } +} diff --git a/src/test/java/it/cnr/isti/epasmed/config/StaticResourcesWebConfigurerTest.java b/src/test/java/it/cnr/isti/epasmed/config/StaticResourcesWebConfigurerTest.java new file mode 100644 index 0000000..72186d2 --- /dev/null +++ b/src/test/java/it/cnr/isti/epasmed/config/StaticResourcesWebConfigurerTest.java @@ -0,0 +1,80 @@ +package it.cnr.isti.epasmed.config; + +import io.github.jhipster.config.JHipsterDefaults; +import io.github.jhipster.config.JHipsterProperties; +import org.junit.jupiter.api.BeforeEach; +import org.junit.jupiter.api.Test; +import org.springframework.http.CacheControl; +import org.springframework.mock.web.MockServletContext; +import org.springframework.web.context.WebApplicationContext; +import org.springframework.web.servlet.config.annotation.ResourceHandlerRegistration; +import org.springframework.web.servlet.config.annotation.ResourceHandlerRegistry; + +import java.util.concurrent.TimeUnit; + +import static it.cnr.isti.epasmed.config.StaticResourcesWebConfiguration.*; +import static org.assertj.core.api.Assertions.assertThat; +import static org.mockito.Mockito.*; + +public class StaticResourcesWebConfigurerTest { + public static final int MAX_AGE_TEST = 5; + public StaticResourcesWebConfiguration staticResourcesWebConfiguration; + private ResourceHandlerRegistry resourceHandlerRegistry; + private MockServletContext servletContext; + private WebApplicationContext applicationContext; + private JHipsterProperties props; + + @BeforeEach + void setUp() { + servletContext = spy(new MockServletContext()); + applicationContext = mock(WebApplicationContext.class); + resourceHandlerRegistry = spy(new ResourceHandlerRegistry(applicationContext, servletContext)); + props = new JHipsterProperties(); + staticResourcesWebConfiguration = spy(new StaticResourcesWebConfiguration(props)); + } + + @Test + public void shouldAppendResourceHandlerAndInitiliazeIt() { + + staticResourcesWebConfiguration.addResourceHandlers(resourceHandlerRegistry); + + verify(resourceHandlerRegistry, times(1)) + .addResourceHandler(RESOURCE_PATHS); + verify(staticResourcesWebConfiguration, times(1)) + .initializeResourceHandler(any(ResourceHandlerRegistration.class)); + for (String testingPath : RESOURCE_PATHS) { + assertThat(resourceHandlerRegistry.hasMappingForPattern(testingPath)).isTrue(); + } + } + + @Test + public void shouldInitializeResourceHandlerWithCacheControlAndLocations() { + CacheControl ccExpected = CacheControl.maxAge(5, TimeUnit.DAYS).cachePublic(); + when(staticResourcesWebConfiguration.getCacheControl()).thenReturn(ccExpected); + ResourceHandlerRegistration resourceHandlerRegistration = spy(new ResourceHandlerRegistration(RESOURCE_PATHS)); + + staticResourcesWebConfiguration.initializeResourceHandler(resourceHandlerRegistration); + + verify(staticResourcesWebConfiguration, times(1)).getCacheControl(); + verify(resourceHandlerRegistration, times(1)).setCacheControl(ccExpected); + verify(resourceHandlerRegistration, times(1)).addResourceLocations(RESOURCE_LOCATIONS); + } + + + @Test + public void shoudCreateCacheControlBasedOnJhipsterDefaultProperties() { + CacheControl cacheExpected = CacheControl.maxAge(JHipsterDefaults.Http.Cache.timeToLiveInDays, TimeUnit.DAYS).cachePublic(); + assertThat(staticResourcesWebConfiguration.getCacheControl()) + .extracting(CacheControl::getHeaderValue) + .isEqualTo(cacheExpected.getHeaderValue()); + } + + @Test + public void shoudCreateCacheControlWithSpecificConfigurationInProperties() { + props.getHttp().getCache().setTimeToLiveInDays(MAX_AGE_TEST); + CacheControl cacheExpected = CacheControl.maxAge(MAX_AGE_TEST, TimeUnit.DAYS).cachePublic(); + assertThat(staticResourcesWebConfiguration.getCacheControl()) + .extracting(CacheControl::getHeaderValue) + .isEqualTo(cacheExpected.getHeaderValue()); + } +} diff --git a/src/test/java/it/cnr/isti/epasmed/config/WebConfigurerTest.java b/src/test/java/it/cnr/isti/epasmed/config/WebConfigurerTest.java new file mode 100644 index 0000000..6348867 --- /dev/null +++ b/src/test/java/it/cnr/isti/epasmed/config/WebConfigurerTest.java @@ -0,0 +1,165 @@ +package it.cnr.isti.epasmed.config; + +import io.github.jhipster.config.JHipsterConstants; +import io.github.jhipster.config.JHipsterProperties; +import io.github.jhipster.web.filter.CachingHttpHeadersFilter; +import org.h2.server.web.WebServlet; +import org.junit.jupiter.api.BeforeEach; +import org.junit.jupiter.api.Test; +import org.springframework.boot.web.embedded.undertow.UndertowServletWebServerFactory; +import org.springframework.http.HttpHeaders; +import org.springframework.mock.env.MockEnvironment; +import org.springframework.mock.web.MockServletContext; +import org.springframework.test.web.servlet.MockMvc; +import org.springframework.test.web.servlet.setup.MockMvcBuilders; + +import javax.servlet.*; +import java.io.File; +import java.util.*; + +import static org.assertj.core.api.Assertions.assertThat; +import static org.mockito.ArgumentMatchers.any; +import static org.mockito.ArgumentMatchers.anyString; +import static org.mockito.Mockito.*; +import static org.springframework.test.web.servlet.request.MockMvcRequestBuilders.get; +import static org.springframework.test.web.servlet.request.MockMvcRequestBuilders.options; +import static org.springframework.test.web.servlet.result.MockMvcResultMatchers.header; +import static org.springframework.test.web.servlet.result.MockMvcResultMatchers.status; + +/** + * Unit tests for the {@link WebConfigurer} class. + */ +public class WebConfigurerTest { + + private WebConfigurer webConfigurer; + + private MockServletContext servletContext; + + private MockEnvironment env; + + private JHipsterProperties props; + + @BeforeEach + public void setup() { + servletContext = spy(new MockServletContext()); + doReturn(mock(FilterRegistration.Dynamic.class)) + .when(servletContext).addFilter(anyString(), any(Filter.class)); + doReturn(mock(ServletRegistration.Dynamic.class)) + .when(servletContext).addServlet(anyString(), any(Servlet.class)); + + env = new MockEnvironment(); + props = new JHipsterProperties(); + + webConfigurer = new WebConfigurer(env, props); + } + + @Test + public void testStartUpProdServletContext() throws ServletException { + env.setActiveProfiles(JHipsterConstants.SPRING_PROFILE_PRODUCTION); + webConfigurer.onStartup(servletContext); + + + verify(servletContext, never()).addServlet(eq("H2Console"), any(WebServlet.class)); + } + + @Test + public void testStartUpDevServletContext() throws ServletException { + env.setActiveProfiles(JHipsterConstants.SPRING_PROFILE_DEVELOPMENT); + webConfigurer.onStartup(servletContext); + + + verify(servletContext).addServlet(eq("H2Console"), any(WebServlet.class)); + } + + @Test + public void testCustomizeServletContainer() { + env.setActiveProfiles(JHipsterConstants.SPRING_PROFILE_PRODUCTION); + UndertowServletWebServerFactory container = new UndertowServletWebServerFactory(); + webConfigurer.customize(container); + assertThat(container.getMimeMappings().get("abs")).isEqualTo("audio/x-mpeg"); + assertThat(container.getMimeMappings().get("html")).isEqualTo("text/html;charset=utf-8"); + assertThat(container.getMimeMappings().get("json")).isEqualTo("text/html;charset=utf-8"); + if (container.getDocumentRoot() != null) { + assertThat(container.getDocumentRoot()).isEqualTo(new File("target/classes/static/")); + } + } + + @Test + public void testCorsFilterOnApiPath() throws Exception { + props.getCors().setAllowedOrigins(Collections.singletonList("*")); + props.getCors().setAllowedMethods(Arrays.asList("GET", "POST", "PUT", "DELETE")); + props.getCors().setAllowedHeaders(Collections.singletonList("*")); + props.getCors().setMaxAge(1800L); + props.getCors().setAllowCredentials(true); + + MockMvc mockMvc = MockMvcBuilders.standaloneSetup(new WebConfigurerTestController()) + .addFilters(webConfigurer.corsFilter()) + .build(); + + mockMvc.perform( + options("/api/test-cors") + .header(HttpHeaders.ORIGIN, "other.domain.com") + .header(HttpHeaders.ACCESS_CONTROL_REQUEST_METHOD, "POST")) + .andExpect(status().isOk()) + .andExpect(header().string(HttpHeaders.ACCESS_CONTROL_ALLOW_ORIGIN, "other.domain.com")) + .andExpect(header().string(HttpHeaders.VARY, "Origin")) + .andExpect(header().string(HttpHeaders.ACCESS_CONTROL_ALLOW_METHODS, "GET,POST,PUT,DELETE")) + .andExpect(header().string(HttpHeaders.ACCESS_CONTROL_ALLOW_CREDENTIALS, "true")) + .andExpect(header().string(HttpHeaders.ACCESS_CONTROL_MAX_AGE, "1800")); + + mockMvc.perform( + get("/api/test-cors") + .header(HttpHeaders.ORIGIN, "other.domain.com")) + .andExpect(status().isOk()) + .andExpect(header().string(HttpHeaders.ACCESS_CONTROL_ALLOW_ORIGIN, "other.domain.com")); + } + + @Test + public void testCorsFilterOnOtherPath() throws Exception { + props.getCors().setAllowedOrigins(Collections.singletonList("*")); + props.getCors().setAllowedMethods(Arrays.asList("GET", "POST", "PUT", "DELETE")); + props.getCors().setAllowedHeaders(Collections.singletonList("*")); + props.getCors().setMaxAge(1800L); + props.getCors().setAllowCredentials(true); + + MockMvc mockMvc = MockMvcBuilders.standaloneSetup(new WebConfigurerTestController()) + .addFilters(webConfigurer.corsFilter()) + .build(); + + mockMvc.perform( + get("/test/test-cors") + .header(HttpHeaders.ORIGIN, "other.domain.com")) + .andExpect(status().isOk()) + .andExpect(header().doesNotExist(HttpHeaders.ACCESS_CONTROL_ALLOW_ORIGIN)); + } + + @Test + public void testCorsFilterDeactivated() throws Exception { + props.getCors().setAllowedOrigins(null); + + MockMvc mockMvc = MockMvcBuilders.standaloneSetup(new WebConfigurerTestController()) + .addFilters(webConfigurer.corsFilter()) + .build(); + + mockMvc.perform( + get("/api/test-cors") + .header(HttpHeaders.ORIGIN, "other.domain.com")) + .andExpect(status().isOk()) + .andExpect(header().doesNotExist(HttpHeaders.ACCESS_CONTROL_ALLOW_ORIGIN)); + } + + @Test + public void testCorsFilterDeactivated2() throws Exception { + props.getCors().setAllowedOrigins(new ArrayList<>()); + + MockMvc mockMvc = MockMvcBuilders.standaloneSetup(new WebConfigurerTestController()) + .addFilters(webConfigurer.corsFilter()) + .build(); + + mockMvc.perform( + get("/api/test-cors") + .header(HttpHeaders.ORIGIN, "other.domain.com")) + .andExpect(status().isOk()) + .andExpect(header().doesNotExist(HttpHeaders.ACCESS_CONTROL_ALLOW_ORIGIN)); + } +} diff --git a/src/test/java/it/cnr/isti/epasmed/config/WebConfigurerTestController.java b/src/test/java/it/cnr/isti/epasmed/config/WebConfigurerTestController.java new file mode 100644 index 0000000..cd422d8 --- /dev/null +++ b/src/test/java/it/cnr/isti/epasmed/config/WebConfigurerTestController.java @@ -0,0 +1,16 @@ +package it.cnr.isti.epasmed.config; + +import org.springframework.web.bind.annotation.GetMapping; +import org.springframework.web.bind.annotation.RestController; + +@RestController +public class WebConfigurerTestController { + + @GetMapping("/api/test-cors") + public void testCorsOnApiPath() { + } + + @GetMapping("/test/test-cors") + public void testCorsOnOtherPath() { + } +} diff --git a/src/test/java/it/cnr/isti/epasmed/repository/CustomAuditEventRepositoryIT.java b/src/test/java/it/cnr/isti/epasmed/repository/CustomAuditEventRepositoryIT.java new file mode 100644 index 0000000..5cd1896 --- /dev/null +++ b/src/test/java/it/cnr/isti/epasmed/repository/CustomAuditEventRepositoryIT.java @@ -0,0 +1,158 @@ +package it.cnr.isti.epasmed.repository; + +import it.cnr.isti.epasmed.EpasmedApp; + +import it.cnr.isti.epasmed.config.Constants; +import it.cnr.isti.epasmed.config.audit.AuditEventConverter; +import it.cnr.isti.epasmed.domain.PersistentAuditEvent; +import org.junit.jupiter.api.BeforeEach; +import org.junit.jupiter.api.Test; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.boot.actuate.audit.AuditEvent; +import org.springframework.boot.test.context.SpringBootTest; +import org.springframework.mock.web.MockHttpServletRequest; +import org.springframework.mock.web.MockHttpSession; +import org.springframework.security.web.authentication.WebAuthenticationDetails; +import org.springframework.transaction.annotation.Transactional; + +import javax.servlet.http.HttpSession; +import java.time.Instant; +import java.time.temporal.ChronoUnit; +import java.util.HashMap; +import java.util.List; +import java.util.Map; + +import static org.assertj.core.api.Assertions.assertThat; +import static it.cnr.isti.epasmed.repository.CustomAuditEventRepository.EVENT_DATA_COLUMN_MAX_LENGTH; + +/** + * Integration tests for {@link CustomAuditEventRepository}. + */ +@SpringBootTest(classes = EpasmedApp.class) +@Transactional(value="epasMedTransactionManager") +public class CustomAuditEventRepositoryIT { + + @Autowired + private PersistenceAuditEventRepository persistenceAuditEventRepository; + + @Autowired + private AuditEventConverter auditEventConverter; + + private CustomAuditEventRepository customAuditEventRepository; + + @BeforeEach + public void setup() { + customAuditEventRepository = new CustomAuditEventRepository(persistenceAuditEventRepository, auditEventConverter); + persistenceAuditEventRepository.deleteAll(); + Instant oneHourAgo = Instant.now().minusSeconds(3600); + + PersistentAuditEvent testUserEvent = new PersistentAuditEvent(); + testUserEvent.setPrincipal("test-user"); + testUserEvent.setAuditEventType("test-type"); + testUserEvent.setAuditEventDate(oneHourAgo); + Map data = new HashMap<>(); + data.put("test-key", "test-value"); + testUserEvent.setData(data); + + PersistentAuditEvent testOldUserEvent = new PersistentAuditEvent(); + testOldUserEvent.setPrincipal("test-user"); + testOldUserEvent.setAuditEventType("test-type"); + testOldUserEvent.setAuditEventDate(oneHourAgo.minusSeconds(10000)); + + PersistentAuditEvent testOtherUserEvent = new PersistentAuditEvent(); + testOtherUserEvent.setPrincipal("other-test-user"); + testOtherUserEvent.setAuditEventType("test-type"); + testOtherUserEvent.setAuditEventDate(oneHourAgo); + } + + @Test + public void addAuditEvent() { + Map data = new HashMap<>(); + data.put("test-key", "test-value"); + AuditEvent event = new AuditEvent("test-user", "test-type", data); + customAuditEventRepository.add(event); + List persistentAuditEvents = persistenceAuditEventRepository.findAll(); + assertThat(persistentAuditEvents).hasSize(1); + PersistentAuditEvent persistentAuditEvent = persistentAuditEvents.get(0); + assertThat(persistentAuditEvent.getPrincipal()).isEqualTo(event.getPrincipal()); + assertThat(persistentAuditEvent.getAuditEventType()).isEqualTo(event.getType()); + assertThat(persistentAuditEvent.getData()).containsKey("test-key"); + assertThat(persistentAuditEvent.getData().get("test-key")).isEqualTo("test-value"); + assertThat(persistentAuditEvent.getAuditEventDate().truncatedTo(ChronoUnit.MILLIS)) + .isEqualTo(event.getTimestamp().truncatedTo(ChronoUnit.MILLIS)); + } + + @Test + public void addAuditEventTruncateLargeData() { + Map data = new HashMap<>(); + StringBuilder largeData = new StringBuilder(); + for (int i = 0; i < EVENT_DATA_COLUMN_MAX_LENGTH + 10; i++) { + largeData.append("a"); + } + data.put("test-key", largeData); + AuditEvent event = new AuditEvent("test-user", "test-type", data); + customAuditEventRepository.add(event); + List persistentAuditEvents = persistenceAuditEventRepository.findAll(); + assertThat(persistentAuditEvents).hasSize(1); + PersistentAuditEvent persistentAuditEvent = persistentAuditEvents.get(0); + assertThat(persistentAuditEvent.getPrincipal()).isEqualTo(event.getPrincipal()); + assertThat(persistentAuditEvent.getAuditEventType()).isEqualTo(event.getType()); + assertThat(persistentAuditEvent.getData()).containsKey("test-key"); + String actualData = persistentAuditEvent.getData().get("test-key"); + assertThat(actualData.length()).isEqualTo(EVENT_DATA_COLUMN_MAX_LENGTH); + assertThat(actualData).isSubstringOf(largeData); + assertThat(persistentAuditEvent.getAuditEventDate().truncatedTo(ChronoUnit.MILLIS)) + .isEqualTo(event.getTimestamp().truncatedTo(ChronoUnit.MILLIS)); + } + + @Test + public void testAddEventWithWebAuthenticationDetails() { + HttpSession session = new MockHttpSession(null, "test-session-id"); + MockHttpServletRequest request = new MockHttpServletRequest(); + request.setSession(session); + request.setRemoteAddr("1.2.3.4"); + WebAuthenticationDetails details = new WebAuthenticationDetails(request); + Map data = new HashMap<>(); + data.put("test-key", details); + AuditEvent event = new AuditEvent("test-user", "test-type", data); + customAuditEventRepository.add(event); + List persistentAuditEvents = persistenceAuditEventRepository.findAll(); + assertThat(persistentAuditEvents).hasSize(1); + PersistentAuditEvent persistentAuditEvent = persistentAuditEvents.get(0); + assertThat(persistentAuditEvent.getData().get("remoteAddress")).isEqualTo("1.2.3.4"); + assertThat(persistentAuditEvent.getData().get("sessionId")).isEqualTo("test-session-id"); + } + + @Test + public void testAddEventWithNullData() { + Map data = new HashMap<>(); + data.put("test-key", null); + AuditEvent event = new AuditEvent("test-user", "test-type", data); + customAuditEventRepository.add(event); + List persistentAuditEvents = persistenceAuditEventRepository.findAll(); + assertThat(persistentAuditEvents).hasSize(1); + PersistentAuditEvent persistentAuditEvent = persistentAuditEvents.get(0); + assertThat(persistentAuditEvent.getData().get("test-key")).isEqualTo("null"); + } + + @Test + public void addAuditEventWithAnonymousUser() { + Map data = new HashMap<>(); + data.put("test-key", "test-value"); + AuditEvent event = new AuditEvent(Constants.ANONYMOUS_USER, "test-type", data); + customAuditEventRepository.add(event); + List persistentAuditEvents = persistenceAuditEventRepository.findAll(); + assertThat(persistentAuditEvents).hasSize(0); + } + + @Test + public void addAuditEventWithAuthorizationFailureType() { + Map data = new HashMap<>(); + data.put("test-key", "test-value"); + AuditEvent event = new AuditEvent("test-user", "AUTHORIZATION_FAILURE", data); + customAuditEventRepository.add(event); + List persistentAuditEvents = persistenceAuditEventRepository.findAll(); + assertThat(persistentAuditEvents).hasSize(0); + } + +} diff --git a/src/test/java/it/cnr/isti/epasmed/security/DomainUserDetailsServiceIT.java b/src/test/java/it/cnr/isti/epasmed/security/DomainUserDetailsServiceIT.java new file mode 100644 index 0000000..8647d53 --- /dev/null +++ b/src/test/java/it/cnr/isti/epasmed/security/DomainUserDetailsServiceIT.java @@ -0,0 +1,115 @@ +package it.cnr.isti.epasmed.security; + +import it.cnr.isti.epasmed.EpasmedApp; +import it.cnr.isti.epasmed.domain.User; +import it.cnr.isti.epasmed.repository.UserRepository; + +import org.apache.commons.lang3.RandomStringUtils; +import org.junit.jupiter.api.BeforeEach; +import org.junit.jupiter.api.Test; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.boot.test.context.SpringBootTest; +import org.springframework.security.core.userdetails.UserDetails; +import org.springframework.security.core.userdetails.UserDetailsService; +import org.springframework.transaction.annotation.Transactional; + +import java.util.Locale; + +import static org.assertj.core.api.Assertions.assertThat; +import static org.assertj.core.api.Assertions.assertThatExceptionOfType; + +/** + * Integrations tests for {@link DomainUserDetailsService}. + */ +@SpringBootTest(classes = EpasmedApp.class) +@Transactional(value="epasMedTransactionManager") +public class DomainUserDetailsServiceIT { + + private static final String USER_ONE_LOGIN = "test-user-one"; + private static final String USER_ONE_EMAIL = "test-user-one@localhost"; + private static final String USER_TWO_LOGIN = "test-user-two"; + private static final String USER_TWO_EMAIL = "test-user-two@localhost"; + private static final String USER_THREE_LOGIN = "test-user-three"; + private static final String USER_THREE_EMAIL = "test-user-three@localhost"; + + @Autowired + private UserRepository userRepository; + + @Autowired + private UserDetailsService domainUserDetailsService; + + @BeforeEach + public void init() { + User userOne = new User(); + userOne.setLogin(USER_ONE_LOGIN); + userOne.setPassword(RandomStringUtils.random(60)); + userOne.setActivated(true); + userOne.setEmail(USER_ONE_EMAIL); + userOne.setFirstName("userOne"); + userOne.setLastName("doe"); + userOne.setLangKey("en"); + userRepository.save(userOne); + + User userTwo = new User(); + userTwo.setLogin(USER_TWO_LOGIN); + userTwo.setPassword(RandomStringUtils.random(60)); + userTwo.setActivated(true); + userTwo.setEmail(USER_TWO_EMAIL); + userTwo.setFirstName("userTwo"); + userTwo.setLastName("doe"); + userTwo.setLangKey("en"); + userRepository.save(userTwo); + + User userThree = new User(); + userThree.setLogin(USER_THREE_LOGIN); + userThree.setPassword(RandomStringUtils.random(60)); + userThree.setActivated(false); + userThree.setEmail(USER_THREE_EMAIL); + userThree.setFirstName("userThree"); + userThree.setLastName("doe"); + userThree.setLangKey("en"); + userRepository.save(userThree); + } + + @Test + public void assertThatUserCanBeFoundByLogin() { + UserDetails userDetails = domainUserDetailsService.loadUserByUsername(USER_ONE_LOGIN); + assertThat(userDetails).isNotNull(); + assertThat(userDetails.getUsername()).isEqualTo(USER_ONE_LOGIN); + } + + @Test + public void assertThatUserCanBeFoundByLoginIgnoreCase() { + UserDetails userDetails = domainUserDetailsService.loadUserByUsername(USER_ONE_LOGIN.toUpperCase(Locale.ENGLISH)); + assertThat(userDetails).isNotNull(); + assertThat(userDetails.getUsername()).isEqualTo(USER_ONE_LOGIN); + } + + @Test + public void assertThatUserCanBeFoundByEmail() { + UserDetails userDetails = domainUserDetailsService.loadUserByUsername(USER_TWO_EMAIL); + assertThat(userDetails).isNotNull(); + assertThat(userDetails.getUsername()).isEqualTo(USER_TWO_LOGIN); + } + + @Test + public void assertThatUserCanBeFoundByEmailIgnoreCase() { + UserDetails userDetails = domainUserDetailsService.loadUserByUsername(USER_TWO_EMAIL.toUpperCase(Locale.ENGLISH)); + assertThat(userDetails).isNotNull(); + assertThat(userDetails.getUsername()).isEqualTo(USER_TWO_LOGIN); + } + + @Test + public void assertThatEmailIsPrioritizedOverLogin() { + UserDetails userDetails = domainUserDetailsService.loadUserByUsername(USER_ONE_EMAIL); + assertThat(userDetails).isNotNull(); + assertThat(userDetails.getUsername()).isEqualTo(USER_ONE_LOGIN); + } + + @Test + public void assertThatUserNotActivatedExceptionIsThrownForNotActivatedUsers() { + assertThatExceptionOfType(UserNotActivatedException.class).isThrownBy( + () -> domainUserDetailsService.loadUserByUsername(USER_THREE_LOGIN)); + } + +} diff --git a/src/test/java/it/cnr/isti/epasmed/security/SecurityUtilsUnitTest.java b/src/test/java/it/cnr/isti/epasmed/security/SecurityUtilsUnitTest.java new file mode 100644 index 0000000..ea53147 --- /dev/null +++ b/src/test/java/it/cnr/isti/epasmed/security/SecurityUtilsUnitTest.java @@ -0,0 +1,62 @@ +package it.cnr.isti.epasmed.security; + +import org.junit.jupiter.api.Test; +import org.springframework.security.authentication.UsernamePasswordAuthenticationToken; +import org.springframework.security.core.GrantedAuthority; +import org.springframework.security.core.authority.SimpleGrantedAuthority; +import org.springframework.security.core.context.SecurityContext; +import org.springframework.security.core.context.SecurityContextHolder; + +import java.util.ArrayList; +import java.util.Collection; +import java.util.Optional; + +import static org.assertj.core.api.Assertions.assertThat; + +/** + * Test class for the {@link SecurityUtils} utility class. + */ +public class SecurityUtilsUnitTest { + + @Test + public void testGetCurrentUserLogin() { + SecurityContext securityContext = SecurityContextHolder.createEmptyContext(); + securityContext.setAuthentication(new UsernamePasswordAuthenticationToken("admin", "admin")); + SecurityContextHolder.setContext(securityContext); + Optional login = SecurityUtils.getCurrentUserLogin(); + assertThat(login).contains("admin"); + } + + @Test + public void testIsAuthenticated() { + SecurityContext securityContext = SecurityContextHolder.createEmptyContext(); + securityContext.setAuthentication(new UsernamePasswordAuthenticationToken("admin", "admin")); + SecurityContextHolder.setContext(securityContext); + boolean isAuthenticated = SecurityUtils.isAuthenticated(); + assertThat(isAuthenticated).isTrue(); + } + + @Test + public void testAnonymousIsNotAuthenticated() { + SecurityContext securityContext = SecurityContextHolder.createEmptyContext(); + Collection authorities = new ArrayList<>(); + authorities.add(new SimpleGrantedAuthority(AuthoritiesConstants.ANONYMOUS)); + securityContext.setAuthentication(new UsernamePasswordAuthenticationToken("anonymous", "anonymous", authorities)); + SecurityContextHolder.setContext(securityContext); + boolean isAuthenticated = SecurityUtils.isAuthenticated(); + assertThat(isAuthenticated).isFalse(); + } + + @Test + public void testIsCurrentUserInRole() { + SecurityContext securityContext = SecurityContextHolder.createEmptyContext(); + Collection authorities = new ArrayList<>(); + authorities.add(new SimpleGrantedAuthority(AuthoritiesConstants.USER)); + securityContext.setAuthentication(new UsernamePasswordAuthenticationToken("user", "user", authorities)); + SecurityContextHolder.setContext(securityContext); + + assertThat(SecurityUtils.isCurrentUserInRole(AuthoritiesConstants.USER)).isTrue(); + assertThat(SecurityUtils.isCurrentUserInRole(AuthoritiesConstants.ADMIN)).isFalse(); + } + +} diff --git a/src/test/java/it/cnr/isti/epasmed/service/AuditEventServiceIT.java b/src/test/java/it/cnr/isti/epasmed/service/AuditEventServiceIT.java new file mode 100644 index 0000000..b1b1d8d --- /dev/null +++ b/src/test/java/it/cnr/isti/epasmed/service/AuditEventServiceIT.java @@ -0,0 +1,73 @@ +package it.cnr.isti.epasmed.service; + +import it.cnr.isti.epasmed.domain.PersistentAuditEvent; +import it.cnr.isti.epasmed.repository.PersistenceAuditEventRepository; +import it.cnr.isti.epasmed.EpasmedApp; +import io.github.jhipster.config.JHipsterProperties; +import org.junit.jupiter.api.BeforeEach; +import org.junit.jupiter.api.Test; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.boot.test.context.SpringBootTest; +import org.springframework.transaction.annotation.Transactional; +import java.time.Instant; +import java.time.temporal.ChronoUnit; + +import static org.assertj.core.api.Assertions.assertThat; + +/** + * Integration tests for {@link AuditEventService}. + */ +@SpringBootTest(classes = EpasmedApp.class) +@Transactional(value="epasMedTransactionManager") +public class AuditEventServiceIT { + @Autowired + private AuditEventService auditEventService; + + @Autowired + private PersistenceAuditEventRepository persistenceAuditEventRepository; + + @Autowired + private JHipsterProperties jHipsterProperties; + + private PersistentAuditEvent auditEventOld; + + private PersistentAuditEvent auditEventWithinRetention; + + private PersistentAuditEvent auditEventNew; + + @BeforeEach + public void init() { + auditEventOld = new PersistentAuditEvent(); + auditEventOld.setAuditEventDate(Instant.now().minus(jHipsterProperties.getAuditEvents().getRetentionPeriod() + 1, ChronoUnit.DAYS)); + auditEventOld.setPrincipal("test-user-old"); + auditEventOld.setAuditEventType("test-type"); + + auditEventWithinRetention = new PersistentAuditEvent(); + auditEventWithinRetention.setAuditEventDate(Instant.now().minus(jHipsterProperties.getAuditEvents().getRetentionPeriod() - 1, ChronoUnit.DAYS)); + auditEventWithinRetention.setPrincipal("test-user-retention"); + auditEventWithinRetention.setAuditEventType("test-type"); + + auditEventNew = new PersistentAuditEvent(); + auditEventNew.setAuditEventDate(Instant.now()); + auditEventNew.setPrincipal("test-user-new"); + auditEventNew.setAuditEventType("test-type"); + } + + @Test + @Transactional(value="epasMedTransactionManager") + public void verifyOldAuditEventsAreDeleted() { + persistenceAuditEventRepository.deleteAll(); + persistenceAuditEventRepository.save(auditEventOld); + persistenceAuditEventRepository.save(auditEventWithinRetention); + persistenceAuditEventRepository.save(auditEventNew); + + persistenceAuditEventRepository.flush(); + auditEventService.removeOldAuditEvents(); + persistenceAuditEventRepository.flush(); + + assertThat(persistenceAuditEventRepository.findAll().size()).isEqualTo(2); + assertThat(persistenceAuditEventRepository.findByPrincipal("test-user-old")).isEmpty(); + assertThat(persistenceAuditEventRepository.findByPrincipal("test-user-retention")).isNotEmpty(); + assertThat(persistenceAuditEventRepository.findByPrincipal("test-user-new")).isNotEmpty(); + } +} diff --git a/src/test/java/it/cnr/isti/epasmed/service/MailServiceIT.java b/src/test/java/it/cnr/isti/epasmed/service/MailServiceIT.java new file mode 100644 index 0000000..cb96516 --- /dev/null +++ b/src/test/java/it/cnr/isti/epasmed/service/MailServiceIT.java @@ -0,0 +1,247 @@ +package it.cnr.isti.epasmed.service; + +import it.cnr.isti.epasmed.config.Constants; + +import it.cnr.isti.epasmed.EpasmedApp; +import it.cnr.isti.epasmed.domain.User; +import io.github.jhipster.config.JHipsterProperties; +import org.junit.jupiter.api.BeforeEach; +import org.junit.jupiter.api.Test; +import org.mockito.ArgumentCaptor; +import org.mockito.Captor; +import org.mockito.MockitoAnnotations; +import org.mockito.Spy; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.boot.test.context.SpringBootTest; +import org.springframework.context.MessageSource; +import org.springframework.mail.MailSendException; +import org.springframework.mail.javamail.JavaMailSenderImpl; +import org.springframework.test.context.junit.jupiter.EnabledIf; +import org.thymeleaf.spring5.SpringTemplateEngine; + +import javax.mail.Multipart; +import javax.mail.internet.MimeBodyPart; +import javax.mail.internet.MimeMessage; +import javax.mail.internet.MimeMultipart; +import java.io.ByteArrayOutputStream; +import java.io.File; +import java.io.FileInputStream; +import java.io.InputStreamReader; +import java.net.URI; +import java.net.URL; +import java.nio.charset.Charset; +import java.util.Properties; +import java.util.regex.Matcher; +import java.util.regex.Pattern; + +import static org.assertj.core.api.Assertions.*; +import static org.mockito.ArgumentMatchers.any; +import static org.mockito.Mockito.*; + +/** + * Integration tests for {@link MailService}. + */ +@SpringBootTest(classes = EpasmedApp.class) +@EnabledIf("false") +public class MailServiceIT { + + private static final String[] languages = { + // jhipster-needle-i18n-language-constant - JHipster will add/remove languages in this array + }; + private static final Pattern PATTERN_LOCALE_3 = Pattern.compile("([a-z]{2})-([a-zA-Z]{4})-([a-z]{2})"); + private static final Pattern PATTERN_LOCALE_2 = Pattern.compile("([a-z]{2})-([a-z]{2})"); + + @Autowired + private JHipsterProperties jHipsterProperties; + + @Autowired + private MessageSource messageSource; + + @Autowired + private SpringTemplateEngine templateEngine; + + @Spy + private JavaMailSenderImpl javaMailSender; + + @Captor + private ArgumentCaptor messageCaptor; + + private MailService mailService; + + @BeforeEach + public void setup() { + MockitoAnnotations.initMocks(this); + doNothing().when(javaMailSender).send(any(MimeMessage.class)); + mailService = new MailService(jHipsterProperties, javaMailSender, messageSource, templateEngine); + } + + @Test + public void testSendEmail() throws Exception { + mailService.sendEmail("john.doe@example.com", "testSubject", "testContent", false, false); + verify(javaMailSender).send(messageCaptor.capture()); + MimeMessage message = messageCaptor.getValue(); + assertThat(message.getSubject()).isEqualTo("testSubject"); + assertThat(message.getAllRecipients()[0].toString()).isEqualTo("john.doe@example.com"); + assertThat(message.getFrom()[0].toString()).isEqualTo(jHipsterProperties.getMail().getFrom()); + assertThat(message.getContent()).isInstanceOf(String.class); + assertThat(message.getContent().toString()).isEqualTo("testContent"); + assertThat(message.getDataHandler().getContentType()).isEqualTo("text/plain; charset=UTF-8"); + } + + @Test + public void testSendHtmlEmail() throws Exception { + mailService.sendEmail("john.doe@example.com", "testSubject", "testContent", false, true); + verify(javaMailSender).send(messageCaptor.capture()); + MimeMessage message = messageCaptor.getValue(); + assertThat(message.getSubject()).isEqualTo("testSubject"); + assertThat(message.getAllRecipients()[0].toString()).isEqualTo("john.doe@example.com"); + assertThat(message.getFrom()[0].toString()).isEqualTo(jHipsterProperties.getMail().getFrom()); + assertThat(message.getContent()).isInstanceOf(String.class); + assertThat(message.getContent().toString()).isEqualTo("testContent"); + assertThat(message.getDataHandler().getContentType()).isEqualTo("text/html;charset=UTF-8"); + } + + @Test + public void testSendMultipartEmail() throws Exception { + mailService.sendEmail("john.doe@example.com", "testSubject", "testContent", true, false); + verify(javaMailSender).send(messageCaptor.capture()); + MimeMessage message = messageCaptor.getValue(); + MimeMultipart mp = (MimeMultipart) message.getContent(); + MimeBodyPart part = (MimeBodyPart) ((MimeMultipart) mp.getBodyPart(0).getContent()).getBodyPart(0); + ByteArrayOutputStream aos = new ByteArrayOutputStream(); + part.writeTo(aos); + assertThat(message.getSubject()).isEqualTo("testSubject"); + assertThat(message.getAllRecipients()[0].toString()).isEqualTo("john.doe@example.com"); + assertThat(message.getFrom()[0].toString()).isEqualTo(jHipsterProperties.getMail().getFrom()); + assertThat(message.getContent()).isInstanceOf(Multipart.class); + assertThat(aos.toString()).isEqualTo("\r\ntestContent"); + assertThat(part.getDataHandler().getContentType()).isEqualTo("text/plain; charset=UTF-8"); + } + + @Test + public void testSendMultipartHtmlEmail() throws Exception { + mailService.sendEmail("john.doe@example.com", "testSubject", "testContent", true, true); + verify(javaMailSender).send(messageCaptor.capture()); + MimeMessage message = messageCaptor.getValue(); + MimeMultipart mp = (MimeMultipart) message.getContent(); + MimeBodyPart part = (MimeBodyPart) ((MimeMultipart) mp.getBodyPart(0).getContent()).getBodyPart(0); + ByteArrayOutputStream aos = new ByteArrayOutputStream(); + part.writeTo(aos); + assertThat(message.getSubject()).isEqualTo("testSubject"); + assertThat(message.getAllRecipients()[0].toString()).isEqualTo("john.doe@example.com"); + assertThat(message.getFrom()[0].toString()).isEqualTo(jHipsterProperties.getMail().getFrom()); + assertThat(message.getContent()).isInstanceOf(Multipart.class); + assertThat(aos.toString()).isEqualTo("\r\ntestContent"); + assertThat(part.getDataHandler().getContentType()).isEqualTo("text/html;charset=UTF-8"); + } + + @Test + public void testSendEmailFromTemplate() throws Exception { + User user = new User(); + user.setLogin("john"); + user.setEmail("john.doe@example.com"); + user.setLangKey("en"); + mailService.sendEmailFromTemplate(user, "mail/testEmail", "email.test.title"); + verify(javaMailSender).send(messageCaptor.capture()); + MimeMessage message = messageCaptor.getValue(); + assertThat(message.getSubject()).isEqualTo("test title"); + assertThat(message.getAllRecipients()[0].toString()).isEqualTo(user.getEmail()); + assertThat(message.getFrom()[0].toString()).isEqualTo(jHipsterProperties.getMail().getFrom()); + assertThat(message.getContent().toString()).isEqualToNormalizingNewlines("test title, http://127.0.0.1:8080, john\n"); + assertThat(message.getDataHandler().getContentType()).isEqualTo("text/html;charset=UTF-8"); + } + + @Test + public void testSendActivationEmail() throws Exception { + User user = new User(); + user.setLangKey(Constants.DEFAULT_LANGUAGE); + user.setLogin("john"); + user.setEmail("john.doe@example.com"); + mailService.sendActivationEmail(user); + verify(javaMailSender).send(messageCaptor.capture()); + MimeMessage message = messageCaptor.getValue(); + assertThat(message.getAllRecipients()[0].toString()).isEqualTo(user.getEmail()); + assertThat(message.getFrom()[0].toString()).isEqualTo(jHipsterProperties.getMail().getFrom()); + assertThat(message.getContent().toString()).isNotEmpty(); + assertThat(message.getDataHandler().getContentType()).isEqualTo("text/html;charset=UTF-8"); + } + + @Test + public void testCreationEmail() throws Exception { + User user = new User(); + user.setLangKey(Constants.DEFAULT_LANGUAGE); + user.setLogin("john"); + user.setEmail("john.doe@example.com"); + mailService.sendCreationEmail(user); + verify(javaMailSender).send(messageCaptor.capture()); + MimeMessage message = messageCaptor.getValue(); + assertThat(message.getAllRecipients()[0].toString()).isEqualTo(user.getEmail()); + assertThat(message.getFrom()[0].toString()).isEqualTo(jHipsterProperties.getMail().getFrom()); + assertThat(message.getContent().toString()).isNotEmpty(); + assertThat(message.getDataHandler().getContentType()).isEqualTo("text/html;charset=UTF-8"); + } + + @Test + public void testSendPasswordResetMail() throws Exception { + User user = new User(); + user.setLangKey(Constants.DEFAULT_LANGUAGE); + user.setLogin("john"); + user.setEmail("john.doe@example.com"); + mailService.sendPasswordResetMail(user); + verify(javaMailSender).send(messageCaptor.capture()); + MimeMessage message = messageCaptor.getValue(); + assertThat(message.getAllRecipients()[0].toString()).isEqualTo(user.getEmail()); + assertThat(message.getFrom()[0].toString()).isEqualTo(jHipsterProperties.getMail().getFrom()); + assertThat(message.getContent().toString()).isNotEmpty(); + assertThat(message.getDataHandler().getContentType()).isEqualTo("text/html;charset=UTF-8"); + } + + @Test + public void testSendEmailWithException() { + doThrow(MailSendException.class).when(javaMailSender).send(any(MimeMessage.class)); + try { + mailService.sendEmail("john.doe@example.com", "testSubject", "testContent", false, false); + } catch (Exception e) { + fail("Exception shouldn't have been thrown"); + } + } + + @Test + public void testSendLocalizedEmailForAllSupportedLanguages() throws Exception { + User user = new User(); + user.setLogin("john"); + user.setEmail("john.doe@example.com"); + for (String langKey : languages) { + user.setLangKey(langKey); + mailService.sendEmailFromTemplate(user, "mail/testEmail", "email.test.title"); + verify(javaMailSender, atLeastOnce()).send(messageCaptor.capture()); + MimeMessage message = messageCaptor.getValue(); + + String propertyFilePath = "i18n/messages_" + getJavaLocale(langKey) + ".properties"; + URL resource = this.getClass().getClassLoader().getResource(propertyFilePath); + File file = new File(new URI(resource.getFile()).getPath()); + Properties properties = new Properties(); + properties.load(new InputStreamReader(new FileInputStream(file), Charset.forName("UTF-8"))); + + String emailTitle = (String) properties.get("email.test.title"); + assertThat(message.getSubject()).isEqualTo(emailTitle); + assertThat(message.getContent().toString()).isEqualToNormalizingNewlines("" + emailTitle + ", http://127.0.0.1:8080, john\n"); + } + } + + /** + * Convert a lang key to the Java locale. + */ + private String getJavaLocale(String langKey) { + String javaLangKey = langKey; + Matcher matcher2 = PATTERN_LOCALE_2.matcher(langKey); + if (matcher2.matches()) { + javaLangKey = matcher2.group(1) + "_"+ matcher2.group(2).toUpperCase(); + } + Matcher matcher3 = PATTERN_LOCALE_3.matcher(langKey); + if (matcher3.matches()) { + javaLangKey = matcher3.group(1) + "_" + matcher3.group(2) + "_" + matcher3.group(3).toUpperCase(); + } + return javaLangKey; + } +} diff --git a/src/test/java/it/cnr/isti/epasmed/service/UserServiceIT.java b/src/test/java/it/cnr/isti/epasmed/service/UserServiceIT.java new file mode 100644 index 0000000..d04e1ab --- /dev/null +++ b/src/test/java/it/cnr/isti/epasmed/service/UserServiceIT.java @@ -0,0 +1,235 @@ +package it.cnr.isti.epasmed.service; + +import it.cnr.isti.epasmed.EpasmedApp; +import it.cnr.isti.epasmed.config.Constants; +import it.cnr.isti.epasmed.domain.PersistentToken; +import it.cnr.isti.epasmed.domain.User; +import it.cnr.isti.epasmed.repository.PersistentTokenRepository; +import it.cnr.isti.epasmed.repository.UserRepository; +import it.cnr.isti.epasmed.service.dto.UserDTO; + +import io.github.jhipster.security.RandomUtil; + +import org.apache.commons.lang3.RandomStringUtils; +import org.junit.jupiter.api.BeforeEach; +import org.junit.jupiter.api.Test; +import org.mockito.Mock; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.boot.test.context.SpringBootTest; +import org.springframework.data.auditing.AuditingHandler; +import org.springframework.data.auditing.DateTimeProvider; +import org.springframework.data.domain.Page; +import org.springframework.data.domain.PageRequest; +import org.springframework.transaction.annotation.Transactional; + +import java.time.Instant; +import java.time.temporal.ChronoUnit; +import java.time.LocalDate; +import java.time.LocalDateTime; +import java.util.List; +import java.util.Optional; + +import static org.assertj.core.api.Assertions.assertThat; +import static org.mockito.Mockito.when; + +/** + * Integration tests for {@link UserService}. + */ +@SpringBootTest(classes = EpasmedApp.class) +@Transactional(value="epasMedTransactionManager") +public class UserServiceIT { + + private static final String DEFAULT_LOGIN = "johndoe"; + + private static final String DEFAULT_EMAIL = "johndoe@localhost"; + + private static final String DEFAULT_FIRSTNAME = "john"; + + private static final String DEFAULT_LASTNAME = "doe"; + + private static final String DEFAULT_IMAGEURL = "http://placehold.it/50x50"; + + private static final String DEFAULT_LANGKEY = "dummy"; + + @Autowired + private PersistentTokenRepository persistentTokenRepository; + + @Autowired + private UserRepository userRepository; + + @Autowired + private UserService userService; + + @Autowired + private AuditingHandler auditingHandler; + + @Mock + private DateTimeProvider dateTimeProvider; + + private User user; + + @BeforeEach + public void init() { + persistentTokenRepository.deleteAll(); + user = new User(); + user.setLogin(DEFAULT_LOGIN); + user.setPassword(RandomStringUtils.random(60)); + user.setActivated(true); + user.setEmail(DEFAULT_EMAIL); + user.setFirstName(DEFAULT_FIRSTNAME); + user.setLastName(DEFAULT_LASTNAME); + user.setImageUrl(DEFAULT_IMAGEURL); + user.setLangKey(DEFAULT_LANGKEY); + + when(dateTimeProvider.getNow()).thenReturn(Optional.of(LocalDateTime.now())); + auditingHandler.setDateTimeProvider(dateTimeProvider); + } + + @Test + @Transactional(value="epasMedTransactionManager") + public void testRemoveOldPersistentTokens() { + userRepository.saveAndFlush(user); + int existingCount = persistentTokenRepository.findByUser(user).size(); + LocalDate today = LocalDate.now(); + generateUserToken(user, "1111-1111", today); + generateUserToken(user, "2222-2222", today.minusDays(32)); + assertThat(persistentTokenRepository.findByUser(user)).hasSize(existingCount + 2); + userService.removeOldPersistentTokens(); + assertThat(persistentTokenRepository.findByUser(user)).hasSize(existingCount + 1); + } + + @Test + @Transactional(value="epasMedTransactionManager") + public void assertThatUserMustExistToResetPassword() { + userRepository.saveAndFlush(user); + Optional maybeUser = userService.requestPasswordReset("invalid.login@localhost"); + assertThat(maybeUser).isNotPresent(); + + maybeUser = userService.requestPasswordReset(user.getEmail()); + assertThat(maybeUser).isPresent(); + assertThat(maybeUser.orElse(null).getEmail()).isEqualTo(user.getEmail()); + assertThat(maybeUser.orElse(null).getResetDate()).isNotNull(); + assertThat(maybeUser.orElse(null).getResetKey()).isNotNull(); + } + + @Test + @Transactional(value="epasMedTransactionManager") + public void assertThatOnlyActivatedUserCanRequestPasswordReset() { + user.setActivated(false); + userRepository.saveAndFlush(user); + + Optional maybeUser = userService.requestPasswordReset(user.getLogin()); + assertThat(maybeUser).isNotPresent(); + userRepository.delete(user); + } + + @Test + @Transactional(value="epasMedTransactionManager") + public void assertThatResetKeyMustNotBeOlderThan24Hours() { + Instant daysAgo = Instant.now().minus(25, ChronoUnit.HOURS); + String resetKey = RandomUtil.generateResetKey(); + user.setActivated(true); + user.setResetDate(daysAgo); + user.setResetKey(resetKey); + userRepository.saveAndFlush(user); + + Optional maybeUser = userService.completePasswordReset("johndoe2", user.getResetKey()); + assertThat(maybeUser).isNotPresent(); + userRepository.delete(user); + } + + @Test + @Transactional(value="epasMedTransactionManager") + public void assertThatResetKeyMustBeValid() { + Instant daysAgo = Instant.now().minus(25, ChronoUnit.HOURS); + user.setActivated(true); + user.setResetDate(daysAgo); + user.setResetKey("1234"); + userRepository.saveAndFlush(user); + + Optional maybeUser = userService.completePasswordReset("johndoe2", user.getResetKey()); + assertThat(maybeUser).isNotPresent(); + userRepository.delete(user); + } + + @Test + @Transactional(value="epasMedTransactionManager") + public void assertThatUserCanResetPassword() { + String oldPassword = user.getPassword(); + Instant daysAgo = Instant.now().minus(2, ChronoUnit.HOURS); + String resetKey = RandomUtil.generateResetKey(); + user.setActivated(true); + user.setResetDate(daysAgo); + user.setResetKey(resetKey); + userRepository.saveAndFlush(user); + + Optional maybeUser = userService.completePasswordReset("johndoe2", user.getResetKey()); + assertThat(maybeUser).isPresent(); + assertThat(maybeUser.orElse(null).getResetDate()).isNull(); + assertThat(maybeUser.orElse(null).getResetKey()).isNull(); + assertThat(maybeUser.orElse(null).getPassword()).isNotEqualTo(oldPassword); + + userRepository.delete(user); + } + + @Test + @Transactional(value="epasMedTransactionManager") + public void assertThatNotActivatedUsersWithNotNullActivationKeyCreatedBefore3DaysAreDeleted() { + Instant now = Instant.now(); + when(dateTimeProvider.getNow()).thenReturn(Optional.of(now.minus(4, ChronoUnit.DAYS))); + user.setActivated(false); + user.setActivationKey(RandomStringUtils.random(20)); + User dbUser = userRepository.saveAndFlush(user); + dbUser.setCreatedDate(now.minus(4, ChronoUnit.DAYS)); + userRepository.saveAndFlush(user); + Instant threeDaysAgo = now.minus(3, ChronoUnit.DAYS); + List users = userRepository.findAllByActivatedIsFalseAndActivationKeyIsNotNullAndCreatedDateBefore(threeDaysAgo); + assertThat(users).isNotEmpty(); + userService.removeNotActivatedUsers(); + users = userRepository.findAllByActivatedIsFalseAndActivationKeyIsNotNullAndCreatedDateBefore(threeDaysAgo); + assertThat(users).isEmpty(); + } + + @Test + @Transactional(value="epasMedTransactionManager") + public void assertThatNotActivatedUsersWithNullActivationKeyCreatedBefore3DaysAreNotDeleted() { + Instant now = Instant.now(); + when(dateTimeProvider.getNow()).thenReturn(Optional.of(now.minus(4, ChronoUnit.DAYS))); + user.setActivated(false); + User dbUser = userRepository.saveAndFlush(user); + dbUser.setCreatedDate(now.minus(4, ChronoUnit.DAYS)); + userRepository.saveAndFlush(user); + Instant threeDaysAgo = now.minus(3, ChronoUnit.DAYS); + List users = userRepository.findAllByActivatedIsFalseAndActivationKeyIsNotNullAndCreatedDateBefore(threeDaysAgo); + assertThat(users).isEmpty(); + userService.removeNotActivatedUsers(); + Optional maybeDbUser = userRepository.findById(dbUser.getId()); + assertThat(maybeDbUser).contains(dbUser); + } + + private void generateUserToken(User user, String tokenSeries, LocalDate localDate) { + PersistentToken token = new PersistentToken(); + token.setSeries(tokenSeries); + token.setUser(user); + token.setTokenValue(tokenSeries + "-data"); + token.setTokenDate(localDate); + token.setIpAddress("127.0.0.1"); + token.setUserAgent("Test agent"); + persistentTokenRepository.saveAndFlush(token); + } + + @Test + @Transactional(value="epasMedTransactionManager") + public void assertThatAnonymousUserIsNotGet() { + user.setLogin(Constants.ANONYMOUS_USER); + if (!userRepository.findOneByLogin(Constants.ANONYMOUS_USER).isPresent()) { + userRepository.saveAndFlush(user); + } + final PageRequest pageable = PageRequest.of(0, (int) userRepository.count()); + final Page allManagedUsers = userService.getAllManagedUsers(pageable); + assertThat(allManagedUsers.getContent().stream() + .noneMatch(user -> Constants.ANONYMOUS_USER.equals(user.getLogin()))) + .isTrue(); + } + +} diff --git a/src/test/java/it/cnr/isti/epasmed/service/mapper/UserMapperTest.java b/src/test/java/it/cnr/isti/epasmed/service/mapper/UserMapperTest.java new file mode 100644 index 0000000..3de492e --- /dev/null +++ b/src/test/java/it/cnr/isti/epasmed/service/mapper/UserMapperTest.java @@ -0,0 +1,136 @@ +package it.cnr.isti.epasmed.service.mapper; + +import it.cnr.isti.epasmed.domain.User; +import it.cnr.isti.epasmed.service.dto.UserDTO; +import org.apache.commons.lang3.RandomStringUtils; +import org.junit.jupiter.api.BeforeEach; +import org.junit.jupiter.api.Test; + +import java.util.ArrayList; +import java.util.HashSet; +import java.util.List; +import java.util.Set; + +import static org.assertj.core.api.Assertions.assertThat; + +/** + * Unit tests for {@link UserMapper}. + */ +public class UserMapperTest { + + private static final String DEFAULT_LOGIN = "johndoe"; + private static final Long DEFAULT_ID = 1L; + + private UserMapper userMapper; + private User user; + private UserDTO userDto; + + @BeforeEach + public void init() { + userMapper = new UserMapper(); + user = new User(); + user.setLogin(DEFAULT_LOGIN); + user.setPassword(RandomStringUtils.random(60)); + user.setActivated(true); + user.setEmail("johndoe@localhost"); + user.setFirstName("john"); + user.setLastName("doe"); + user.setImageUrl("image_url"); + user.setLangKey("en"); + + userDto = new UserDTO(user); + } + + @Test + public void usersToUserDTOsShouldMapOnlyNonNullUsers() { + List users = new ArrayList<>(); + users.add(user); + users.add(null); + + List userDTOS = userMapper.usersToUserDTOs(users); + + assertThat(userDTOS).isNotEmpty(); + assertThat(userDTOS).size().isEqualTo(1); + } + + @Test + public void userDTOsToUsersShouldMapOnlyNonNullUsers() { + List usersDto = new ArrayList<>(); + usersDto.add(userDto); + usersDto.add(null); + + List users = userMapper.userDTOsToUsers(usersDto); + + assertThat(users).isNotEmpty(); + assertThat(users).size().isEqualTo(1); + } + + @Test + public void userDTOsToUsersWithAuthoritiesStringShouldMapToUsersWithAuthoritiesDomain() { + Set authoritiesAsString = new HashSet<>(); + authoritiesAsString.add("ADMIN"); + userDto.setAuthorities(authoritiesAsString); + + List usersDto = new ArrayList<>(); + usersDto.add(userDto); + + List users = userMapper.userDTOsToUsers(usersDto); + + assertThat(users).isNotEmpty(); + assertThat(users).size().isEqualTo(1); + assertThat(users.get(0).getAuthorities()).isNotNull(); + assertThat(users.get(0).getAuthorities()).isNotEmpty(); + assertThat(users.get(0).getAuthorities().iterator().next().getName()).isEqualTo("ADMIN"); + } + + @Test + public void userDTOsToUsersMapWithNullAuthoritiesStringShouldReturnUserWithEmptyAuthorities() { + userDto.setAuthorities(null); + + List usersDto = new ArrayList<>(); + usersDto.add(userDto); + + List users = userMapper.userDTOsToUsers(usersDto); + + assertThat(users).isNotEmpty(); + assertThat(users).size().isEqualTo(1); + assertThat(users.get(0).getAuthorities()).isNotNull(); + assertThat(users.get(0).getAuthorities()).isEmpty(); + } + + @Test + public void userDTOToUserMapWithAuthoritiesStringShouldReturnUserWithAuthorities() { + Set authoritiesAsString = new HashSet<>(); + authoritiesAsString.add("ADMIN"); + userDto.setAuthorities(authoritiesAsString); + + User user = userMapper.userDTOToUser(userDto); + + assertThat(user).isNotNull(); + assertThat(user.getAuthorities()).isNotNull(); + assertThat(user.getAuthorities()).isNotEmpty(); + assertThat(user.getAuthorities().iterator().next().getName()).isEqualTo("ADMIN"); + } + + @Test + public void userDTOToUserMapWithNullAuthoritiesStringShouldReturnUserWithEmptyAuthorities() { + userDto.setAuthorities(null); + + User user = userMapper.userDTOToUser(userDto); + + assertThat(user).isNotNull(); + assertThat(user.getAuthorities()).isNotNull(); + assertThat(user.getAuthorities()).isEmpty(); + } + + @Test + public void userDTOToUserMapWithNullUserShouldReturnNull() { + assertThat(userMapper.userDTOToUser(null)).isNull(); + } + + @Test + public void testUserFromId() { + assertThat(userMapper.userFromId(DEFAULT_ID).getId()).isEqualTo(DEFAULT_ID); + assertThat(userMapper.userFromId(null)).isNull(); + } +} diff --git a/src/test/java/it/cnr/isti/epasmed/web/rest/AccountResourceIT.java b/src/test/java/it/cnr/isti/epasmed/web/rest/AccountResourceIT.java new file mode 100644 index 0000000..ba04949 --- /dev/null +++ b/src/test/java/it/cnr/isti/epasmed/web/rest/AccountResourceIT.java @@ -0,0 +1,570 @@ +package it.cnr.isti.epasmed.web.rest; + +import it.cnr.isti.epasmed.EpasmedApp; +import it.cnr.isti.epasmed.config.Constants; +import it.cnr.isti.epasmed.domain.PersistentToken; +import it.cnr.isti.epasmed.domain.User; +import it.cnr.isti.epasmed.repository.AuthorityRepository; +import it.cnr.isti.epasmed.repository.PersistentTokenRepository; +import it.cnr.isti.epasmed.repository.UserRepository; +import it.cnr.isti.epasmed.security.AuthoritiesConstants; +import it.cnr.isti.epasmed.service.UserService; +import it.cnr.isti.epasmed.service.dto.PasswordChangeDTO; +import it.cnr.isti.epasmed.service.dto.UserDTO; +import it.cnr.isti.epasmed.web.rest.vm.KeyAndPasswordVM; +import it.cnr.isti.epasmed.web.rest.vm.ManagedUserVM; +import org.apache.commons.lang3.RandomStringUtils; +import org.junit.jupiter.api.BeforeEach; +import org.junit.jupiter.api.Test; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.boot.test.autoconfigure.web.servlet.AutoConfigureMockMvc; +import org.springframework.boot.test.context.SpringBootTest; +import org.springframework.http.MediaType; +import org.springframework.http.converter.HttpMessageConverter; +import org.springframework.security.crypto.password.PasswordEncoder; +import org.springframework.security.test.context.support.WithMockUser; +import org.springframework.test.context.junit.jupiter.EnabledIf; +import org.springframework.test.web.servlet.MockMvc; +import org.springframework.transaction.annotation.Transactional; + +import java.time.Instant; +import java.time.LocalDate; +import java.util.*; + +import static org.assertj.core.api.Assertions.assertThat; +import static org.hamcrest.Matchers.hasItem; +import static it.cnr.isti.epasmed.web.rest.AccountResourceIT.TEST_USER_LOGIN; +import static org.springframework.security.test.web.servlet.request.SecurityMockMvcRequestPostProcessors.csrf; +import static org.springframework.test.web.servlet.request.MockMvcRequestBuilders.*; +import static org.springframework.test.web.servlet.result.MockMvcResultMatchers.*; + +/** + * Integration tests for the {@link AccountResource} REST controller. + */ +@AutoConfigureMockMvc +@WithMockUser(value = TEST_USER_LOGIN) +@SpringBootTest(classes = EpasmedApp.class) +@EnabledIf("false") +public class AccountResourceIT { + static final String TEST_USER_LOGIN = "test"; + + @Autowired + private UserRepository userRepository; + + @Autowired + private AuthorityRepository authorityRepository; + + @Autowired + private UserService userService; + + @Autowired + private PersistentTokenRepository persistentTokenRepository; + + @Autowired + private PasswordEncoder passwordEncoder; + + @Autowired + private MockMvc restAccountMockMvc; + + @Test + @WithUnauthenticatedMockUser + public void testNonAuthenticatedUser() throws Exception { + restAccountMockMvc.perform(get("/api/authenticate") + .accept(MediaType.APPLICATION_JSON)) + .andExpect(status().isOk()) + .andExpect(content().string("")); + } + + @Test + public void testAuthenticatedUser() throws Exception { + restAccountMockMvc.perform(get("/api/authenticate") + .with(request -> { + request.setRemoteUser(TEST_USER_LOGIN); + return request; + }) + .accept(MediaType.APPLICATION_JSON)) + .andExpect(status().isOk()) + .andExpect(content().string(TEST_USER_LOGIN)); + } + + + + + + + + + @Test + @Transactional(value="epasMedTransactionManager") + public void testRegisterInvalidEmail() throws Exception { + ManagedUserVM invalidUser = new ManagedUserVM(); + invalidUser.setLogin("bob"); + invalidUser.setPassword("password"); + invalidUser.setFirstName("Bob"); + invalidUser.setLastName("Green"); + invalidUser.setEmail("invalid");// <-- invalid + invalidUser.setActivated(true); + invalidUser.setImageUrl("http://placehold.it/50x50"); + invalidUser.setLangKey(Constants.DEFAULT_LANGUAGE); + invalidUser.setAuthorities(Collections.singleton(AuthoritiesConstants.USER)); + + restAccountMockMvc.perform( + post("/api/register") + .contentType(MediaType.APPLICATION_JSON) + .content(TestUtil.convertObjectToJsonBytes(invalidUser)) + .with(csrf())) + .andExpect(status().is4xxClientError()); + + Optional user = userRepository.findOneByLogin("bob"); + assertThat(user.isPresent()).isFalse(); + } + + + + + + + + + @Test + @Transactional(value="epasMedTransactionManager") + public void testActivateAccount() throws Exception { + final String activationKey = "some activation key"; + User user = new User(); + user.setLogin("activate-account"); + user.setEmail("activate-account@example.com"); + user.setPassword(RandomStringUtils.random(60)); + user.setActivated(false); + user.setActivationKey(activationKey); + + userRepository.saveAndFlush(user); + + restAccountMockMvc.perform(get("/api/activate?key={activationKey}", activationKey)) + .andExpect(status().isOk()); + + user = userRepository.findOneByLogin(user.getLogin()).orElse(null); + assertThat(user.getActivated()).isTrue(); + } + + @Test + @Transactional(value="epasMedTransactionManager") + public void testActivateAccountWithWrongKey() throws Exception { + restAccountMockMvc.perform(get("/api/activate?key=wrongActivationKey")) + .andExpect(status().isInternalServerError()); + } + + @Test + @Transactional(value="epasMedTransactionManager") + @WithMockUser("save-account") + public void testSaveAccount() throws Exception { + User user = new User(); + user.setLogin("save-account"); + user.setEmail("save-account@example.com"); + user.setPassword(RandomStringUtils.random(60)); + user.setActivated(true); + userRepository.saveAndFlush(user); + + UserDTO userDTO = new UserDTO(); + userDTO.setLogin("not-used"); + userDTO.setFirstName("firstname"); + userDTO.setLastName("lastname"); + userDTO.setEmail("save-account@example.com"); + userDTO.setActivated(false); + userDTO.setImageUrl("http://placehold.it/50x50"); + userDTO.setLangKey(Constants.DEFAULT_LANGUAGE); + userDTO.setAuthorities(Collections.singleton(AuthoritiesConstants.ADMIN)); + + restAccountMockMvc.perform( + post("/api/account") + .contentType(MediaType.APPLICATION_JSON) + .content(TestUtil.convertObjectToJsonBytes(userDTO)) + .with(csrf())) + .andExpect(status().isOk()); + + User updatedUser = userRepository.findOneWithAuthoritiesByLogin(user.getLogin()).orElse(null); + assertThat(updatedUser.getFirstName()).isEqualTo(userDTO.getFirstName()); + assertThat(updatedUser.getLastName()).isEqualTo(userDTO.getLastName()); + assertThat(updatedUser.getEmail()).isEqualTo(userDTO.getEmail()); + assertThat(updatedUser.getLangKey()).isEqualTo(userDTO.getLangKey()); + assertThat(updatedUser.getPassword()).isEqualTo(user.getPassword()); + assertThat(updatedUser.getImageUrl()).isEqualTo(userDTO.getImageUrl()); + assertThat(updatedUser.getActivated()).isEqualTo(true); + assertThat(updatedUser.getAuthorities()).isEmpty(); + } + + @Test + @Transactional(value="epasMedTransactionManager") + @WithMockUser("save-invalid-email") + public void testSaveInvalidEmail() throws Exception { + User user = new User(); + user.setLogin("save-invalid-email"); + user.setEmail("save-invalid-email@example.com"); + user.setPassword(RandomStringUtils.random(60)); + user.setActivated(true); + + userRepository.saveAndFlush(user); + + UserDTO userDTO = new UserDTO(); + userDTO.setLogin("not-used"); + userDTO.setFirstName("firstname"); + userDTO.setLastName("lastname"); + userDTO.setEmail("invalid email"); + userDTO.setActivated(false); + userDTO.setImageUrl("http://placehold.it/50x50"); + userDTO.setLangKey(Constants.DEFAULT_LANGUAGE); + userDTO.setAuthorities(Collections.singleton(AuthoritiesConstants.ADMIN)); + + restAccountMockMvc.perform( + post("/api/account") + .contentType(MediaType.APPLICATION_JSON) + .content(TestUtil.convertObjectToJsonBytes(userDTO)) + .with(csrf())) + .andExpect(status().isBadRequest()); + + assertThat(userRepository.findOneByEmailIgnoreCase("invalid email")).isNotPresent(); + } + + @Test + @Transactional(value="epasMedTransactionManager") + @WithMockUser("save-existing-email") + public void testSaveExistingEmail() throws Exception { + User user = new User(); + user.setLogin("save-existing-email"); + user.setEmail("save-existing-email@example.com"); + user.setPassword(RandomStringUtils.random(60)); + user.setActivated(true); + userRepository.saveAndFlush(user); + + User anotherUser = new User(); + anotherUser.setLogin("save-existing-email2"); + anotherUser.setEmail("save-existing-email2@example.com"); + anotherUser.setPassword(RandomStringUtils.random(60)); + anotherUser.setActivated(true); + + userRepository.saveAndFlush(anotherUser); + + UserDTO userDTO = new UserDTO(); + userDTO.setLogin("not-used"); + userDTO.setFirstName("firstname"); + userDTO.setLastName("lastname"); + userDTO.setEmail("save-existing-email2@example.com"); + userDTO.setActivated(false); + userDTO.setImageUrl("http://placehold.it/50x50"); + userDTO.setLangKey(Constants.DEFAULT_LANGUAGE); + userDTO.setAuthorities(Collections.singleton(AuthoritiesConstants.ADMIN)); + + restAccountMockMvc.perform( + post("/api/account") + .contentType(MediaType.APPLICATION_JSON) + .content(TestUtil.convertObjectToJsonBytes(userDTO)) + .with(csrf())) + .andExpect(status().isBadRequest()); + + User updatedUser = userRepository.findOneByLogin("save-existing-email").orElse(null); + assertThat(updatedUser.getEmail()).isEqualTo("save-existing-email@example.com"); + } + + @Test + @Transactional(value="epasMedTransactionManager") + @WithMockUser("save-existing-email-and-login") + public void testSaveExistingEmailAndLogin() throws Exception { + User user = new User(); + user.setLogin("save-existing-email-and-login"); + user.setEmail("save-existing-email-and-login@example.com"); + user.setPassword(RandomStringUtils.random(60)); + user.setActivated(true); + userRepository.saveAndFlush(user); + + UserDTO userDTO = new UserDTO(); + userDTO.setLogin("not-used"); + userDTO.setFirstName("firstname"); + userDTO.setLastName("lastname"); + userDTO.setEmail("save-existing-email-and-login@example.com"); + userDTO.setActivated(false); + userDTO.setImageUrl("http://placehold.it/50x50"); + userDTO.setLangKey(Constants.DEFAULT_LANGUAGE); + userDTO.setAuthorities(Collections.singleton(AuthoritiesConstants.ADMIN)); + + restAccountMockMvc.perform( + post("/api/account") + .contentType(MediaType.APPLICATION_JSON) + .content(TestUtil.convertObjectToJsonBytes(userDTO)) + .with(csrf())) + .andExpect(status().isOk()); + + User updatedUser = userRepository.findOneByLogin("save-existing-email-and-login").orElse(null); + assertThat(updatedUser.getEmail()).isEqualTo("save-existing-email-and-login@example.com"); + } + + @Test + @Transactional(value="epasMedTransactionManager") + @WithMockUser("change-password-wrong-existing-password") + public void testChangePasswordWrongExistingPassword() throws Exception { + User user = new User(); + String currentPassword = RandomStringUtils.random(60); + user.setPassword(passwordEncoder.encode(currentPassword)); + user.setLogin("change-password-wrong-existing-password"); + user.setEmail("change-password-wrong-existing-password@example.com"); + userRepository.saveAndFlush(user); + + restAccountMockMvc.perform(post("/api/account/change-password") + .contentType(MediaType.APPLICATION_JSON) + .content(TestUtil.convertObjectToJsonBytes(new PasswordChangeDTO("1"+currentPassword, "new password"))) + .with(csrf())) + .andExpect(status().isBadRequest()); + + User updatedUser = userRepository.findOneByLogin("change-password-wrong-existing-password").orElse(null); + assertThat(passwordEncoder.matches("new password", updatedUser.getPassword())).isFalse(); + assertThat(passwordEncoder.matches(currentPassword, updatedUser.getPassword())).isTrue(); + } + + @Test + @Transactional(value="epasMedTransactionManager") + @WithMockUser("change-password") + public void testChangePassword() throws Exception { + User user = new User(); + String currentPassword = RandomStringUtils.random(60); + user.setPassword(passwordEncoder.encode(currentPassword)); + user.setLogin("change-password"); + user.setEmail("change-password@example.com"); + userRepository.saveAndFlush(user); + + restAccountMockMvc.perform(post("/api/account/change-password") + .contentType(MediaType.APPLICATION_JSON) + .content(TestUtil.convertObjectToJsonBytes(new PasswordChangeDTO(currentPassword, "new password"))) + .with(csrf())) + .andExpect(status().isOk()); + + User updatedUser = userRepository.findOneByLogin("change-password").orElse(null); + assertThat(passwordEncoder.matches("new password", updatedUser.getPassword())).isTrue(); + } + + @Test + @Transactional(value="epasMedTransactionManager") + @WithMockUser("change-password-too-small") + public void testChangePasswordTooSmall() throws Exception { + User user = new User(); + String currentPassword = RandomStringUtils.random(60); + user.setPassword(passwordEncoder.encode(currentPassword)); + user.setLogin("change-password-too-small"); + user.setEmail("change-password-too-small@example.com"); + userRepository.saveAndFlush(user); + + String newPassword = RandomStringUtils.random(ManagedUserVM.PASSWORD_MIN_LENGTH - 1); + + restAccountMockMvc.perform(post("/api/account/change-password") + .contentType(MediaType.APPLICATION_JSON) + .content(TestUtil.convertObjectToJsonBytes(new PasswordChangeDTO(currentPassword, newPassword))) + .with(csrf())) + .andExpect(status().isBadRequest()); + + User updatedUser = userRepository.findOneByLogin("change-password-too-small").orElse(null); + assertThat(updatedUser.getPassword()).isEqualTo(user.getPassword()); + } + + @Test + @Transactional(value="epasMedTransactionManager") + @WithMockUser("change-password-too-long") + public void testChangePasswordTooLong() throws Exception { + User user = new User(); + String currentPassword = RandomStringUtils.random(60); + user.setPassword(passwordEncoder.encode(currentPassword)); + user.setLogin("change-password-too-long"); + user.setEmail("change-password-too-long@example.com"); + userRepository.saveAndFlush(user); + + String newPassword = RandomStringUtils.random(ManagedUserVM.PASSWORD_MAX_LENGTH + 1); + + restAccountMockMvc.perform(post("/api/account/change-password") + .contentType(MediaType.APPLICATION_JSON) + .content(TestUtil.convertObjectToJsonBytes(new PasswordChangeDTO(currentPassword, newPassword))) + .with(csrf())) + .andExpect(status().isBadRequest()); + + User updatedUser = userRepository.findOneByLogin("change-password-too-long").orElse(null); + assertThat(updatedUser.getPassword()).isEqualTo(user.getPassword()); + } + + @Test + @Transactional(value="epasMedTransactionManager") + @WithMockUser("change-password-empty") + public void testChangePasswordEmpty() throws Exception { + User user = new User(); + String currentPassword = RandomStringUtils.random(60); + user.setPassword(passwordEncoder.encode(currentPassword)); + user.setLogin("change-password-empty"); + user.setEmail("change-password-empty@example.com"); + userRepository.saveAndFlush(user); + + restAccountMockMvc.perform(post("/api/account/change-password") + .contentType(MediaType.APPLICATION_JSON) + .content(TestUtil.convertObjectToJsonBytes(new PasswordChangeDTO(currentPassword, ""))) + .with(csrf())) + .andExpect(status().isBadRequest()); + + User updatedUser = userRepository.findOneByLogin("change-password-empty").orElse(null); + assertThat(updatedUser.getPassword()).isEqualTo(user.getPassword()); + } + + @Test + @Transactional(value="epasMedTransactionManager") + @WithMockUser("current-sessions") + public void testGetCurrentSessions() throws Exception { + User user = new User(); + user.setPassword(RandomStringUtils.random(60)); + user.setLogin("current-sessions"); + user.setEmail("current-sessions@example.com"); + userRepository.saveAndFlush(user); + + PersistentToken token = new PersistentToken(); + token.setSeries("current-sessions"); + token.setUser(user); + token.setTokenValue("current-session-data"); + token.setTokenDate(LocalDate.of(2017, 3, 23)); + token.setIpAddress("127.0.0.1"); + token.setUserAgent("Test agent"); + persistentTokenRepository.saveAndFlush(token); + + restAccountMockMvc.perform(get("/api/account/sessions")) + .andExpect(status().isOk()) + .andExpect(jsonPath("$.[*].series").value(hasItem(token.getSeries()))) + .andExpect(jsonPath("$.[*].ipAddress").value(hasItem(token.getIpAddress()))) + .andExpect(jsonPath("$.[*].userAgent").value(hasItem(token.getUserAgent()))) + .andExpect(jsonPath("$.[*].tokenDate").value(hasItem(token.getTokenDate().toString()))); + } + + @Test + @Transactional(value="epasMedTransactionManager") + @WithMockUser("invalidate-session") + public void testInvalidateSession() throws Exception { + User user = new User(); + user.setPassword(RandomStringUtils.random(60)); + user.setLogin("invalidate-session"); + user.setEmail("invalidate-session@example.com"); + userRepository.saveAndFlush(user); + + PersistentToken token = new PersistentToken(); + token.setSeries("invalidate-session"); + token.setUser(user); + token.setTokenValue("invalidate-data"); + token.setTokenDate(LocalDate.of(2017, 3, 23)); + token.setIpAddress("127.0.0.1"); + token.setUserAgent("Test agent"); + persistentTokenRepository.saveAndFlush(token); + + assertThat(persistentTokenRepository.findByUser(user)).hasSize(1); + + restAccountMockMvc.perform(delete("/api/account/sessions/invalidate-session").with(csrf())) + .andExpect(status().isOk()); + + assertThat(persistentTokenRepository.findByUser(user)).isEmpty(); + } + + @Test + @Transactional(value="epasMedTransactionManager") + public void testRequestPasswordReset() throws Exception { + User user = new User(); + user.setPassword(RandomStringUtils.random(60)); + user.setActivated(true); + user.setLogin("password-reset"); + user.setEmail("password-reset@example.com"); + userRepository.saveAndFlush(user); + + restAccountMockMvc.perform(post("/api/account/reset-password/init") + .content("password-reset@example.com") + .with(csrf())) + .andExpect(status().isOk()); + } + + @Test + @Transactional(value="epasMedTransactionManager") + public void testRequestPasswordResetUpperCaseEmail() throws Exception { + User user = new User(); + user.setPassword(RandomStringUtils.random(60)); + user.setActivated(true); + user.setLogin("password-reset-upper-case"); + user.setEmail("password-reset-upper-case@example.com"); + userRepository.saveAndFlush(user); + + restAccountMockMvc.perform(post("/api/account/reset-password/init") + .content("password-reset-upper-case@EXAMPLE.COM") + .with(csrf())) + .andExpect(status().isOk()); + } + + @Test + public void testRequestPasswordResetWrongEmail() throws Exception { + restAccountMockMvc.perform( + post("/api/account/reset-password/init") + .content("password-reset-wrong-email@example.com") + .with(csrf())) + .andExpect(status().isOk()); + } + + @Test + @Transactional(value="epasMedTransactionManager") + public void testFinishPasswordReset() throws Exception { + User user = new User(); + user.setPassword(RandomStringUtils.random(60)); + user.setLogin("finish-password-reset"); + user.setEmail("finish-password-reset@example.com"); + user.setResetDate(Instant.now().plusSeconds(60)); + user.setResetKey("reset key"); + userRepository.saveAndFlush(user); + + KeyAndPasswordVM keyAndPassword = new KeyAndPasswordVM(); + keyAndPassword.setKey(user.getResetKey()); + keyAndPassword.setNewPassword("new password"); + + restAccountMockMvc.perform( + post("/api/account/reset-password/finish") + .contentType(MediaType.APPLICATION_JSON) + .content(TestUtil.convertObjectToJsonBytes(keyAndPassword)) + .with(csrf())) + .andExpect(status().isOk()); + + User updatedUser = userRepository.findOneByLogin(user.getLogin()).orElse(null); + assertThat(passwordEncoder.matches(keyAndPassword.getNewPassword(), updatedUser.getPassword())).isTrue(); + } + + @Test + @Transactional(value="epasMedTransactionManager") + public void testFinishPasswordResetTooSmall() throws Exception { + User user = new User(); + user.setPassword(RandomStringUtils.random(60)); + user.setLogin("finish-password-reset-too-small"); + user.setEmail("finish-password-reset-too-small@example.com"); + user.setResetDate(Instant.now().plusSeconds(60)); + user.setResetKey("reset key too small"); + userRepository.saveAndFlush(user); + + KeyAndPasswordVM keyAndPassword = new KeyAndPasswordVM(); + keyAndPassword.setKey(user.getResetKey()); + keyAndPassword.setNewPassword("foo"); + + restAccountMockMvc.perform( + post("/api/account/reset-password/finish") + .contentType(MediaType.APPLICATION_JSON) + .content(TestUtil.convertObjectToJsonBytes(keyAndPassword)) + .with(csrf())) + .andExpect(status().isBadRequest()); + + User updatedUser = userRepository.findOneByLogin(user.getLogin()).orElse(null); + assertThat(passwordEncoder.matches(keyAndPassword.getNewPassword(), updatedUser.getPassword())).isFalse(); + } + + @Test + @Transactional(value="epasMedTransactionManager") + public void testFinishPasswordResetWrongKey() throws Exception { + KeyAndPasswordVM keyAndPassword = new KeyAndPasswordVM(); + keyAndPassword.setKey("wrong reset key"); + keyAndPassword.setNewPassword("new password"); + + restAccountMockMvc.perform( + post("/api/account/reset-password/finish") + .contentType(MediaType.APPLICATION_JSON) + .content(TestUtil.convertObjectToJsonBytes(keyAndPassword)) + .with(csrf())) + .andExpect(status().isInternalServerError()); + } +} diff --git a/src/test/java/it/cnr/isti/epasmed/web/rest/AuditResourceIT.java b/src/test/java/it/cnr/isti/epasmed/web/rest/AuditResourceIT.java new file mode 100644 index 0000000..52b84bb --- /dev/null +++ b/src/test/java/it/cnr/isti/epasmed/web/rest/AuditResourceIT.java @@ -0,0 +1,135 @@ +package it.cnr.isti.epasmed.web.rest; + +import it.cnr.isti.epasmed.EpasmedApp; +import it.cnr.isti.epasmed.domain.PersistentAuditEvent; +import it.cnr.isti.epasmed.repository.PersistenceAuditEventRepository; +import it.cnr.isti.epasmed.security.AuthoritiesConstants; + +import org.junit.jupiter.api.BeforeEach; +import org.junit.jupiter.api.Test; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.boot.test.autoconfigure.web.servlet.AutoConfigureMockMvc; +import org.springframework.boot.test.context.SpringBootTest; +import org.springframework.data.web.PageableHandlerMethodArgumentResolver; +import org.springframework.http.MediaType; +import org.springframework.security.test.context.support.WithMockUser; +import org.springframework.http.converter.json.MappingJackson2HttpMessageConverter; +import org.springframework.test.web.servlet.MockMvc; +import org.springframework.test.web.servlet.setup.MockMvcBuilders; +import org.springframework.transaction.annotation.Transactional; + +import java.time.Instant; + +import static org.assertj.core.api.AssertionsForClassTypes.assertThat; +import static org.hamcrest.Matchers.hasItem; +import static org.springframework.test.web.servlet.request.MockMvcRequestBuilders.get; +import static org.springframework.test.web.servlet.result.MockMvcResultMatchers.*; + +/** + * Integration tests for the {@link AuditResource} REST controller. + */ +@AutoConfigureMockMvc +@WithMockUser(authorities = AuthoritiesConstants.ADMIN) +@SpringBootTest(classes = EpasmedApp.class) +@Transactional(value="epasMedTransactionManager") +public class AuditResourceIT { + + private static final String SAMPLE_PRINCIPAL = "SAMPLE_PRINCIPAL"; + private static final String SAMPLE_TYPE = "SAMPLE_TYPE"; + private static final Instant SAMPLE_TIMESTAMP = Instant.parse("2015-08-04T10:11:30Z"); + private static final long SECONDS_PER_DAY = 60 * 60 * 24; + + @Autowired + private PersistenceAuditEventRepository auditEventRepository; + + private PersistentAuditEvent auditEvent; + + @Autowired + private MockMvc restAuditMockMvc; + + @BeforeEach + public void initTest() { + auditEventRepository.deleteAll(); + auditEvent = new PersistentAuditEvent(); + auditEvent.setAuditEventType(SAMPLE_TYPE); + auditEvent.setPrincipal(SAMPLE_PRINCIPAL); + auditEvent.setAuditEventDate(SAMPLE_TIMESTAMP); + } + + @Test + public void getAllAudits() throws Exception { + // Initialize the database + auditEventRepository.save(auditEvent); + + // Get all the audits + restAuditMockMvc.perform(get("/management/audits")) + .andExpect(status().isOk()) + .andExpect(content().contentType(MediaType.APPLICATION_JSON_VALUE)) + .andExpect(jsonPath("$.[*].principal").value(hasItem(SAMPLE_PRINCIPAL))); + } + + @Test + public void getAudit() throws Exception { + // Initialize the database + auditEventRepository.save(auditEvent); + + // Get the audit + restAuditMockMvc.perform(get("/management/audits/{id}", auditEvent.getId())) + .andExpect(status().isOk()) + .andExpect(content().contentType(MediaType.APPLICATION_JSON_VALUE)) + .andExpect(jsonPath("$.principal").value(SAMPLE_PRINCIPAL)); + } + + @Test + public void getAuditsByDate() throws Exception { + // Initialize the database + auditEventRepository.save(auditEvent); + + // Generate dates for selecting audits by date, making sure the period will contain the audit + String fromDate = SAMPLE_TIMESTAMP.minusSeconds(SECONDS_PER_DAY).toString().substring(0, 10); + String toDate = SAMPLE_TIMESTAMP.plusSeconds(SECONDS_PER_DAY).toString().substring(0, 10); + + // Get the audit + restAuditMockMvc.perform(get("/management/audits?fromDate="+fromDate+"&toDate="+toDate)) + .andExpect(status().isOk()) + .andExpect(content().contentType(MediaType.APPLICATION_JSON_VALUE)) + .andExpect(jsonPath("$.[*].principal").value(hasItem(SAMPLE_PRINCIPAL))); + } + + @Test + public void getNonExistingAuditsByDate() throws Exception { + // Initialize the database + auditEventRepository.save(auditEvent); + + // Generate dates for selecting audits by date, making sure the period will not contain the sample audit + String fromDate = SAMPLE_TIMESTAMP.minusSeconds(2*SECONDS_PER_DAY).toString().substring(0, 10); + String toDate = SAMPLE_TIMESTAMP.minusSeconds(SECONDS_PER_DAY).toString().substring(0, 10); + + // Query audits but expect no results + restAuditMockMvc.perform(get("/management/audits?fromDate=" + fromDate + "&toDate=" + toDate)) + .andExpect(status().isOk()) + .andExpect(content().contentType(MediaType.APPLICATION_JSON_VALUE)) + .andExpect(header().string("X-Total-Count", "0")); + } + + @Test + public void getNonExistingAudit() throws Exception { + // Get the audit + restAuditMockMvc.perform(get("/management/audits/{id}", Long.MAX_VALUE)) + .andExpect(status().isNotFound()); + } + + @Test + public void testPersistentAuditEventEquals() throws Exception { + TestUtil.equalsVerifier(PersistentAuditEvent.class); + PersistentAuditEvent auditEvent1 = new PersistentAuditEvent(); + auditEvent1.setId(1L); + PersistentAuditEvent auditEvent2 = new PersistentAuditEvent(); + auditEvent2.setId(auditEvent1.getId()); + assertThat(auditEvent1).isEqualTo(auditEvent2); + auditEvent2.setId(2L); + assertThat(auditEvent1).isNotEqualTo(auditEvent2); + auditEvent1.setId(null); + assertThat(auditEvent1).isNotEqualTo(auditEvent2); + } +} diff --git a/src/test/java/it/cnr/isti/epasmed/web/rest/ClientForwardControllerTest.java b/src/test/java/it/cnr/isti/epasmed/web/rest/ClientForwardControllerTest.java new file mode 100644 index 0000000..4e4404f --- /dev/null +++ b/src/test/java/it/cnr/isti/epasmed/web/rest/ClientForwardControllerTest.java @@ -0,0 +1,64 @@ +package it.cnr.isti.epasmed.web.rest; + +import org.junit.jupiter.api.BeforeEach; +import org.junit.jupiter.api.Test; +import org.springframework.http.MediaType; +import org.springframework.test.web.servlet.MockMvc; +import org.springframework.test.web.servlet.ResultActions; +import org.springframework.test.web.servlet.setup.MockMvcBuilders; +import org.springframework.web.bind.annotation.RequestMapping; +import org.springframework.web.bind.annotation.RestController; + +import static org.springframework.test.web.servlet.request.MockMvcRequestBuilders.get; +import static org.springframework.test.web.servlet.result.MockMvcResultMatchers.content; +import static org.springframework.test.web.servlet.result.MockMvcResultMatchers.forwardedUrl; +import static org.springframework.test.web.servlet.result.MockMvcResultMatchers.status; + +/** + * Unit tests for the {@link ClientForwardController} REST controller. + */ +public class ClientForwardControllerTest { + + private MockMvc restMockMvc; + + @BeforeEach + public void setup() { + ClientForwardController clientForwardController = new ClientForwardController(); + this.restMockMvc = MockMvcBuilders + .standaloneSetup(clientForwardController, new TestController()) + .build(); + } + + @Test + public void getBackendEndpoint() throws Exception { + restMockMvc.perform(get("/test")) + .andExpect(status().isOk()) + .andExpect(content().contentTypeCompatibleWith(MediaType.TEXT_PLAIN_VALUE)) + .andExpect(content().string("test")); + } + + @Test + public void getClientEndpoint() throws Exception { + ResultActions perform = restMockMvc.perform(get("/non-existant-mapping")); + perform + .andExpect(status().isOk()) + .andExpect(forwardedUrl("/")); + } + + @Test + public void getNestedClientEndpoint() throws Exception { + restMockMvc.perform(get("/admin/user-management")) + .andExpect(status().isOk()) + .andExpect(forwardedUrl("/")); + } + + + @RestController + public static class TestController { + + @RequestMapping(value = "/test") + public String test() { + return "test"; + } + } +} diff --git a/src/test/java/it/cnr/isti/epasmed/web/rest/SyncResourceIT.java b/src/test/java/it/cnr/isti/epasmed/web/rest/SyncResourceIT.java new file mode 100644 index 0000000..dfc0ad3 --- /dev/null +++ b/src/test/java/it/cnr/isti/epasmed/web/rest/SyncResourceIT.java @@ -0,0 +1,57 @@ +package it.cnr.isti.epasmed.web.rest; + +import static org.springframework.security.test.web.servlet.request.SecurityMockMvcRequestPostProcessors.csrf; +import static org.springframework.test.web.servlet.request.MockMvcRequestBuilders.post; +import static org.springframework.test.web.servlet.result.MockMvcResultMatchers.status; + +import org.junit.jupiter.api.BeforeEach; +import org.junit.jupiter.api.Test; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.boot.test.autoconfigure.web.servlet.AutoConfigureMockMvc; +import org.springframework.boot.test.context.SpringBootTest; +import org.springframework.core.env.Environment; +import org.springframework.security.test.context.support.WithMockUser; +import org.springframework.test.context.junit.jupiter.EnabledIf; +import org.springframework.test.web.servlet.MockMvc; + +import it.cnr.isti.epasmed.EpasmedApp; +import it.cnr.isti.epasmed.security.AuthoritiesConstants; + +/** + * Integration tests for the {@link SyncResource} REST controller. + */ +@AutoConfigureMockMvc +@WithMockUser(authorities = AuthoritiesConstants.ADMIN) +@SpringBootTest(classes = EpasmedApp.class) +@EnabledIf("false") +public class SyncResourceIT { + + private final Logger log = LoggerFactory.getLogger(getClass()); + + + @Autowired + private MockMvc restSyncMockMvc; + + @Autowired + private Environment environment; + + + + @BeforeEach + public void initTest() { + for (String profileName : environment.getActiveProfiles()) { + log.info("Currently active profile - " + profileName); + } + log.info("System env - " + System.getenv("spring.profiles.active")); + } + + @Test + public void syncReads() throws Exception { + restSyncMockMvc.perform(post("/api/sync/reads").with(csrf())) + .andExpect(status().is2xxSuccessful()); + } + + +} diff --git a/src/test/java/it/cnr/isti/epasmed/web/rest/TestUtil.java b/src/test/java/it/cnr/isti/epasmed/web/rest/TestUtil.java new file mode 100644 index 0000000..3eea4d5 --- /dev/null +++ b/src/test/java/it/cnr/isti/epasmed/web/rest/TestUtil.java @@ -0,0 +1,157 @@ +package it.cnr.isti.epasmed.web.rest; + +import com.fasterxml.jackson.annotation.JsonInclude; +import com.fasterxml.jackson.databind.ObjectMapper; +import com.fasterxml.jackson.databind.SerializationFeature; +import com.fasterxml.jackson.datatype.jsr310.JavaTimeModule; +import org.hamcrest.Description; +import org.hamcrest.TypeSafeDiagnosingMatcher; +import org.springframework.format.datetime.standard.DateTimeFormatterRegistrar; +import org.springframework.format.support.DefaultFormattingConversionService; +import org.springframework.format.support.FormattingConversionService; + +import java.io.IOException; +import java.time.ZonedDateTime; +import java.time.format.DateTimeParseException; +import java.util.List; + +import javax.persistence.EntityManager; +import javax.persistence.TypedQuery; +import javax.persistence.criteria.CriteriaBuilder; +import javax.persistence.criteria.CriteriaQuery; +import javax.persistence.criteria.Root; + +import static org.assertj.core.api.Assertions.assertThat; + +/** + * Utility class for testing REST controllers. + */ +public final class TestUtil { + + private static final ObjectMapper mapper = createObjectMapper(); + + private static ObjectMapper createObjectMapper() { + ObjectMapper mapper = new ObjectMapper(); + mapper.configure(SerializationFeature.WRITE_DURATIONS_AS_TIMESTAMPS, false); + mapper.setSerializationInclusion(JsonInclude.Include.NON_EMPTY); + mapper.registerModule(new JavaTimeModule()); + return mapper; + } + + /** + * Convert an object to JSON byte array. + * + * @param object the object to convert. + * @return the JSON byte array. + * @throws IOException + */ + public static byte[] convertObjectToJsonBytes(Object object) throws IOException { + return mapper.writeValueAsBytes(object); + } + + /** + * Create a byte array with a specific size filled with specified data. + * + * @param size the size of the byte array. + * @param data the data to put in the byte array. + * @return the JSON byte array. + */ + public static byte[] createByteArray(int size, String data) { + byte[] byteArray = new byte[size]; + for (int i = 0; i < size; i++) { + byteArray[i] = Byte.parseByte(data, 2); + } + return byteArray; + } + + /** + * A matcher that tests that the examined string represents the same instant as the reference datetime. + */ + public static class ZonedDateTimeMatcher extends TypeSafeDiagnosingMatcher { + + private final ZonedDateTime date; + + public ZonedDateTimeMatcher(ZonedDateTime date) { + this.date = date; + } + + @Override + protected boolean matchesSafely(String item, Description mismatchDescription) { + try { + if (!date.isEqual(ZonedDateTime.parse(item))) { + mismatchDescription.appendText("was ").appendValue(item); + return false; + } + return true; + } catch (DateTimeParseException e) { + mismatchDescription.appendText("was ").appendValue(item) + .appendText(", which could not be parsed as a ZonedDateTime"); + return false; + } + + } + + @Override + public void describeTo(Description description) { + description.appendText("a String representing the same Instant as ").appendValue(date); + } + } + + /** + * Creates a matcher that matches when the examined string represents the same instant as the reference datetime. + * + * @param date the reference datetime against which the examined string is checked. + */ + public static ZonedDateTimeMatcher sameInstant(ZonedDateTime date) { + return new ZonedDateTimeMatcher(date); + } + + /** + * Verifies the equals/hashcode contract on the domain object. + */ + public static void equalsVerifier(Class clazz) throws Exception { + T domainObject1 = clazz.getConstructor().newInstance(); + assertThat(domainObject1.toString()).isNotNull(); + assertThat(domainObject1).isEqualTo(domainObject1); + assertThat(domainObject1.hashCode()).isEqualTo(domainObject1.hashCode()); + // Test with an instance of another class + Object testOtherObject = new Object(); + assertThat(domainObject1).isNotEqualTo(testOtherObject); + assertThat(domainObject1).isNotEqualTo(null); + // Test with an instance of the same class + T domainObject2 = clazz.getConstructor().newInstance(); + assertThat(domainObject1).isNotEqualTo(domainObject2); + // HashCodes are equals because the objects are not persisted yet + assertThat(domainObject1.hashCode()).isEqualTo(domainObject2.hashCode()); + } + + /** + * Create a {@link FormattingConversionService} which use ISO date format, instead of the localized one. + * @return the {@link FormattingConversionService}. + */ + public static FormattingConversionService createFormattingConversionService() { + DefaultFormattingConversionService dfcs = new DefaultFormattingConversionService (); + DateTimeFormatterRegistrar registrar = new DateTimeFormatterRegistrar(); + registrar.setUseIsoFormat(true); + registrar.registerFormatters(dfcs); + return dfcs; + } + + /** + * Makes a an executes a query to the EntityManager finding all stored objects. + * @param The type of objects to be searched + * @param em The instance of the EntityManager + * @param clss The class type to be searched + * @return A list of all found objects + */ + public static List findAll(EntityManager em, Class clss) { + CriteriaBuilder cb = em.getCriteriaBuilder(); + CriteriaQuery cq = cb.createQuery(clss); + Root rootEntry = cq.from(clss); + CriteriaQuery all = cq.select(rootEntry); + TypedQuery allQuery = em.createQuery(all); + return allQuery.getResultList(); + } + + private TestUtil() {} +} diff --git a/src/test/java/it/cnr/isti/epasmed/web/rest/UserResourceIT.java b/src/test/java/it/cnr/isti/epasmed/web/rest/UserResourceIT.java new file mode 100644 index 0000000..693679b --- /dev/null +++ b/src/test/java/it/cnr/isti/epasmed/web/rest/UserResourceIT.java @@ -0,0 +1,590 @@ +package it.cnr.isti.epasmed.web.rest; + +import it.cnr.isti.epasmed.EpasmedApp; +import it.cnr.isti.epasmed.domain.Authority; +import it.cnr.isti.epasmed.domain.User; +import it.cnr.isti.epasmed.repository.UserRepository; +import it.cnr.isti.epasmed.security.AuthoritiesConstants; +import it.cnr.isti.epasmed.service.dto.UserDTO; +import it.cnr.isti.epasmed.service.mapper.UserMapper; +import it.cnr.isti.epasmed.web.rest.vm.ManagedUserVM; +import org.apache.commons.lang3.RandomStringUtils; +import org.junit.jupiter.api.BeforeEach; +import org.junit.jupiter.api.Test; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.boot.test.autoconfigure.web.servlet.AutoConfigureMockMvc; +import org.springframework.boot.test.context.SpringBootTest; +import org.springframework.http.MediaType; +import org.springframework.security.test.context.support.WithMockUser; +import org.springframework.test.web.servlet.MockMvc; +import org.springframework.transaction.annotation.Transactional; + +import javax.persistence.EntityManager; +import java.time.Instant; +import java.util.*; +import java.util.function.Consumer; + +import static org.assertj.core.api.Assertions.assertThat; +import static org.hamcrest.Matchers.hasItems; +import static org.hamcrest.Matchers.hasItem; +import static org.springframework.security.test.web.servlet.request.SecurityMockMvcRequestPostProcessors.csrf; +import static org.springframework.test.web.servlet.request.MockMvcRequestBuilders.*; +import static org.springframework.test.web.servlet.result.MockMvcResultMatchers.*; + +/** + * Integration tests for the {@link UserResource} REST controller. + */ +@AutoConfigureMockMvc +@WithMockUser(authorities = AuthoritiesConstants.ADMIN) +@SpringBootTest(classes = EpasmedApp.class) +public class UserResourceIT { + + private static final String DEFAULT_LOGIN = "johndoe"; + private static final String UPDATED_LOGIN = "jhipster"; + + private static final Long DEFAULT_ID = 1L; + + private static final String DEFAULT_PASSWORD = "passjohndoe"; + private static final String UPDATED_PASSWORD = "passjhipster"; + + private static final String DEFAULT_EMAIL = "johndoe@localhost"; + private static final String UPDATED_EMAIL = "jhipster@localhost"; + + private static final String DEFAULT_FIRSTNAME = "john"; + private static final String UPDATED_FIRSTNAME = "jhipsterFirstName"; + + private static final String DEFAULT_LASTNAME = "doe"; + private static final String UPDATED_LASTNAME = "jhipsterLastName"; + + private static final String DEFAULT_IMAGEURL = "http://placehold.it/50x50"; + private static final String UPDATED_IMAGEURL = "http://placehold.it/40x40"; + + private static final String DEFAULT_LANGKEY = "en"; + private static final String UPDATED_LANGKEY = "fr"; + + @Autowired + private UserRepository userRepository; + + @Autowired + private UserMapper userMapper; + + @Autowired + private EntityManager em; + + @Autowired + private MockMvc restUserMockMvc; + + private User user; + + /** + * Create a User. + * + * This is a static method, as tests for other entities might also need it, + * if they test an entity which has a required relationship to the User entity. + */ + public static User createEntity(EntityManager em) { + User user = new User(); + user.setLogin(DEFAULT_LOGIN + RandomStringUtils.randomAlphabetic(5)); + user.setPassword(RandomStringUtils.random(60)); + user.setActivated(true); + user.setEmail(RandomStringUtils.randomAlphabetic(5) + DEFAULT_EMAIL); + user.setFirstName(DEFAULT_FIRSTNAME); + user.setLastName(DEFAULT_LASTNAME); + user.setImageUrl(DEFAULT_IMAGEURL); + user.setLangKey(DEFAULT_LANGKEY); + return user; + } + + @BeforeEach + public void initTest() { + user = createEntity(em); + user.setLogin(DEFAULT_LOGIN); + user.setEmail(DEFAULT_EMAIL); + } + + @Test + @Transactional(value="epasMedTransactionManager") + public void createUser() throws Exception { + int databaseSizeBeforeCreate = userRepository.findAll().size(); + + // Create the User + ManagedUserVM managedUserVM = new ManagedUserVM(); + managedUserVM.setLogin(DEFAULT_LOGIN); + managedUserVM.setPassword(DEFAULT_PASSWORD); + managedUserVM.setFirstName(DEFAULT_FIRSTNAME); + managedUserVM.setLastName(DEFAULT_LASTNAME); + managedUserVM.setEmail(DEFAULT_EMAIL); + managedUserVM.setActivated(true); + managedUserVM.setImageUrl(DEFAULT_IMAGEURL); + managedUserVM.setLangKey(DEFAULT_LANGKEY); + managedUserVM.setAuthorities(Collections.singleton(AuthoritiesConstants.USER)); + + restUserMockMvc.perform(post("/api/users") + .contentType(MediaType.APPLICATION_JSON) + .content(TestUtil.convertObjectToJsonBytes(managedUserVM)) + .with(csrf())) + .andExpect(status().isCreated()); + + // Validate the User in the database + assertPersistedUsers(users -> { + assertThat(users).hasSize(databaseSizeBeforeCreate + 1); + User testUser = users.get(users.size() - 1); + assertThat(testUser.getLogin()).isEqualTo(DEFAULT_LOGIN); + assertThat(testUser.getFirstName()).isEqualTo(DEFAULT_FIRSTNAME); + assertThat(testUser.getLastName()).isEqualTo(DEFAULT_LASTNAME); + assertThat(testUser.getEmail()).isEqualTo(DEFAULT_EMAIL); + assertThat(testUser.getImageUrl()).isEqualTo(DEFAULT_IMAGEURL); + assertThat(testUser.getLangKey()).isEqualTo(DEFAULT_LANGKEY); + }); + } + + @Test + @Transactional(value="epasMedTransactionManager") + public void createUserWithExistingId() throws Exception { + int databaseSizeBeforeCreate = userRepository.findAll().size(); + + ManagedUserVM managedUserVM = new ManagedUserVM(); + managedUserVM.setId(1L); + managedUserVM.setLogin(DEFAULT_LOGIN); + managedUserVM.setPassword(DEFAULT_PASSWORD); + managedUserVM.setFirstName(DEFAULT_FIRSTNAME); + managedUserVM.setLastName(DEFAULT_LASTNAME); + managedUserVM.setEmail(DEFAULT_EMAIL); + managedUserVM.setActivated(true); + managedUserVM.setImageUrl(DEFAULT_IMAGEURL); + managedUserVM.setLangKey(DEFAULT_LANGKEY); + managedUserVM.setAuthorities(Collections.singleton(AuthoritiesConstants.USER)); + + // An entity with an existing ID cannot be created, so this API call must fail + restUserMockMvc.perform(post("/api/users") + .contentType(MediaType.APPLICATION_JSON) + .content(TestUtil.convertObjectToJsonBytes(managedUserVM)) + .with(csrf())) + .andExpect(status().isBadRequest()); + + // Validate the User in the database + assertPersistedUsers(users -> assertThat(users).hasSize(databaseSizeBeforeCreate)); + } + + @Test + @Transactional(value="epasMedTransactionManager") + public void createUserWithExistingLogin() throws Exception { + // Initialize the database + userRepository.saveAndFlush(user); + int databaseSizeBeforeCreate = userRepository.findAll().size(); + + ManagedUserVM managedUserVM = new ManagedUserVM(); + managedUserVM.setLogin(DEFAULT_LOGIN);// this login should already be used + managedUserVM.setPassword(DEFAULT_PASSWORD); + managedUserVM.setFirstName(DEFAULT_FIRSTNAME); + managedUserVM.setLastName(DEFAULT_LASTNAME); + managedUserVM.setEmail("anothermail@localhost"); + managedUserVM.setActivated(true); + managedUserVM.setImageUrl(DEFAULT_IMAGEURL); + managedUserVM.setLangKey(DEFAULT_LANGKEY); + managedUserVM.setAuthorities(Collections.singleton(AuthoritiesConstants.USER)); + + // Create the User + restUserMockMvc.perform(post("/api/users") + .contentType(MediaType.APPLICATION_JSON) + .content(TestUtil.convertObjectToJsonBytes(managedUserVM)) + .with(csrf())) + .andExpect(status().isBadRequest()); + + // Validate the User in the database + assertPersistedUsers(users -> assertThat(users).hasSize(databaseSizeBeforeCreate)); + } + + @Test + @Transactional(value="epasMedTransactionManager") + public void createUserWithExistingEmail() throws Exception { + // Initialize the database + userRepository.saveAndFlush(user); + int databaseSizeBeforeCreate = userRepository.findAll().size(); + + ManagedUserVM managedUserVM = new ManagedUserVM(); + managedUserVM.setLogin("anotherlogin"); + managedUserVM.setPassword(DEFAULT_PASSWORD); + managedUserVM.setFirstName(DEFAULT_FIRSTNAME); + managedUserVM.setLastName(DEFAULT_LASTNAME); + managedUserVM.setEmail(DEFAULT_EMAIL);// this email should already be used + managedUserVM.setActivated(true); + managedUserVM.setImageUrl(DEFAULT_IMAGEURL); + managedUserVM.setLangKey(DEFAULT_LANGKEY); + managedUserVM.setAuthorities(Collections.singleton(AuthoritiesConstants.USER)); + + // Create the User + restUserMockMvc.perform(post("/api/users") + .contentType(MediaType.APPLICATION_JSON) + .content(TestUtil.convertObjectToJsonBytes(managedUserVM)) + .with(csrf())) + .andExpect(status().isBadRequest()); + + // Validate the User in the database + assertPersistedUsers(users -> assertThat(users).hasSize(databaseSizeBeforeCreate)); + } + + @Test + @Transactional(value="epasMedTransactionManager") + public void getAllUsers() throws Exception { + // Initialize the database + userRepository.saveAndFlush(user); + + // Get all the users + restUserMockMvc.perform(get("/api/users?sort=id,desc") + .accept(MediaType.APPLICATION_JSON)) + .andExpect(status().isOk()) + .andExpect(content().contentType(MediaType.APPLICATION_JSON_VALUE)) + .andExpect(jsonPath("$.[*].login").value(hasItem(DEFAULT_LOGIN))) + .andExpect(jsonPath("$.[*].firstName").value(hasItem(DEFAULT_FIRSTNAME))) + .andExpect(jsonPath("$.[*].lastName").value(hasItem(DEFAULT_LASTNAME))) + .andExpect(jsonPath("$.[*].email").value(hasItem(DEFAULT_EMAIL))) + .andExpect(jsonPath("$.[*].imageUrl").value(hasItem(DEFAULT_IMAGEURL))) + .andExpect(jsonPath("$.[*].langKey").value(hasItem(DEFAULT_LANGKEY))); + } + + @Test + @Transactional(value="epasMedTransactionManager") + void getAllUsersSortedByParameters() throws Exception { + // Initialize the database + userRepository.saveAndFlush(user); + + restUserMockMvc.perform(get("/api/users?sort=resetKey,desc").accept(MediaType.APPLICATION_JSON)).andExpect(status().isBadRequest()); + restUserMockMvc.perform(get("/api/users?sort=password,desc").accept(MediaType.APPLICATION_JSON)).andExpect(status().isBadRequest()); + restUserMockMvc.perform(get("/api/users?sort=resetKey,id,desc").accept(MediaType.APPLICATION_JSON)).andExpect(status().isBadRequest()); + restUserMockMvc.perform(get("/api/users?sort=id,desc").accept(MediaType.APPLICATION_JSON)).andExpect(status().isOk()); + } + + @Test + @Transactional(value="epasMedTransactionManager") + public void getUser() throws Exception { + // Initialize the database + userRepository.saveAndFlush(user); + + // Get the user + restUserMockMvc.perform(get("/api/users/{login}", user.getLogin())) + .andExpect(status().isOk()) + .andExpect(content().contentType(MediaType.APPLICATION_JSON_VALUE)) + .andExpect(jsonPath("$.login").value(user.getLogin())) + .andExpect(jsonPath("$.firstName").value(DEFAULT_FIRSTNAME)) + .andExpect(jsonPath("$.lastName").value(DEFAULT_LASTNAME)) + .andExpect(jsonPath("$.email").value(DEFAULT_EMAIL)) + .andExpect(jsonPath("$.imageUrl").value(DEFAULT_IMAGEURL)) + .andExpect(jsonPath("$.langKey").value(DEFAULT_LANGKEY)); + + } + + @Test + @Transactional(value="epasMedTransactionManager") + public void getNonExistingUser() throws Exception { + restUserMockMvc.perform(get("/api/users/unknown")) + .andExpect(status().isNotFound()); + } + + @Test + @Transactional(value="epasMedTransactionManager") + public void updateUser() throws Exception { + // Initialize the database + userRepository.saveAndFlush(user); + int databaseSizeBeforeUpdate = userRepository.findAll().size(); + + // Update the user + User updatedUser = userRepository.findById(user.getId()).get(); + + ManagedUserVM managedUserVM = new ManagedUserVM(); + managedUserVM.setId(updatedUser.getId()); + managedUserVM.setLogin(updatedUser.getLogin()); + managedUserVM.setPassword(UPDATED_PASSWORD); + managedUserVM.setFirstName(UPDATED_FIRSTNAME); + managedUserVM.setLastName(UPDATED_LASTNAME); + managedUserVM.setEmail(UPDATED_EMAIL); + managedUserVM.setActivated(updatedUser.getActivated()); + managedUserVM.setImageUrl(UPDATED_IMAGEURL); + managedUserVM.setLangKey(UPDATED_LANGKEY); + managedUserVM.setCreatedBy(updatedUser.getCreatedBy()); + managedUserVM.setCreatedDate(updatedUser.getCreatedDate()); + managedUserVM.setLastModifiedBy(updatedUser.getLastModifiedBy()); + managedUserVM.setLastModifiedDate(updatedUser.getLastModifiedDate()); + managedUserVM.setAuthorities(Collections.singleton(AuthoritiesConstants.USER)); + + restUserMockMvc.perform(put("/api/users") + .contentType(MediaType.APPLICATION_JSON) + .content(TestUtil.convertObjectToJsonBytes(managedUserVM)) + .with(csrf())) + .andExpect(status().isOk()); + + // Validate the User in the database + assertPersistedUsers(users -> { + assertThat(users).hasSize(databaseSizeBeforeUpdate); + User testUser = users.get(users.size() - 1); + assertThat(testUser.getFirstName()).isEqualTo(UPDATED_FIRSTNAME); + assertThat(testUser.getLastName()).isEqualTo(UPDATED_LASTNAME); + assertThat(testUser.getEmail()).isEqualTo(UPDATED_EMAIL); + assertThat(testUser.getImageUrl()).isEqualTo(UPDATED_IMAGEURL); + assertThat(testUser.getLangKey()).isEqualTo(UPDATED_LANGKEY); + }); + } + + @Test + @Transactional(value="epasMedTransactionManager") + public void updateUserLogin() throws Exception { + // Initialize the database + userRepository.saveAndFlush(user); + int databaseSizeBeforeUpdate = userRepository.findAll().size(); + + // Update the user + User updatedUser = userRepository.findById(user.getId()).get(); + + ManagedUserVM managedUserVM = new ManagedUserVM(); + managedUserVM.setId(updatedUser.getId()); + managedUserVM.setLogin(UPDATED_LOGIN); + managedUserVM.setPassword(UPDATED_PASSWORD); + managedUserVM.setFirstName(UPDATED_FIRSTNAME); + managedUserVM.setLastName(UPDATED_LASTNAME); + managedUserVM.setEmail(UPDATED_EMAIL); + managedUserVM.setActivated(updatedUser.getActivated()); + managedUserVM.setImageUrl(UPDATED_IMAGEURL); + managedUserVM.setLangKey(UPDATED_LANGKEY); + managedUserVM.setCreatedBy(updatedUser.getCreatedBy()); + managedUserVM.setCreatedDate(updatedUser.getCreatedDate()); + managedUserVM.setLastModifiedBy(updatedUser.getLastModifiedBy()); + managedUserVM.setLastModifiedDate(updatedUser.getLastModifiedDate()); + managedUserVM.setAuthorities(Collections.singleton(AuthoritiesConstants.USER)); + + restUserMockMvc.perform(put("/api/users") + .contentType(MediaType.APPLICATION_JSON) + .content(TestUtil.convertObjectToJsonBytes(managedUserVM)) + .with(csrf())) + .andExpect(status().isOk()); + + // Validate the User in the database + assertPersistedUsers(users -> { + assertThat(users).hasSize(databaseSizeBeforeUpdate); + User testUser = users.get(users.size() - 1); + assertThat(testUser.getLogin()).isEqualTo(UPDATED_LOGIN); + assertThat(testUser.getFirstName()).isEqualTo(UPDATED_FIRSTNAME); + assertThat(testUser.getLastName()).isEqualTo(UPDATED_LASTNAME); + assertThat(testUser.getEmail()).isEqualTo(UPDATED_EMAIL); + assertThat(testUser.getImageUrl()).isEqualTo(UPDATED_IMAGEURL); + assertThat(testUser.getLangKey()).isEqualTo(UPDATED_LANGKEY); + }); + } + + @Test + @Transactional(value="epasMedTransactionManager") + public void updateUserExistingEmail() throws Exception { + // Initialize the database with 2 users + userRepository.saveAndFlush(user); + + User anotherUser = new User(); + anotherUser.setLogin("jhipster"); + anotherUser.setPassword(RandomStringUtils.random(60)); + anotherUser.setActivated(true); + anotherUser.setEmail("jhipster@localhost"); + anotherUser.setFirstName("java"); + anotherUser.setLastName("hipster"); + anotherUser.setImageUrl(""); + anotherUser.setLangKey("en"); + userRepository.saveAndFlush(anotherUser); + + // Update the user + User updatedUser = userRepository.findById(user.getId()).get(); + + ManagedUserVM managedUserVM = new ManagedUserVM(); + managedUserVM.setId(updatedUser.getId()); + managedUserVM.setLogin(updatedUser.getLogin()); + managedUserVM.setPassword(updatedUser.getPassword()); + managedUserVM.setFirstName(updatedUser.getFirstName()); + managedUserVM.setLastName(updatedUser.getLastName()); + managedUserVM.setEmail("jhipster@localhost");// this email should already be used by anotherUser + managedUserVM.setActivated(updatedUser.getActivated()); + managedUserVM.setImageUrl(updatedUser.getImageUrl()); + managedUserVM.setLangKey(updatedUser.getLangKey()); + managedUserVM.setCreatedBy(updatedUser.getCreatedBy()); + managedUserVM.setCreatedDate(updatedUser.getCreatedDate()); + managedUserVM.setLastModifiedBy(updatedUser.getLastModifiedBy()); + managedUserVM.setLastModifiedDate(updatedUser.getLastModifiedDate()); + managedUserVM.setAuthorities(Collections.singleton(AuthoritiesConstants.USER)); + + restUserMockMvc.perform(put("/api/users") + .contentType(MediaType.APPLICATION_JSON) + .content(TestUtil.convertObjectToJsonBytes(managedUserVM)) + .with(csrf())) + .andExpect(status().isBadRequest()); + } + + @Test + @Transactional(value="epasMedTransactionManager") + public void updateUserExistingLogin() throws Exception { + // Initialize the database + userRepository.saveAndFlush(user); + + User anotherUser = new User(); + anotherUser.setLogin("jhipster"); + anotherUser.setPassword(RandomStringUtils.random(60)); + anotherUser.setActivated(true); + anotherUser.setEmail("jhipster@localhost"); + anotherUser.setFirstName("java"); + anotherUser.setLastName("hipster"); + anotherUser.setImageUrl(""); + anotherUser.setLangKey("en"); + userRepository.saveAndFlush(anotherUser); + + // Update the user + User updatedUser = userRepository.findById(user.getId()).get(); + + ManagedUserVM managedUserVM = new ManagedUserVM(); + managedUserVM.setId(updatedUser.getId()); + managedUserVM.setLogin("jhipster");// this login should already be used by anotherUser + managedUserVM.setPassword(updatedUser.getPassword()); + managedUserVM.setFirstName(updatedUser.getFirstName()); + managedUserVM.setLastName(updatedUser.getLastName()); + managedUserVM.setEmail(updatedUser.getEmail()); + managedUserVM.setActivated(updatedUser.getActivated()); + managedUserVM.setImageUrl(updatedUser.getImageUrl()); + managedUserVM.setLangKey(updatedUser.getLangKey()); + managedUserVM.setCreatedBy(updatedUser.getCreatedBy()); + managedUserVM.setCreatedDate(updatedUser.getCreatedDate()); + managedUserVM.setLastModifiedBy(updatedUser.getLastModifiedBy()); + managedUserVM.setLastModifiedDate(updatedUser.getLastModifiedDate()); + managedUserVM.setAuthorities(Collections.singleton(AuthoritiesConstants.USER)); + + restUserMockMvc.perform(put("/api/users") + .contentType(MediaType.APPLICATION_JSON) + .content(TestUtil.convertObjectToJsonBytes(managedUserVM)) + .with(csrf())) + .andExpect(status().isBadRequest()); + } + + @Test + @Transactional(value="epasMedTransactionManager") + public void deleteUser() throws Exception { + // Initialize the database + userRepository.saveAndFlush(user); + int databaseSizeBeforeDelete = userRepository.findAll().size(); + + // Delete the user + restUserMockMvc.perform(delete("/api/users/{login}", user.getLogin()) + .accept(MediaType.APPLICATION_JSON) + .with(csrf())) + .andExpect(status().isNoContent()); + + // Validate the database is empty + assertPersistedUsers(users -> assertThat(users).hasSize(databaseSizeBeforeDelete - 1)); + } + + @Test + @Transactional(value="epasMedTransactionManager") + public void getAllAuthorities() throws Exception { + restUserMockMvc.perform(get("/api/users/authorities") + .accept(MediaType.APPLICATION_JSON) + .contentType(MediaType.APPLICATION_JSON)) + .andExpect(status().isOk()) + .andExpect(content().contentType(MediaType.APPLICATION_JSON_VALUE)) + .andExpect(jsonPath("$").isArray()) + .andExpect(jsonPath("$").value(hasItems(AuthoritiesConstants.USER, AuthoritiesConstants.ADMIN))); + } + + @Test + public void testUserEquals() throws Exception { + TestUtil.equalsVerifier(User.class); + User user1 = new User(); + user1.setId(1L); + User user2 = new User(); + user2.setId(user1.getId()); + assertThat(user1).isEqualTo(user2); + user2.setId(2L); + assertThat(user1).isNotEqualTo(user2); + user1.setId(null); + assertThat(user1).isNotEqualTo(user2); + } + + @Test + public void testUserDTOtoUser() { + UserDTO userDTO = new UserDTO(); + userDTO.setId(DEFAULT_ID); + userDTO.setLogin(DEFAULT_LOGIN); + userDTO.setFirstName(DEFAULT_FIRSTNAME); + userDTO.setLastName(DEFAULT_LASTNAME); + userDTO.setEmail(DEFAULT_EMAIL); + userDTO.setActivated(true); + userDTO.setImageUrl(DEFAULT_IMAGEURL); + userDTO.setLangKey(DEFAULT_LANGKEY); + userDTO.setCreatedBy(DEFAULT_LOGIN); + userDTO.setLastModifiedBy(DEFAULT_LOGIN); + userDTO.setAuthorities(Collections.singleton(AuthoritiesConstants.USER)); + + User user = userMapper.userDTOToUser(userDTO); + assertThat(user.getId()).isEqualTo(DEFAULT_ID); + assertThat(user.getLogin()).isEqualTo(DEFAULT_LOGIN); + assertThat(user.getFirstName()).isEqualTo(DEFAULT_FIRSTNAME); + assertThat(user.getLastName()).isEqualTo(DEFAULT_LASTNAME); + assertThat(user.getEmail()).isEqualTo(DEFAULT_EMAIL); + assertThat(user.getActivated()).isEqualTo(true); + assertThat(user.getImageUrl()).isEqualTo(DEFAULT_IMAGEURL); + assertThat(user.getLangKey()).isEqualTo(DEFAULT_LANGKEY); + assertThat(user.getCreatedBy()).isNull(); + assertThat(user.getCreatedDate()).isNotNull(); + assertThat(user.getLastModifiedBy()).isNull(); + assertThat(user.getLastModifiedDate()).isNotNull(); + assertThat(user.getAuthorities()).extracting("name").containsExactly(AuthoritiesConstants.USER); + } + + @Test + public void testUserToUserDTO() { + user.setId(DEFAULT_ID); + user.setCreatedBy(DEFAULT_LOGIN); + user.setCreatedDate(Instant.now()); + user.setLastModifiedBy(DEFAULT_LOGIN); + user.setLastModifiedDate(Instant.now()); + Set authorities = new HashSet<>(); + Authority authority = new Authority(); + authority.setName(AuthoritiesConstants.USER); + authorities.add(authority); + user.setAuthorities(authorities); + + UserDTO userDTO = userMapper.userToUserDTO(user); + + assertThat(userDTO.getId()).isEqualTo(DEFAULT_ID); + assertThat(userDTO.getLogin()).isEqualTo(DEFAULT_LOGIN); + assertThat(userDTO.getFirstName()).isEqualTo(DEFAULT_FIRSTNAME); + assertThat(userDTO.getLastName()).isEqualTo(DEFAULT_LASTNAME); + assertThat(userDTO.getEmail()).isEqualTo(DEFAULT_EMAIL); + assertThat(userDTO.isActivated()).isEqualTo(true); + assertThat(userDTO.getImageUrl()).isEqualTo(DEFAULT_IMAGEURL); + assertThat(userDTO.getLangKey()).isEqualTo(DEFAULT_LANGKEY); + assertThat(userDTO.getCreatedBy()).isEqualTo(DEFAULT_LOGIN); + assertThat(userDTO.getCreatedDate()).isEqualTo(user.getCreatedDate()); + assertThat(userDTO.getLastModifiedBy()).isEqualTo(DEFAULT_LOGIN); + assertThat(userDTO.getLastModifiedDate()).isEqualTo(user.getLastModifiedDate()); + assertThat(userDTO.getAuthorities()).containsExactly(AuthoritiesConstants.USER); + assertThat(userDTO.toString()).isNotNull(); + } + + @Test + public void testAuthorityEquals() { + Authority authorityA = new Authority(); + assertThat(authorityA).isEqualTo(authorityA); + assertThat(authorityA).isNotEqualTo(null); + assertThat(authorityA).isNotEqualTo(new Object()); + assertThat(authorityA.hashCode()).isEqualTo(0); + assertThat(authorityA.toString()).isNotNull(); + + Authority authorityB = new Authority(); + assertThat(authorityA).isEqualTo(authorityB); + + authorityB.setName(AuthoritiesConstants.ADMIN); + assertThat(authorityA).isNotEqualTo(authorityB); + + authorityA.setName(AuthoritiesConstants.USER); + assertThat(authorityA).isNotEqualTo(authorityB); + + authorityB.setName(AuthoritiesConstants.USER); + assertThat(authorityA).isEqualTo(authorityB); + assertThat(authorityA.hashCode()).isEqualTo(authorityB.hashCode()); + } + + private void assertPersistedUsers(Consumer> userAssertion) { + userAssertion.accept(userRepository.findAll()); + } +} diff --git a/src/test/java/it/cnr/isti/epasmed/web/rest/WithUnauthenticatedMockUser.java b/src/test/java/it/cnr/isti/epasmed/web/rest/WithUnauthenticatedMockUser.java new file mode 100644 index 0000000..8452ed8 --- /dev/null +++ b/src/test/java/it/cnr/isti/epasmed/web/rest/WithUnauthenticatedMockUser.java @@ -0,0 +1,24 @@ +package it.cnr.isti.epasmed.web.rest; + +import org.springframework.security.core.context.SecurityContext; +import org.springframework.security.core.context.SecurityContextHolder; +import org.springframework.security.test.context.support.WithSecurityContext; +import org.springframework.security.test.context.support.WithSecurityContextFactory; + +import java.lang.annotation.ElementType; +import java.lang.annotation.Retention; +import java.lang.annotation.RetentionPolicy; +import java.lang.annotation.Target; + +@Target({ElementType.METHOD, ElementType.TYPE}) +@Retention(RetentionPolicy.RUNTIME) +@WithSecurityContext(factory = WithUnauthenticatedMockUser.Factory.class) +public @interface WithUnauthenticatedMockUser { + + class Factory implements WithSecurityContextFactory { + @Override + public SecurityContext createSecurityContext(WithUnauthenticatedMockUser annotation) { + return SecurityContextHolder.createEmptyContext(); + } + } +} diff --git a/src/test/java/it/cnr/isti/epasmed/web/rest/epas/EPASAffiliationsResourceIT.java b/src/test/java/it/cnr/isti/epasmed/web/rest/epas/EPASAffiliationsResourceIT.java new file mode 100644 index 0000000..eb053ec --- /dev/null +++ b/src/test/java/it/cnr/isti/epasmed/web/rest/epas/EPASAffiliationsResourceIT.java @@ -0,0 +1,318 @@ +package it.cnr.isti.epasmed.web.rest.epas; + +import static org.springframework.security.test.web.servlet.request.SecurityMockMvcRequestPostProcessors.csrf; +import static org.springframework.test.web.servlet.request.MockMvcRequestBuilders.delete; +import static org.springframework.test.web.servlet.request.MockMvcRequestBuilders.get; +import static org.springframework.test.web.servlet.request.MockMvcRequestBuilders.post; +import static org.springframework.test.web.servlet.request.MockMvcRequestBuilders.put; +import static org.springframework.test.web.servlet.result.MockMvcResultMatchers.status; + +import java.nio.file.Files; +import java.nio.file.Paths; +import java.util.ArrayList; +import java.util.List; +import java.util.function.Function; +import java.util.stream.Collectors; +import java.util.stream.Stream; + +import org.junit.jupiter.api.BeforeEach; +import org.junit.jupiter.api.Test; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.boot.test.autoconfigure.web.servlet.AutoConfigureMockMvc; +import org.springframework.boot.test.context.SpringBootTest; +import org.springframework.core.env.Environment; +import org.springframework.http.MediaType; +import org.springframework.security.test.context.support.WithMockUser; +import org.springframework.test.context.junit.jupiter.EnabledIf; +import org.springframework.test.web.servlet.MockMvc; +import org.springframework.test.web.servlet.MvcResult; + +import com.fasterxml.jackson.core.type.TypeReference; +import com.fasterxml.jackson.databind.ObjectMapper; + +import it.cnr.isti.epasmed.EpasmedApp; +import it.cnr.isti.epasmed.epas.dto.EPASAffiliationsDTO; +import it.cnr.isti.epasmed.epas.mapper.EPASAffiliationsMapper; +import it.cnr.isti.epasmed.epas.model.EPASAffiliations; +import it.cnr.isti.epasmed.epas.model.EPASGroups; +import it.cnr.isti.epasmed.epas.model.EPASPersons; +import it.cnr.isti.epasmed.security.AuthoritiesConstants; +import it.cnr.isti.epasmed.web.rest.TestUtil; + +/** + * Integration tests for the {@link EPASAffiliationsResource} REST controller. + */ +@AutoConfigureMockMvc +@WithMockUser(authorities = AuthoritiesConstants.ADMIN) +@SpringBootTest(classes = EpasmedApp.class) +@EnabledIf("false") +public class EPASAffiliationsResourceIT { + + private final Logger log = LoggerFactory.getLogger(getClass()); + + private static final String OFFICE_DEFAULT_ID = "1"; + private static final String OFFICE_DEFAULT_NAME = "ISTI - Pisa"; + private static final String OFFICE_DEFAULT_CODE = "074000"; + private static final String OFFICE_DEFAULT_CODEID = "225200"; + + private static final String AFFILIATION_DEFAULT_ID = "1"; + private static final String AFFILIATION_DEFAULT_BEGINDATE = "2021-01-01"; + private static final String AFFILIATION_DEFAULT_ENDDATE = null; + private static final String AFFILIATION_DEFAULT_GROUPID = "1"; + private static final String AFFILIATION_DEFAULT_PERCENTAGE = "100"; + private static final String AFFILIATION_DEFAULT_PERSONID = "1"; + private static final String AFFILIATION_DEFAULT_EXTERNALID = "1"; + + private ArrayList epasAffiliationsDTOList; + + @Autowired + private MockMvc restEPASAffiliationsMockMvc; + + @Autowired + private EPASAffiliationsMapper epasAffiliationsMapper; + + @Autowired + private Environment environment; + + private EPASAffiliations epasAffiliations; + + private EPASAffiliationsDTO epasAffiliationsDTO; + + private EPASAffiliationsDTO createEPASAffiliationsDTO() { + EPASAffiliationsDTO epasAffiliationsDTO = new EPASAffiliationsDTO(); + epasAffiliationsDTO.setBeginDate(AFFILIATION_DEFAULT_BEGINDATE); + epasAffiliationsDTO.setEndDate(AFFILIATION_DEFAULT_ENDDATE); + epasAffiliationsDTO.setGroupId(AFFILIATION_DEFAULT_GROUPID); + epasAffiliationsDTO.setId(AFFILIATION_DEFAULT_ID); + epasAffiliationsDTO.setPercentage(AFFILIATION_DEFAULT_PERCENTAGE); + epasAffiliationsDTO.setPersonId(AFFILIATION_DEFAULT_PERSONID); + epasAffiliationsDTO.setExternalId(AFFILIATION_DEFAULT_EXTERNALID); + return epasAffiliationsDTO; + } + + @BeforeEach + public void initTest() { + for (String profileName : environment.getActiveProfiles()) { + log.info("Currently active profile - " + profileName); + } + epasAffiliationsDTO = createEPASAffiliationsDTO(); + } + + @Test + public void getAffiliation() throws Exception { + restEPASAffiliationsMockMvc.perform(get("/api/epas/affiliations/" + 1)).andExpect(status().isOk()); + } + + @Test + public void getAffiliationByGroup() throws Exception { + restEPASAffiliationsMockMvc.perform(get("/api/epas/affiliations/byGroup?id=" + 23)).andExpect(status().isOk()); + } + + @Test + public void getAffiliationByPersonId() throws Exception { + restEPASAffiliationsMockMvc.perform(get("/api/epas/affiliations/byPerson?id=" + 43)).andExpect(status().isOk()); + } + + @Test + public void getAffiliationByPersonFiscalcode() throws Exception { + restEPASAffiliationsMockMvc.perform(get("/api/epas/affiliations/byPerson?fiscalCode=DZZPRZ79S23B832W")) + .andExpect(status().isOk()); + } + + @Test + public void getAffiliationList() throws Exception { + restEPASAffiliationsMockMvc.perform(get("/api/epas/affiliations?officeId=" + OFFICE_DEFAULT_ID)) + .andExpect(status().isOk()); + } + + @Test + public void createAffiliation() throws Exception { + restEPASAffiliationsMockMvc + .perform(post("/api/epas/affiliations").contentType(MediaType.APPLICATION_JSON) + .content(TestUtil.convertObjectToJsonBytes(epasAffiliationsDTO)).with(csrf())) + .andExpect(status().isCreated()); + } + + @Test + public void istiCreateAffiliations() throws Exception { + + String userDirectory = System.getProperty("user.dir"); + log.info(userDirectory); + List istiAffiliations = null; + try (Stream stream = Files + .lines(Paths.get("src/test/resources/it/cnr/isti/epasmed/web/rest/epas/affiliazioniRepartiISTIRimanenti.csv"))) { + istiAffiliations = stream.skip(1).map(mapToAffiliation).collect(Collectors.toList()); + } catch (Exception e) { + log.error(e.getLocalizedMessage(), e); + return; + } + + log.info("ISTI Persons loaded"); + + List pISTIAffiliations = new ArrayList<>(); + for (EPASAffiliations a : istiAffiliations) { + if (a.getPerson() != null && a.getPerson().getFiscalCode() != null + && !a.getPerson().getFiscalCode().isEmpty()) { + a.setPerson(retrievePersonByFiscalCode(a.getPerson().getFiscalCode())); + pISTIAffiliations.add(a); + + } + + } + + log.info("Added to ISTI Affiliations the EPAS Persons: {}", pISTIAffiliations.size()); + + List epasGroups = retrieveEPASGroups(); + log.info("EPAS Groups loaded"); + + List gISTIAffiliations = new ArrayList<>(); + for (EPASAffiliations a : pISTIAffiliations) { + if (a.getGroup() != null && a.getGroup().getExternalId() != null + && !a.getGroup().getExternalId().isEmpty()) { + EPASGroups g = retrieveEPASGroupByExternalId(epasGroups, a.getGroup().getExternalId()); + if (g != null) { + a.setGroup(g); + gISTIAffiliations.add(a); + } + } + } + + log.info("Added to ISTI Affiliations the EPAS Groups: "+gISTIAffiliations.size()); + + + List istiAffList = epasAffiliationsMapper + .epasAffiliationsToEPASAffiliationsDTOs(gISTIAffiliations); + + + + for (EPASAffiliationsDTO ia : istiAffList) { + log.info("Isti Aff: {}",ia); + restEPASAffiliationsMockMvc + .perform(post("/api/epas/affiliations").contentType(MediaType.APPLICATION_JSON) + .content(TestUtil.convertObjectToJsonBytes(ia)).with(csrf())) + .andExpect(status().is2xxSuccessful()); + } + + log.info("Added to ISTI Affiliations"); + + } + + private EPASGroups retrieveEPASGroupByExternalId(List epasGroups, String externalId) { + for (EPASGroups g : epasGroups) { + if (g.getExternalId() != null && !g.getExternalId().isEmpty() + && g.getExternalId().compareTo(externalId) == 0) { + return g; + } + } + return null; + } + + private List retrieveEPASGroups() throws Exception { + MvcResult result = restEPASAffiliationsMockMvc.perform(get("/api/epas/groups?officeId=" + OFFICE_DEFAULT_ID)) + .andExpect(status().isOk()).andReturn(); + ObjectMapper mapper = new ObjectMapper(); + List epasGroups = mapper.readValue(result.getResponse().getContentAsString(), + new TypeReference>() { + }); + return epasGroups; + } + + private EPASPersons retrievePersonByFiscalCode(String fiscalCode) throws Exception { + MvcResult result = restEPASAffiliationsMockMvc.perform(get("/api/epas/persons/show?fiscalCode=" + fiscalCode)) + .andExpect(status().isOk()).andReturn(); + + ObjectMapper mapper = new ObjectMapper(); + EPASPersons epasPersons = mapper.readValue(result.getResponse().getContentAsString(), + new TypeReference() { + }); + return epasPersons; + } + + private Function mapToAffiliation = (line) -> { + + String[] a = line.split(",");// a CSV has comma separated lines + + EPASAffiliations aff = new EPASAffiliations(); + + // <-- this is the first column in the csv file + if (a[0] != null && a[0].length() > 0) { + String externalId = a[0]; + if (externalId != null && !externalId.isEmpty()) { + aff.setExternalId(externalId); + } + } + + EPASGroups g = new EPASGroups(); + if (a[1] != null && a[1].length() > 0) { + String groupId = a[1]; + if (groupId != null && !groupId.isEmpty()) { + g.setExternalId(groupId); + + } + } + aff.setGroup(g); + + EPASPersons pe = new EPASPersons(); + /* + * if (a[2] != null && a[2].length() > 0) { String personId = a[2]; if (personId + * != null && !personId.isEmpty()) { + * + * } } + */ + + if (a[3] != null && a[3].length() > 0) { + String fiscalCode = a[3].substring(1, a[3].length() - 1); + if (fiscalCode != null && !fiscalCode.isEmpty()) { + pe.setFiscalCode(fiscalCode); + } + } + aff.setPerson(pe); + + if (a[4] != null && a[4].length() > 0) { + String beginDate = a[4].substring(1, a[4].length() - 1); + if (beginDate != null && !beginDate.isEmpty()) { + aff.setBeginDate(beginDate.substring(0, 10)); + } + } + + if (a[5] != null && a[5].length() > 0) { + String endDate = a[5].substring(1, a[5].length() - 1); + if (endDate != null && !endDate.isEmpty()) { + aff.setEndDate(endDate.substring(0, 10)); + } + } + + if (a[6] != null && a[6].length() > 0) { + String percentage = a[6]; + if (percentage != null && !percentage.isEmpty()) { + aff.setPercentage(percentage); + } + } + + return aff; + }; + + @Test + public void updateAffiliationById() throws Exception { + epasAffiliationsDTO.setPercentage("50"); + MvcResult mvcResult = restEPASAffiliationsMockMvc + .perform(put("/api/epas/affiliations/" + AFFILIATION_DEFAULT_ID).contentType(MediaType.APPLICATION_JSON) + .content(TestUtil.convertObjectToJsonBytes(epasAffiliationsDTO)).with(csrf())) + .andExpect(status().isOk()).andReturn(); + ObjectMapper mapper = new ObjectMapper(); + + EPASAffiliations epasAffiliations = mapper.readValue(mvcResult.getResponse().getContentAsByteArray(), + EPASAffiliations.class); + log.info("Updated EPAS Affiliation by id: " + epasAffiliations); + } + + @Test + public void deleteAffiliationById() throws Exception { + restEPASAffiliationsMockMvc.perform(delete("/api/epas/affiliations/" + AFFILIATION_DEFAULT_ID).with(csrf())) + .andExpect(status().is2xxSuccessful()); + log.info("Deleted EPAS Affiliation by id: " + AFFILIATION_DEFAULT_ID); + } + +} diff --git a/src/test/java/it/cnr/isti/epasmed/web/rest/epas/EPASContractsResourceIT.java b/src/test/java/it/cnr/isti/epasmed/web/rest/epas/EPASContractsResourceIT.java new file mode 100644 index 0000000..d4f2f61 --- /dev/null +++ b/src/test/java/it/cnr/isti/epasmed/web/rest/epas/EPASContractsResourceIT.java @@ -0,0 +1,330 @@ +package it.cnr.isti.epasmed.web.rest.epas; + +import static org.springframework.security.test.web.servlet.request.SecurityMockMvcRequestPostProcessors.csrf; +import static org.springframework.test.web.servlet.request.MockMvcRequestBuilders.delete; +import static org.springframework.test.web.servlet.request.MockMvcRequestBuilders.get; +import static org.springframework.test.web.servlet.request.MockMvcRequestBuilders.post; +import static org.springframework.test.web.servlet.request.MockMvcRequestBuilders.put; +import static org.springframework.test.web.servlet.result.MockMvcResultMatchers.status; + +import java.io.UnsupportedEncodingException; +import java.nio.file.Files; +import java.nio.file.Paths; +import java.util.ArrayList; +import java.util.List; +import java.util.function.Function; +import java.util.stream.Collectors; +import java.util.stream.Stream; + +import org.junit.jupiter.api.BeforeEach; +import org.junit.jupiter.api.Test; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.boot.test.autoconfigure.web.servlet.AutoConfigureMockMvc; +import org.springframework.boot.test.context.SpringBootTest; +import org.springframework.core.env.Environment; +import org.springframework.http.MediaType; +import org.springframework.security.test.context.support.WithMockUser; +import org.springframework.test.context.junit.jupiter.EnabledIf; +import org.springframework.test.web.servlet.MockMvc; +import org.springframework.test.web.servlet.MvcResult; + +import com.fasterxml.jackson.core.JsonProcessingException; +import com.fasterxml.jackson.core.type.TypeReference; +import com.fasterxml.jackson.databind.JsonMappingException; +import com.fasterxml.jackson.databind.ObjectMapper; + +import it.cnr.isti.epasmed.EpasmedApp; +import it.cnr.isti.epasmed.epas.dto.EPASContractsDTO; +import it.cnr.isti.epasmed.epas.mapper.EPASContractsMapper; +import it.cnr.isti.epasmed.epas.model.EPASContracts; +import it.cnr.isti.epasmed.epas.model.EPASPersons; +import it.cnr.isti.epasmed.security.AuthoritiesConstants; +import it.cnr.isti.epasmed.web.rest.TestUtil; + +/** + * Integration tests for the {@link EPASContractsResource} REST controller. + */ +@AutoConfigureMockMvc +@WithMockUser(authorities = AuthoritiesConstants.ADMIN) +@SpringBootTest(classes = EpasmedApp.class) +@EnabledIf("false") +public class EPASContractsResourceIT { + + private final Logger log = LoggerFactory.getLogger(getClass()); + + private static final String PERSON_DEFAULT_ID = "1"; + private static final String PERSON_DEFAULT_NAME = "Giancarlo"; + private static final String PERSON_DEFAULT_SURNAME = "Panichi"; + private static final String PERSON_DEFAULT_FISCAL_CODE = "PNCGCR75S04L103G"; + private static final String PERSON_DEFAULT_EMAIL = "giancarlo.panichi@cnr.it"; + private static final String PERSON_DEFAULT_QUALIFICATION = "6"; + + private static final String CONTRACT_DEFAULT_ID = "1"; + private static final String CONTRACT_DEFAULT_BEGINDATE = "2018-12-27"; + private static final String CONTRACT_DEFAUL_ENDDATE = null; + private static final String CONTRACT_DEFAULT_ENDCONTRACT = null; + private static final boolean CONTRACT_DEFAULT_ONCERTIFICATE = true; + private static final String CONTRACT_DEFAULT_PREVIOUSCONTRACT = null; + private static final String CONTRACT_DEFAULT_EXTERNALID="2496"; + + private static final String CONTRACT_UPDATE_ENDATE = "2021-12-31"; + + @Autowired + private MockMvc restEPASContractsMockMvc; + + @Autowired + private EPASContractsMapper epasContractsMapper; + + @Autowired + private Environment environment; + + private EPASContracts epasContracts; + + private EPASContractsDTO epasContractsDTO; + + /** + * Create a EPAS Contracts. + * + * This is a static method + */ + public static EPASContracts createEPASContracts() { + EPASContracts epasContracts = new EPASContracts(); + epasContracts.setBeginDate(CONTRACT_DEFAULT_BEGINDATE); + epasContracts.setEndDate(CONTRACT_DEFAUL_ENDDATE); + epasContracts.setEndContract(CONTRACT_DEFAULT_ENDCONTRACT); + epasContracts.setOnCertificate(CONTRACT_DEFAULT_ONCERTIFICATE); + epasContracts.setPerson(createEPASPerson()); + epasContracts.setPreviousContract(CONTRACT_DEFAULT_PREVIOUSCONTRACT); + epasContracts.setExternalId(CONTRACT_DEFAULT_EXTERNALID); + return epasContracts; + } + + /** + * Create a EPAS Persons. + * + * This is a static method + */ + public static EPASPersons createEPASPerson() { + EPASPersons epasPersons = new EPASPersons(); + epasPersons.setId(PERSON_DEFAULT_ID); + epasPersons.setName(PERSON_DEFAULT_NAME); + epasPersons.setSurname(PERSON_DEFAULT_SURNAME); + epasPersons.setFiscalCode(PERSON_DEFAULT_FISCAL_CODE); + epasPersons.setEmail(PERSON_DEFAULT_EMAIL); + epasPersons.setQualification(PERSON_DEFAULT_QUALIFICATION); + return epasPersons; + } + + @BeforeEach + public void initTest() { + for (String profileName : environment.getActiveProfiles()) { + log.info("Currently active profile - " + profileName); + } + epasContracts = createEPASContracts(); + epasContractsDTO = epasContractsMapper.epasContractsToEPASContractsDTO(epasContracts); + } + + @Test + public void getContract() throws Exception { + restEPASContractsMockMvc.perform(get("/api/epas/contracts/" + CONTRACT_DEFAULT_ID)).andExpect(status().isOk()); + } + + @Test + public void getContractByPersonId() throws Exception { + restEPASContractsMockMvc.perform(get("/api/epas/contracts/byPerson?id=" + PERSON_DEFAULT_ID)) + .andExpect(status().isOk()); + } + + @Test + public void getContractByPersonFiscalcode() throws Exception { + restEPASContractsMockMvc.perform(get("/api/epas/contracts/byPerson?fiscalCode=" + PERSON_DEFAULT_FISCAL_CODE)) + .andExpect(status().isOk()); + } + + @Test + public void createContract() throws Exception { + restEPASContractsMockMvc + .perform(post("/api/epas/contracts").contentType(MediaType.APPLICATION_JSON) + .content(TestUtil.convertObjectToJsonBytes(epasContractsDTO)).with(csrf())) + .andExpect(status().isCreated()); + } + + @Test + public void updateContractById() throws Exception { + epasContractsDTO.setEndDate(CONTRACT_UPDATE_ENDATE); + MvcResult mvcResult = restEPASContractsMockMvc + .perform(put("/api/epas/contracts/" + CONTRACT_DEFAULT_ID).contentType(MediaType.APPLICATION_JSON) + .content(TestUtil.convertObjectToJsonBytes(epasContractsDTO)).with(csrf())) + .andExpect(status().isOk()).andReturn(); + ObjectMapper mapper = new ObjectMapper(); + + EPASContracts epasContracts = mapper.readValue(mvcResult.getResponse().getContentAsByteArray(), + EPASContracts.class); + log.info("Updated EPAS Contract by id: " + epasContracts); + } + + @Test + public void setEPASContractPrevious() throws Exception { + restEPASContractsMockMvc + .perform(delete("/api/epas/contracts/setPreviousContract?id=" + CONTRACT_DEFAULT_ID).with(csrf())) + .andExpect(status().is2xxSuccessful()); + log.info("Set to Previous EPAS Contract by id: " + CONTRACT_DEFAULT_ID); + } + + @Test + public void setEPASContractUnsetPrevious() throws Exception { + restEPASContractsMockMvc + .perform(delete("/api/epas/contracts/unsetPreviousContract?id=" + CONTRACT_DEFAULT_ID).with(csrf())) + .andExpect(status().is2xxSuccessful()); + log.info("Set to Previous EPAS Contract by id: " + CONTRACT_DEFAULT_ID); + } + + @Test + public void deleteContractById() throws Exception { + restEPASContractsMockMvc.perform(delete("/api/epas/contracts/" + CONTRACT_DEFAULT_ID).with(csrf())) + .andExpect(status().is2xxSuccessful()); + log.info("Deleted EPAS Contract by id: " + CONTRACT_DEFAULT_ID); + } + + @Test + public void istiUpdateContract() throws Exception { + String userDirectory = System.getProperty("user.dir"); + log.info(userDirectory); + List istiContracts = null; + try (Stream stream = Files + .lines(Paths.get("src/test/resources/it/cnr/isti/epasmed/web/rest/epas/posizioniISTI.csv"))) { + istiContracts = stream.skip(1).map(istiMapToContract).collect(Collectors.toList()); + } catch (Exception e) { + log.error(e.getLocalizedMessage(), e); + return; + } + log.info("ISTI Contracts loaded"); + + for (EPASContracts c : istiContracts) { + log.info("Contract: {}", c); + } + + List enrichedISTIContracts = new ArrayList<>(); + for (EPASContracts c : istiContracts) { + if (c.getPerson() != null && c.getPerson().getFiscalCode() != null + && !c.getPerson().getFiscalCode().isEmpty()) { + c.setPerson(retrievePersonByFiscalCode(c.getPerson().getFiscalCode())); + enrichedISTIContracts.add(c); + } + } + + log.info("ISTI enriched contracts with persons"); + + List updatableContracts=new ArrayList<>(); + for (EPASContracts ec : enrichedISTIContracts) { + if (ec.getPerson() != null && ec.getPerson().getFiscalCode() != null + && !ec.getPerson().getFiscalCode().isEmpty()) { + List personContract=retrieveContractsByFiscalCode(ec); + for(EPASContracts persCont:personContract) { + if(persCont.getBeginDate().compareTo(ec.getBeginDate())==0) { + persCont.setExternalId(ec.getExternalId()); + updatableContracts.add(persCont); + break; + } + } + } + + } + + log.info("ISTI Updateble contracts created"); + + List updatableContractsDTO=epasContractsMapper.epasContractsToEPASContractsDTOs(updatableContracts); + + for(EPASContractsDTO u: updatableContractsDTO) { + log.info("Contract Updatable: {}",u); + restEPASContractsMockMvc + .perform(put("/api/epas/contracts/" + u.getId()).contentType(MediaType.APPLICATION_JSON) + .content(TestUtil.convertObjectToJsonBytes(u)).with(csrf())) + .andExpect(status().isOk()); + + } + log.info("ISTI Update contracts end"); + + } + + private List retrieveContractsByFiscalCode(EPASContracts ec) + throws Exception, JsonProcessingException, JsonMappingException, UnsupportedEncodingException { + MvcResult result=restEPASContractsMockMvc + .perform(get("/api/epas/contracts/byPerson?fiscalCode=" + ec.getPerson().getFiscalCode())) + .andExpect(status().isOk()).andReturn(); + ObjectMapper mapper = new ObjectMapper(); + List epasContractsForPerson = mapper.readValue(result.getResponse().getContentAsString(), + new TypeReference>() { + }); + return epasContractsForPerson; + } + + private EPASPersons retrievePersonByFiscalCode(String fiscalCode) throws Exception { + MvcResult result = restEPASContractsMockMvc.perform(get("/api/epas/persons/show?fiscalCode=" + fiscalCode)) + .andExpect(status().isOk()).andReturn(); + + ObjectMapper mapper = new ObjectMapper(); + EPASPersons epasPersons = mapper.readValue(result.getResponse().getContentAsString(), + new TypeReference() { + }); + return epasPersons; + } + + private Function istiMapToContract = (line) -> { + + String[] c = line.split(",");// a CSV has comma separated lines + + EPASContracts contract = new EPASContracts(); + + // <-- this is the first column in the csv file + if (c[0] != null && c[0].length() > 0) { + String externalId = c[0].substring(1, c[0].length() - 1); + if (externalId != null && !externalId.isEmpty()) { + contract.setExternalId(externalId); + } + } + + if (c[1] != null && c[1].length() > 0) { + String beginDate = c[1].substring(1, c[1].length() - 1); + if (beginDate != null && !beginDate.isEmpty()) { + contract.setBeginDate(beginDate); + } + } + + if (c[2] != null && c[2].length() > 0) { + String endDate = c[2].substring(1, c[2].length() - 1); + if (endDate != null && !endDate.isEmpty()) { + contract.setEndDate(endDate); + } + } + + EPASPersons ep = new EPASPersons(); + if (c[4] != null && c[4].length() > 0) { + String fiscalcode = c[4].substring(1, c[4].length() - 1); + if (fiscalcode != null && !fiscalcode.isEmpty()) { + ep.setFiscalCode(fiscalcode); + } + } + + if (c[5] != null && c[5].length() > 0) { + String surname = c[5].substring(1, c[5].length() - 1); + if (surname != null && !surname.isEmpty()) { + ep.setSurname(surname); + } + } + + if (c[6] != null && c[6].length() > 0) { + String name = c[6].substring(1, c[6].length() - 1); + if (name != null && !name.isEmpty()) { + ep.setName(name); + } + } + + contract.setPerson(ep); + + return contract; + }; + +} diff --git a/src/test/java/it/cnr/isti/epasmed/web/rest/epas/EPASGroupsResourceIT.java b/src/test/java/it/cnr/isti/epasmed/web/rest/epas/EPASGroupsResourceIT.java new file mode 100644 index 0000000..8f206ce --- /dev/null +++ b/src/test/java/it/cnr/isti/epasmed/web/rest/epas/EPASGroupsResourceIT.java @@ -0,0 +1,302 @@ +package it.cnr.isti.epasmed.web.rest.epas; + +import static org.springframework.security.test.web.servlet.request.SecurityMockMvcRequestPostProcessors.csrf; +import static org.springframework.test.web.servlet.request.MockMvcRequestBuilders.delete; +import static org.springframework.test.web.servlet.request.MockMvcRequestBuilders.get; +import static org.springframework.test.web.servlet.request.MockMvcRequestBuilders.post; +import static org.springframework.test.web.servlet.request.MockMvcRequestBuilders.put; +import static org.springframework.test.web.servlet.result.MockMvcResultMatchers.status; + +import java.util.ArrayList; +import java.util.List; + +import org.junit.jupiter.api.BeforeEach; +import org.junit.jupiter.api.Test; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.boot.test.autoconfigure.web.servlet.AutoConfigureMockMvc; +import org.springframework.boot.test.context.SpringBootTest; +import org.springframework.core.env.Environment; +import org.springframework.http.MediaType; +import org.springframework.security.test.context.support.WithMockUser; +import org.springframework.test.context.junit.jupiter.EnabledIf; +import org.springframework.test.web.servlet.MockMvc; +import org.springframework.test.web.servlet.MvcResult; + +import com.fasterxml.jackson.core.type.TypeReference; +import com.fasterxml.jackson.databind.ObjectMapper; + +import it.cnr.isti.epasmed.EpasmedApp; +import it.cnr.isti.epasmed.epas.dto.EPASGroupsDTO; +import it.cnr.isti.epasmed.epas.mapper.EPASGroupsMapper; +import it.cnr.isti.epasmed.epas.model.EPASGroups; +import it.cnr.isti.epasmed.epas.model.EPASGroupsManager; +import it.cnr.isti.epasmed.epas.model.EPASOffice; +import it.cnr.isti.epasmed.security.AuthoritiesConstants; +import it.cnr.isti.epasmed.web.rest.TestUtil; + +/** + * Integration tests for the {@link EPASGroupsResource} REST controller. + */ +@AutoConfigureMockMvc +@WithMockUser(authorities = AuthoritiesConstants.ADMIN) +@SpringBootTest(classes = EpasmedApp.class) +@EnabledIf("false") +public class EPASGroupsResourceIT { + + private final Logger log = LoggerFactory.getLogger(getClass()); + + private static final String OFFICE_DEFAULT_ID = "1"; + private static final String OFFICE_DEFAULT_NAME = "ISTI - Pisa"; + private static final String OFFICE_DEFAULT_CODE = "074000"; + private static final String OFFICE_DEFAULT_CODEID = "225200"; + + private static final String MANAGER_DEFAULT_ID = "1"; + private static final String MANAGER_DEFAULT_FULLNAME = "Giancarlo Panichi"; + private static final String MANAGER_DEFAULT_FISCALCODE = "PNCGCRL75S04L103G"; + private static final String MANAGER_DEFAULT_EMAIL = "giancarlo.panichi@cnr.it"; + private static final String MANAGER_DEFULT_EPPN = "giancarlo.panichi@cnr.it"; + private static final String MANAGER_DEFAULT_NUMBER = "15409"; + + private static final String GROUP_DEFAULT_ID = "1"; + private static final String GROUP_DEFAULT_NAME = "GruppoTestRest"; + private static final String GROUP_DEFAULT_DESCRIPTION = "Gruppo di Test Rest"; + private static final String GROUP_DEFAUL_ENDDATE = null; + private static final String GROUP_DEFAULT_EXTERNALID = "1"; + private static final String GROUP_DEFAULT_UPDATEDAT = null; + + private static final String GROUP_UPDATED_DESCRIPTION = "Gruppo di Test Updated"; + + private ArrayList epasGroupsDTOList; + + @Autowired + private MockMvc restEPASGroupsMockMvc; + + @Autowired + private EPASGroupsMapper epasGroupsMapper; + + @Autowired + private Environment environment; + + private EPASGroups epasGroups; + + private EPASGroupsDTO epasGroupsDTO; + + /** + * Create a EPAS Groups. + * + * This is a static method + */ + public static EPASGroups createEPASGroups() { + EPASGroups epasGroups = new EPASGroups(); + epasGroups.setName(GROUP_DEFAULT_NAME); + epasGroups.setDescription(GROUP_DEFAULT_DESCRIPTION); + epasGroups.setEndDate(GROUP_DEFAUL_ENDDATE); + epasGroups.setExternalId(GROUP_DEFAULT_EXTERNALID); + epasGroups.setManager(createManager()); + epasGroups.setUpdatedAt(GROUP_DEFAULT_UPDATEDAT); + epasGroups.setOffice(createOffice()); + return epasGroups; + } + + private static EPASGroupsManager createManager() { + EPASGroupsManager epasGroupsManager = new EPASGroupsManager(); + epasGroupsManager.setId(MANAGER_DEFAULT_ID); + epasGroupsManager.setFullname(MANAGER_DEFAULT_FULLNAME); + epasGroupsManager.setFiscalCode(MANAGER_DEFAULT_FISCALCODE); + epasGroupsManager.setEmail(MANAGER_DEFAULT_EMAIL); + epasGroupsManager.setEppn(MANAGER_DEFULT_EPPN); + epasGroupsManager.setNumber(MANAGER_DEFAULT_NUMBER); + return epasGroupsManager; + } + + public static EPASOffice createOffice() { + EPASOffice epasOffice = new EPASOffice(); + epasOffice.setId(OFFICE_DEFAULT_ID); + epasOffice.setName(OFFICE_DEFAULT_NAME); + epasOffice.setCode(OFFICE_DEFAULT_CODE); + epasOffice.setCodeId(OFFICE_DEFAULT_CODEID); + return epasOffice; + } + + private ArrayList createEPASGroupsDTOList() { + epasGroupsDTOList = new ArrayList<>(); + epasGroupsDTOList.add(new EPASGroupsDTO("AREA", null, "1", null, "1", "AREA", "1")); + epasGroupsDTOList.add(new EPASGroupsDTO("Consiglio di Istituto", null, "2", null, "1", "CI", "1")); + epasGroupsDTOList.add(new EPASGroupsDTO("Dependable Computing", null, "3", null, "1", "DC", "1")); + epasGroupsDTOList + .add(new EPASGroupsDTO("Formal Methods and Tools", null, "4", null, "1", "FMT", "1")); + epasGroupsDTOList + .add(new EPASGroupsDTO("Sistema Informativo", null, "5", null, "1", "GL - SI", "1")); + epasGroupsDTOList.add(new EPASGroupsDTO("Domotics", null, "6", null, "1", "HA", "1")); + epasGroupsDTOList.add(new EPASGroupsDTO("Human Interfaces in Information Systems", null, "7", null, "1", "HIIS", + "1")); + epasGroupsDTOList + .add(new EPASGroupsDTO("High Performance Computing", null, "8", null, "1", "HPC", "1")); + epasGroupsDTOList.add( + new EPASGroupsDTO("Knowledge Discovery and Data Mining", null, "9", null, "1", "KDD", "1")); + epasGroupsDTOList.add(new EPASGroupsDTO("Mechanics of Materials and Structures", null, "10", null, "1", "MMS", + "1")); + epasGroupsDTOList.add(new EPASGroupsDTO("Networked Multimedia Information System", null, "11", null, "1", + "NeMIS", "1")); + epasGroupsDTOList.add(new EPASGroupsDTO("Servizi Internet", null, "12", null, "1", "NS", "1")); + epasGroupsDTOList + .add(new EPASGroupsDTO("Servizio Amministrazione", null, "13", null, "1", "SA", "1")); + epasGroupsDTOList + .add(new EPASGroupsDTO("Servizio Attività Generali", null, "14", null, "1", "SAG", "1")); + epasGroupsDTOList + .add(new EPASGroupsDTO("Servizio Attività Logistiche", null, "15", null, "1", "SAL", "1")); + epasGroupsDTOList.add(new EPASGroupsDTO("Servizio Attività Scientifiche - Progetti e Contratti", null, "16", + null, "1", "SAS - PC", "1")); + epasGroupsDTOList.add(new EPASGroupsDTO("Servizio Attività Scientifiche - Segreteria Scientifica", null, "17", + null, "1", "SAS - SS", "1")); + epasGroupsDTOList.add(new EPASGroupsDTO("Servizio Biblioteca e Centro di Documentazione Scientifica", null, + "18", null, "1", "SB", "1")); + epasGroupsDTOList.add(new EPASGroupsDTO("Software Engineering", null, "19", null, "1", "SE", "1")); + epasGroupsDTOList.add(new EPASGroupsDTO("Software Engineering & Dependable Computing", null, "20", null, "1", + "SEDC", "1")); + epasGroupsDTOList + .add(new EPASGroupsDTO("Space Flight Dynamics", null, "21", null, "1", "SFD", "1")); + epasGroupsDTOList.add(new EPASGroupsDTO("Signals and Images", null, "22", null, "1", "SI", "1")); + epasGroupsDTOList.add( + new EPASGroupsDTO("System and Software Evaluation", null, "23", null, "1", "SSE", "1")); + epasGroupsDTOList.add(new EPASGroupsDTO("Servizio Supporto Informatico e Formazione - Formazione e Sviluppo", + null, "24", null, "1", "SSIF - FS", "1")); + epasGroupsDTOList.add(new EPASGroupsDTO("Servizio Supporto Informatico e Formazione - Supporto Informatico", + null, "25", null, "1", "SSIF - SI", "1")); + epasGroupsDTOList.add(new EPASGroupsDTO("Visual Computing", null, "26", null, "1", "VC", "1")); + epasGroupsDTOList.add(new EPASGroupsDTO("Wireless Networks", null, "27", null, "1", "WN", "1")); + epasGroupsDTOList.add(new EPASGroupsDTO("Sistemi di Rete e Servizi Internet", null, "28", null, "1", "SRSI", + "1")); + epasGroupsDTOList.add(new EPASGroupsDTO("Servizio Attività Scientifiche - Gestione Privacy", null, "29", null, + "1", "SAS - GP", "1")); + epasGroupsDTOList.add(new EPASGroupsDTO("Infrastruttura Informatica D4Science", null, "30", null, "1", + "GL - D4Science", "1")); + epasGroupsDTOList.add(new EPASGroupsDTO("Servizio Infrastruttura Informatica ISTI e Supporto ai Servizi", null, + "31", null, "1", "S2I2S", "1")); + epasGroupsDTOList.add(new EPASGroupsDTO("Artificial Intelligence for Media and Humanities", null, "32", null, + "1", "AIMH", "1")); + epasGroupsDTOList.add(new EPASGroupsDTO("Infrastructures for Science", null, "33", null, "1", "InfraScience", + "1")); + return epasGroupsDTOList; + } + + @BeforeEach + public void initTest() { + for (String profileName : environment.getActiveProfiles()) { + log.info("Currently active profile - " + profileName); + } + epasGroups = createEPASGroups(); + epasGroupsDTO = epasGroupsMapper.epasGroupsToEPASGroupsDTO(epasGroups); + epasGroupsDTOList = createEPASGroupsDTOList(); + } + + @Test + public void getGroup() throws Exception { + restEPASGroupsMockMvc.perform(get("/api/epas/groups/" + 9)).andExpect(status().isOk()); + } + + @Test + public void getGroupList() throws Exception { + restEPASGroupsMockMvc.perform(get("/api/epas/groups?officeId=" + OFFICE_DEFAULT_ID)).andExpect(status().isOk()); + } + + @Test + public void createGroup() throws Exception { + restEPASGroupsMockMvc + .perform(post("/api/epas/groups").contentType(MediaType.APPLICATION_JSON) + .content(TestUtil.convertObjectToJsonBytes(epasGroupsDTO)).with(csrf())) + .andExpect(status().isCreated()); + } + + @Test + public void istiCreateGroups() throws Exception { + for (EPASGroupsDTO istiGroup : epasGroupsDTOList) { + restEPASGroupsMockMvc + .perform(post("/api/epas/groups").contentType(MediaType.APPLICATION_JSON) + .content(TestUtil.convertObjectToJsonBytes(istiGroup)).with(csrf())) + .andExpect(status().isCreated()); + } + } + + @Test + public void istiUpdateGroups() throws Exception { + MvcResult result = restEPASGroupsMockMvc.perform(get("/api/epas/groups?officeId=" + OFFICE_DEFAULT_ID)) + .andExpect(status().isOk()).andReturn(); + ObjectMapper mapper = new ObjectMapper(); + List epasG = mapper.readValue(result.getResponse().getContentAsString(), + new TypeReference>() { + }); + + log.info("EPASGroups: {}", epasG.size()); + + List epasGDTO = epasGroupsMapper.epasGroupsToEPASGroupsDTOs(epasG); + + istiEnrichEPASGroupData(epasGDTO); + + for (EPASGroupsDTO g : epasGDTO) { + log.info("Updated EPASGroup: {}", g); + } + + for (EPASGroupsDTO g : epasGDTO) { + restEPASGroupsMockMvc.perform(put("/api/epas/groups/" + g.getId()).contentType(MediaType.APPLICATION_JSON) + .content(TestUtil.convertObjectToJsonBytes(g)).with(csrf())).andExpect(status().isOk()); + } + log.info("EPASGroups ISTI Updated"); + + } + + private void istiEnrichEPASGroupData(List epasGDTO) { + for (EPASGroupsDTO g : epasGDTO) { + for (EPASGroupsDTO m : epasGroupsDTOList) { + if (m.getName().compareTo(g.getName()) == 0) { + g.setExternalId(m.getExternalId()); + g.setOfficeId("1"); + break; + } + } + } + + } + + @Test + public void getGroupByExternalId() throws Exception{ + String externalId="29"; + MvcResult mvcResult=restEPASGroupsMockMvc.perform(get("/api/epas/groups?officeId=" + OFFICE_DEFAULT_ID)).andExpect(status().isOk()).andReturn(); + ObjectMapper mapper = new ObjectMapper(); + + List epasGroups = mapper.readValue(mvcResult.getResponse().getContentAsString(), + new TypeReference>() { + }); + for(EPASGroups g:epasGroups) { + if(g.getExternalId()!=null&&!g.getExternalId().isEmpty()&& + g.getExternalId().compareTo(externalId)==0) { + log.info("Found Group: {}",g); + break; + } + } + + } + + @Test + public void updateGroupById() throws Exception { + epasGroupsDTO.setDescription(GROUP_UPDATED_DESCRIPTION); + MvcResult mvcResult = restEPASGroupsMockMvc + .perform(put("/api/epas/groups/" + GROUP_DEFAULT_ID).contentType(MediaType.APPLICATION_JSON) + .content(TestUtil.convertObjectToJsonBytes(epasGroupsDTO)).with(csrf())) + .andExpect(status().isOk()).andReturn(); + ObjectMapper mapper = new ObjectMapper(); + + EPASGroups epasGroups = mapper.readValue(mvcResult.getResponse().getContentAsByteArray(), EPASGroups.class); + log.info("Updated EPAS Group by id: " + epasGroups); + } + + @Test + public void deleteGroupById() throws Exception { + restEPASGroupsMockMvc.perform(delete("/api/epas/groups/" + GROUP_DEFAULT_ID).with(csrf())) + .andExpect(status().is2xxSuccessful()); + log.info("Deleted EPAS Group by id: " + GROUP_DEFAULT_ID); + } + +} diff --git a/src/test/java/it/cnr/isti/epasmed/web/rest/epas/EPASPersonsResourceIT.java b/src/test/java/it/cnr/isti/epasmed/web/rest/epas/EPASPersonsResourceIT.java new file mode 100644 index 0000000..bf47965 --- /dev/null +++ b/src/test/java/it/cnr/isti/epasmed/web/rest/epas/EPASPersonsResourceIT.java @@ -0,0 +1,299 @@ +package it.cnr.isti.epasmed.web.rest.epas; + +import static org.springframework.security.test.web.servlet.request.SecurityMockMvcRequestPostProcessors.csrf; +import static org.springframework.test.web.servlet.request.MockMvcRequestBuilders.delete; +import static org.springframework.test.web.servlet.request.MockMvcRequestBuilders.get; +import static org.springframework.test.web.servlet.request.MockMvcRequestBuilders.post; +import static org.springframework.test.web.servlet.request.MockMvcRequestBuilders.put; +import static org.springframework.test.web.servlet.result.MockMvcResultMatchers.status; + +import java.nio.file.Files; +import java.nio.file.Paths; +import java.util.ArrayList; +import java.util.List; +import java.util.function.Function; +import java.util.stream.Collectors; +import java.util.stream.Stream; + +import org.junit.jupiter.api.BeforeEach; +import org.junit.jupiter.api.Test; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.boot.test.autoconfigure.web.servlet.AutoConfigureMockMvc; +import org.springframework.boot.test.context.SpringBootTest; +import org.springframework.core.env.Environment; +import org.springframework.http.MediaType; +import org.springframework.security.test.context.support.WithMockUser; +import org.springframework.test.context.junit.jupiter.EnabledIf; +import org.springframework.test.web.servlet.MockMvc; +import org.springframework.test.web.servlet.MvcResult; + +import com.fasterxml.jackson.core.type.TypeReference; +import com.fasterxml.jackson.databind.ObjectMapper; + +import it.cnr.isti.epasmed.EpasmedApp; +import it.cnr.isti.epasmed.epas.dto.EPASPersonsDTO; +import it.cnr.isti.epasmed.epas.mapper.EPASPersonsMapper; +import it.cnr.isti.epasmed.epas.model.EPASOffice; +import it.cnr.isti.epasmed.epas.model.EPASPersons; +import it.cnr.isti.epasmed.security.AuthoritiesConstants; +import it.cnr.isti.epasmed.web.rest.TestUtil; + +/** + * Integration tests for the {@link EPASPersonsResource} REST controller. + */ +@AutoConfigureMockMvc +@WithMockUser(authorities = AuthoritiesConstants.ADMIN) +@SpringBootTest(classes = EpasmedApp.class) +@EnabledIf("false") +public class EPASPersonsResourceIT { + + private final Logger log = LoggerFactory.getLogger(getClass()); + + private static final String OFFICE_DEFAULT_ID = "1"; + private static final String OFFICE_DEFAULT_NAME = "ISTI - Pisa"; + private static final String OFFICE_DEFAULT_CODE = "074000"; + private static final String OFFICE_DEFAULT_CODEID = "225200"; + + private static final String PERSON_DEFAULT_ID = "176"; + private static final String PERSON_DEFAULT_NAME = "nometest1"; + private static final String PERSON_DEFAULT_SURNAME = "cognometest1"; + private static final String PERSON_DEFAULT_FISCAL_CODE = "codicefiscaletes"; + private static final String PERSON_DEFAULT_EMAIL = "nometest1.cognometest1@test.it"; + private static final String PERSON_DEFAULT_QUALIFICATION = "3"; + + private static final String PERSON_UPDATE_EMAIL = "nometest1.cognometest1@updated.it"; + private static final String PERSON_UPDATE_2_EMAIL = "nometest1.cognometest1@updated2.it"; + + @Autowired + private MockMvc restEPASPersonsMockMvc; + + @Autowired + private EPASPersonsMapper epasPersonsMapper; + + @Autowired + private Environment environment; + + private EPASPersons epasPersons; + + private EPASPersonsDTO epasPersonsDTO; + + /** + * Create a EPAS Persons. + * + * This is a static method + */ + public static EPASPersons createEPASPerson() { + EPASPersons epasPersons = new EPASPersons(); + epasPersons.setName(PERSON_DEFAULT_NAME); + epasPersons.setSurname(PERSON_DEFAULT_SURNAME); + epasPersons.setFiscalCode(PERSON_DEFAULT_FISCAL_CODE); + epasPersons.setEmail(PERSON_DEFAULT_EMAIL); + epasPersons.setQualification(PERSON_DEFAULT_QUALIFICATION); + epasPersons.setOffice(createOffice()); + return epasPersons; + } + + public static EPASOffice createOffice() { + EPASOffice epasOffice = new EPASOffice(); + epasOffice.setId(OFFICE_DEFAULT_ID); + epasOffice.setName(OFFICE_DEFAULT_NAME); + epasOffice.setCode(OFFICE_DEFAULT_CODE); + epasOffice.setCodeId(OFFICE_DEFAULT_CODEID); + return epasOffice; + } + + @BeforeEach + public void initTest() { + for (String profileName : environment.getActiveProfiles()) { + log.info("Currently active profile - " + profileName); + } + log.info("System env - " + System.getenv("spring.profiles.active")); + epasPersons = createEPASPerson(); + epasPersonsDTO = epasPersonsMapper.epasPersonsToEPASPersonsDTO(epasPersons); + } + + @Test + public void getPersonById() throws Exception { + restEPASPersonsMockMvc.perform(get("/api/epas/persons/72")).andExpect(status().isOk()); + } + + @Test + public void getPersonByFiscalCode() throws Exception { + restEPASPersonsMockMvc.perform(get("/api/epas/persons/show?fiscalCode=PRIGRL74E31G491R")).andExpect(status().isOk()); + } + + @Test + public void getPersonList() throws Exception { + restEPASPersonsMockMvc.perform(get("/api/epas/persons?officeId=" + OFFICE_DEFAULT_ID)) + .andExpect(status().isOk()); + } + + @Test + public void createPerson() throws Exception { + restEPASPersonsMockMvc + .perform(post("/api/epas/persons").contentType(MediaType.APPLICATION_JSON) + .content(TestUtil.convertObjectToJsonBytes(epasPersonsDTO)).with(csrf())) + .andExpect(status().isCreated()); + + } + + @Test + public void istiUpdatedPerson() { + String userDirectory = System.getProperty("user.dir"); + log.info(userDirectory); + List istiPersons = null; + try (Stream stream = Files + .lines(Paths.get("src/test/resources/it/cnr/isti/epasmed/web/rest/epas/personale.csv"))) { + istiPersons = stream.skip(1).map(istiMapToPerson).collect(Collectors.toList()); + } catch (Exception e) { + log.error(e.getLocalizedMessage(), e); + return; + } + log.info("ISTI Persons loaded"); + + List epasPersons = null; + try { + MvcResult result = restEPASPersonsMockMvc.perform(get("/api/epas/persons?officeId=" + OFFICE_DEFAULT_ID)) + .andExpect(status().isOk()).andReturn(); + + ObjectMapper mapper = new ObjectMapper(); + epasPersons = mapper.readValue(result.getResponse().getContentAsString(), + new TypeReference>() { + }); + + } catch (Exception e) { + log.error(e.getLocalizedMessage(), e); + return; + } + + log.info("EPAS Persons loaded"); + + List updateList = new ArrayList<>(); + + for (EPASPersons ep : epasPersons) { + if (ep.getNumber() != null && !ep.getNumber().isEmpty()) { + String number = ep.getNumber(); + boolean foundMatch = false; + EPASPersonsDTO istiPers = null; + for (EPASPersonsDTO istip : istiPersons) { + if (istip.getNumber() != null && !istip.getNumber().isEmpty() + && istip.getNumber().compareTo(number) == 0) { + foundMatch = true; + istiPers = istip; + break; + } + } + if (foundMatch) { + ep.setFiscalCode(istiPers.getFiscalCode()); + ep.setTelephone(istiPers.getTelephone()); + updateList.add(ep); + } + } + } + + log.info("UpdateList created"); + + List updateDTOList = epasPersonsMapper.epasPersonsToEPASPersonsDTOs(updateList); + + for (EPASPersonsDTO up : updateDTOList) { + try { + restEPASPersonsMockMvc + .perform(put("/api/epas/persons/" + up.getId()).contentType(MediaType.APPLICATION_JSON) + .content(TestUtil.convertObjectToJsonBytes(up)).with(csrf())) + .andExpect(status().isOk()); + } catch (Exception e) { + log.error(e.getLocalizedMessage(), e); + } + } + } + + private Function istiMapToPerson = (line) -> { + + String[] p = line.split(",");// a CSV has comma separated lines + + EPASPersonsDTO pers = new EPASPersonsDTO(); + + // <-- this is the first column in the csv file + if (p[1] != null && p[1].length() > 0) { + String surname = p[1].substring(1, p[1].length() - 1); + if (surname != null && !surname.isEmpty()) { + pers.setSurname(surname); + } + } + + if (p[2] != null && p[2].length() > 0) { + String name = p[2].substring(1, p[2].length() - 1); + if (name != null && !name.isEmpty()) { + pers.setName(name); + } + } + + if (p[3] != null && p[3].length() > 0) { + String matricola = p[3]; + if (matricola != null && !matricola.isEmpty()) { + pers.setNumber(matricola); + } + } + + if (p[4] != null && p[4].length() > 0) { + String fiscalCode = p[4].substring(1, p[4].length() - 1); + if (fiscalCode != null && !fiscalCode.isEmpty()) { + pers.setFiscalCode(fiscalCode); + } + } + + if (p[5] != null && p[5].length() > 0) { + String telephone = p[5].substring(1, p[5].length() - 1); + if (telephone != null && !telephone.isEmpty()) { + pers.setTelephone(telephone); + } + } + + return pers; + }; + + @Test + public void updatePersonById() throws Exception { + epasPersonsDTO.setEmail(PERSON_UPDATE_EMAIL); + MvcResult mvcResult = restEPASPersonsMockMvc + .perform(put("/api/epas/persons/" + PERSON_DEFAULT_ID).contentType(MediaType.APPLICATION_JSON) + .content(TestUtil.convertObjectToJsonBytes(epasPersonsDTO)).with(csrf())) + .andExpect(status().isOk()).andReturn(); + ObjectMapper mapper = new ObjectMapper(); + + EPASPersons epasPersons = mapper.readValue(mvcResult.getResponse().getContentAsByteArray(), EPASPersons.class); + log.info("Updated EPAS Person by id: " + epasPersons); + } + + @Test + public void updatePersonByFiscalCode() throws Exception { + epasPersonsDTO.setEmail(PERSON_UPDATE_2_EMAIL); + MvcResult mvcResult = restEPASPersonsMockMvc + .perform(put("/api/epas/persons/update?fiscalCode=" + PERSON_DEFAULT_FISCAL_CODE) + .contentType(MediaType.APPLICATION_JSON) + .content(TestUtil.convertObjectToJsonBytes(epasPersonsDTO)).with(csrf())) + .andExpect(status().isOk()).andReturn(); + ObjectMapper mapper = new ObjectMapper(); + + EPASPersons epasPersons = mapper.readValue(mvcResult.getResponse().getContentAsByteArray(), EPASPersons.class); + log.info("Updated EPAS Person by fiscalcode: " + epasPersons); + } + + @Test + public void deletePersonById() throws Exception { + restEPASPersonsMockMvc.perform(delete("/api/epas/persons/" + PERSON_DEFAULT_ID).with(csrf())) + .andExpect(status().is2xxSuccessful()).andReturn(); + log.info("Deleted EPAS Person by id: " + PERSON_DEFAULT_ID); + } + + @Test + public void deletePersonByFiscalCode() throws Exception { + restEPASPersonsMockMvc + .perform(delete("/api/epas/persons/delete?fiscalCode=" + PERSON_DEFAULT_FISCAL_CODE).with(csrf())) + .andExpect(status().is2xxSuccessful()); + log.info("Deleted EPAS Person by id: " + PERSON_DEFAULT_ID); + } + +} diff --git a/src/test/java/it/cnr/isti/epasmed/web/rest/epas/EPASWorkingTimeTypesResourceIT.java b/src/test/java/it/cnr/isti/epasmed/web/rest/epas/EPASWorkingTimeTypesResourceIT.java new file mode 100644 index 0000000..98b2050 --- /dev/null +++ b/src/test/java/it/cnr/isti/epasmed/web/rest/epas/EPASWorkingTimeTypesResourceIT.java @@ -0,0 +1,91 @@ +package it.cnr.isti.epasmed.web.rest.epas; + +import static org.springframework.test.web.servlet.request.MockMvcRequestBuilders.get; +import static org.springframework.test.web.servlet.result.MockMvcResultMatchers.status; + +import java.util.List; + +import org.junit.jupiter.api.BeforeEach; +import org.junit.jupiter.api.Test; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.boot.test.autoconfigure.web.servlet.AutoConfigureMockMvc; +import org.springframework.boot.test.context.SpringBootTest; +import org.springframework.core.env.Environment; +import org.springframework.security.test.context.support.WithMockUser; +import org.springframework.test.context.junit.jupiter.EnabledIf; +import org.springframework.test.web.servlet.MockMvc; +import org.springframework.test.web.servlet.MvcResult; + +import com.fasterxml.jackson.core.type.TypeReference; +import com.fasterxml.jackson.databind.ObjectMapper; + +import it.cnr.isti.epasmed.EpasmedApp; +import it.cnr.isti.epasmed.epas.model.EPASWorkingTimeTypeDays; +import it.cnr.isti.epasmed.epas.model.EPASWorkingTimeTypes; +import it.cnr.isti.epasmed.security.AuthoritiesConstants; + +/** + * Integration tests for the {@link EPASWorkingTimeTypesResource} REST + * controller. + */ +@AutoConfigureMockMvc +@WithMockUser(authorities = AuthoritiesConstants.ADMIN) +@SpringBootTest(classes = EpasmedApp.class) +@EnabledIf("false") +public class EPASWorkingTimeTypesResourceIT { + + private final Logger log = LoggerFactory.getLogger(getClass()); + + private static final String OFFICE_DEFAULT_ID = "1"; + // private static final String OFFICE_DEFAULT_NAME = "ISTI - Pisa"; + // private static final String OFFICE_DEFAULT_CODE = "074000"; + // private static final String OFFICE_DEFAULT_CODEID = "225200"; + + @Autowired + private MockMvc restEPASWorkingTimeTypesMockMvc; + + @Autowired + private Environment environment; + + @BeforeEach + public void initTest() { + for (String profileName : environment.getActiveProfiles()) { + log.info("Currently active profile - " + profileName); + } + } + + @Test + public void getWorkingTimeTypesById() throws Exception { + MvcResult result = restEPASWorkingTimeTypesMockMvc.perform(get("/api/epas/workingtimetypes/" + 1)) + .andExpect(status().isOk()).andReturn(); + ObjectMapper mapper = new ObjectMapper(); + EPASWorkingTimeTypes epasWTT = mapper.readValue(result.getResponse().getContentAsString(), + new TypeReference() { + }); + log.info("EPASWorkingTimeTypes: {}", epasWTT.getDescription()); + for (EPASWorkingTimeTypeDays d : epasWTT.getWorkingTimeTypeDays()) { + log.info("EPAS WorkingTimeTypeDay: {}", d); + } + } + + @Test + public void getWorkingTimeTypesList() throws Exception { + MvcResult result = restEPASWorkingTimeTypesMockMvc + .perform(get("/api/epas/workingtimetypes?officeId=" + OFFICE_DEFAULT_ID)).andExpect(status().isOk()) + .andReturn(); + + ObjectMapper mapper = new ObjectMapper(); + List epasWTTList = mapper.readValue(result.getResponse().getContentAsString(), + new TypeReference>() { + }); + + log.info("EPASWorkingTimeTypes: {}", epasWTTList.size()); + for (EPASWorkingTimeTypes o : epasWTTList) { + log.info("EPAS WorkingTimeType: {}", o); + } + + } + +} diff --git a/src/test/java/it/cnr/isti/epasmed/web/rest/errors/ExceptionTranslatorIT.java b/src/test/java/it/cnr/isti/epasmed/web/rest/errors/ExceptionTranslatorIT.java new file mode 100644 index 0000000..f5b35a7 --- /dev/null +++ b/src/test/java/it/cnr/isti/epasmed/web/rest/errors/ExceptionTranslatorIT.java @@ -0,0 +1,111 @@ +package it.cnr.isti.epasmed.web.rest.errors; + +import it.cnr.isti.epasmed.EpasmedApp; +import org.junit.jupiter.api.Test; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.boot.test.context.SpringBootTest; +import org.springframework.security.test.context.support.WithMockUser; +import org.springframework.boot.test.autoconfigure.web.servlet.AutoConfigureMockMvc; +import org.springframework.http.MediaType; +import org.springframework.test.web.servlet.MockMvc; + +import static org.springframework.security.test.web.servlet.request.SecurityMockMvcRequestPostProcessors.csrf; +import static org.springframework.test.web.servlet.request.MockMvcRequestBuilders.get; +import static org.springframework.test.web.servlet.request.MockMvcRequestBuilders.post; +import static org.springframework.test.web.servlet.result.MockMvcResultMatchers.content; +import static org.springframework.test.web.servlet.result.MockMvcResultMatchers.jsonPath; +import static org.springframework.test.web.servlet.result.MockMvcResultMatchers.status; + +/** + * Integration tests {@link ExceptionTranslator} controller advice. + */ +@WithMockUser +@AutoConfigureMockMvc +@SpringBootTest(classes = EpasmedApp.class) +public class ExceptionTranslatorIT { + + @Autowired + private MockMvc mockMvc; + + @Test + public void testConcurrencyFailure() throws Exception { + mockMvc.perform(get("/api/exception-translator-test/concurrency-failure").with(csrf())) + .andExpect(status().isConflict()) + .andExpect(content().contentType(MediaType.APPLICATION_PROBLEM_JSON)) + .andExpect(jsonPath("$.message").value(ErrorConstants.ERR_CONCURRENCY_FAILURE)); + } + + @Test + public void testMethodArgumentNotValid() throws Exception { + mockMvc.perform(post("/api/exception-translator-test/method-argument").content("{}").contentType(MediaType.APPLICATION_JSON).with(csrf())) + .andExpect(status().isBadRequest()) + .andExpect(content().contentType(MediaType.APPLICATION_PROBLEM_JSON)) + .andExpect(jsonPath("$.message").value(ErrorConstants.ERR_VALIDATION)) + .andExpect(jsonPath("$.fieldErrors.[0].objectName").value("test")) + .andExpect(jsonPath("$.fieldErrors.[0].field").value("test")) + .andExpect(jsonPath("$.fieldErrors.[0].message").value("NotNull")); + } + + @Test + public void testMissingServletRequestPartException() throws Exception { + mockMvc.perform(get("/api/exception-translator-test/missing-servlet-request-part").with(csrf())) + .andExpect(status().isBadRequest()) + .andExpect(content().contentType(MediaType.APPLICATION_PROBLEM_JSON)) + .andExpect(jsonPath("$.message").value("error.http.400")); + } + + @Test + public void testMissingServletRequestParameterException() throws Exception { + mockMvc.perform(get("/api/exception-translator-test/missing-servlet-request-parameter").with(csrf())) + .andExpect(status().isBadRequest()) + .andExpect(content().contentType(MediaType.APPLICATION_PROBLEM_JSON)) + .andExpect(jsonPath("$.message").value("error.http.400")); + } + + @Test + public void testAccessDenied() throws Exception { + mockMvc.perform(get("/api/exception-translator-test/access-denied").with(csrf())) + .andExpect(status().isForbidden()) + .andExpect(content().contentType(MediaType.APPLICATION_PROBLEM_JSON)) + .andExpect(jsonPath("$.message").value("error.http.403")) + .andExpect(jsonPath("$.detail").value("test access denied!")); + } + + @Test + public void testUnauthorized() throws Exception { + mockMvc.perform(get("/api/exception-translator-test/unauthorized").with(csrf())) + .andExpect(status().isUnauthorized()) + .andExpect(content().contentType(MediaType.APPLICATION_PROBLEM_JSON)) + .andExpect(jsonPath("$.message").value("error.http.401")) + .andExpect(jsonPath("$.path").value("/api/exception-translator-test/unauthorized")) + .andExpect(jsonPath("$.detail").value("test authentication failed!")); + } + + @Test + public void testMethodNotSupported() throws Exception { + mockMvc.perform(post("/api/exception-translator-test/access-denied").with(csrf())) + .andExpect(status().isMethodNotAllowed()) + .andExpect(content().contentType(MediaType.APPLICATION_PROBLEM_JSON)) + .andExpect(jsonPath("$.message").value("error.http.405")) + .andExpect(jsonPath("$.detail").value("Request method 'POST' not supported")); + } + + @Test + public void testExceptionWithResponseStatus() throws Exception { + mockMvc.perform(get("/api/exception-translator-test/response-status").with(csrf())) + .andExpect(status().isBadRequest()) + .andExpect(content().contentType(MediaType.APPLICATION_PROBLEM_JSON)) + .andExpect(jsonPath("$.message").value("error.http.400")) + .andExpect(jsonPath("$.title").value("test response status")); + } + + @Test + public void testInternalServerError() throws Exception { + mockMvc.perform(get("/api/exception-translator-test/internal-server-error").with(csrf())) + .andExpect(status().isInternalServerError()) + .andExpect(content().contentType(MediaType.APPLICATION_PROBLEM_JSON)) + .andExpect(jsonPath("$.message").value("error.http.500")) + .andExpect(jsonPath("$.title").value("Internal Server Error")); + } + +} diff --git a/src/test/java/it/cnr/isti/epasmed/web/rest/errors/ExceptionTranslatorTestController.java b/src/test/java/it/cnr/isti/epasmed/web/rest/errors/ExceptionTranslatorTestController.java new file mode 100644 index 0000000..2651331 --- /dev/null +++ b/src/test/java/it/cnr/isti/epasmed/web/rest/errors/ExceptionTranslatorTestController.java @@ -0,0 +1,72 @@ +package it.cnr.isti.epasmed.web.rest.errors; + +import org.springframework.dao.ConcurrencyFailureException; +import org.springframework.http.HttpStatus; +import org.springframework.security.access.AccessDeniedException; +import org.springframework.security.authentication.BadCredentialsException; +import org.springframework.web.bind.annotation.*; + +import javax.validation.Valid; +import javax.validation.constraints.NotNull; + +@RestController +@RequestMapping("/api/exception-translator-test") +public class ExceptionTranslatorTestController { + + @GetMapping("/concurrency-failure") + public void concurrencyFailure() { + throw new ConcurrencyFailureException("test concurrency failure"); + } + + @PostMapping("/method-argument") + public void methodArgument(@Valid @RequestBody TestDTO testDTO) { + } + + @GetMapping("/missing-servlet-request-part") + public void missingServletRequestPartException(@RequestPart String part) { + } + + @GetMapping("/missing-servlet-request-parameter") + public void missingServletRequestParameterException(@RequestParam String param) { + } + + @GetMapping("/access-denied") + public void accessdenied() { + throw new AccessDeniedException("test access denied!"); + } + + @GetMapping("/unauthorized") + public void unauthorized() { + throw new BadCredentialsException("test authentication failed!"); + } + + @GetMapping("/response-status") + public void exceptionWithResponseStatus() { + throw new TestResponseStatusException(); + } + + @GetMapping("/internal-server-error") + public void internalServerError() { + throw new RuntimeException(); + } + + public static class TestDTO { + + @NotNull + private String test; + + public String getTest() { + return test; + } + + public void setTest(String test) { + this.test = test; + } + } + + @ResponseStatus(value = HttpStatus.BAD_REQUEST, reason = "test response status") + @SuppressWarnings("serial") + public static class TestResponseStatusException extends RuntimeException { + } + +} diff --git a/src/test/javascript/jest-global-mocks.ts b/src/test/javascript/jest-global-mocks.ts new file mode 100644 index 0000000..9159843 --- /dev/null +++ b/src/test/javascript/jest-global-mocks.ts @@ -0,0 +1,3 @@ +Object.defineProperty(window, 'getComputedStyle', { + value: () => ['-webkit-appearance'], +}); diff --git a/src/test/javascript/jest.conf.js b/src/test/javascript/jest.conf.js new file mode 100644 index 0000000..1654531 --- /dev/null +++ b/src/test/javascript/jest.conf.js @@ -0,0 +1,58 @@ +const tsconfig = require('../../../tsconfig.base.json'); + +module.exports = { + preset: 'jest-preset-angular', + setupFiles: ['jest-date-mock'], + setupFilesAfterEnv: ['/src/test/javascript/jest.ts'], + cacheDirectory: '/target/jest-cache', + coverageDirectory: '/target/test-results/', + globals: { + 'ts-jest': { + stringifyContentPathRegex: '\\.html$', + tsConfig: '/tsconfig.base.json', + astTransformers: ['jest-preset-angular/build/InlineFilesTransformer', 'jest-preset-angular/build/StripStylesTransformer'] + } + }, + coveragePathIgnorePatterns: [ + '/src/test/javascript' + ], + moduleNameMapper: mapTypescriptAliasToJestAlias(), + reporters: [ + 'default', + [ 'jest-junit', { outputDirectory: './target/test-results/', outputName: 'TESTS-results-jest.xml' } ] + ], + testResultsProcessor: 'jest-sonar-reporter', + transformIgnorePatterns: ['node_modules/'], + testMatch: ['/src/test/javascript/spec/**/@(*.)@(spec.ts)'], + rootDir: '../../../', + testURL: 'http://localhost/' +}; + +function mapTypescriptAliasToJestAlias(alias = {}) { + const jestAliases = { ...alias }; + if (!tsconfig.compilerOptions.paths) { + return jestAliases; + } + Object.entries(tsconfig.compilerOptions.paths) + .filter(([key, value]) => { + // use Typescript alias in Jest only if this has value + if (value.length) { + return true; + } + return false; + }) + .map(([key, value]) => { + // if Typescript alias ends with /* then in Jest: + // - alias key must end with /(.*) + // - alias value must end with /$1 + const regexToReplace = /(.*)\/\*$/; + const aliasKey = key.replace(regexToReplace, '$1/(.*)'); + const aliasValue = value[0].replace(regexToReplace, '$1/$$1'); + return [aliasKey, `/${aliasValue}`]; + }) + .reduce((aliases, [key, value]) => { + aliases[key] = value; + return aliases; + }, jestAliases); + return jestAliases; +} diff --git a/src/test/javascript/jest.ts b/src/test/javascript/jest.ts new file mode 100644 index 0000000..904329f --- /dev/null +++ b/src/test/javascript/jest.ts @@ -0,0 +1,2 @@ +import 'jest-preset-angular'; +import './jest-global-mocks'; diff --git a/src/test/javascript/spec/app/account/activate/activate.component.spec.ts b/src/test/javascript/spec/app/account/activate/activate.component.spec.ts new file mode 100644 index 0000000..e3151e0 --- /dev/null +++ b/src/test/javascript/spec/app/account/activate/activate.component.spec.ts @@ -0,0 +1,72 @@ +import { TestBed, async, tick, fakeAsync, inject } from '@angular/core/testing'; +import { ActivatedRoute } from '@angular/router'; +import { of, throwError } from 'rxjs'; + +import { EpasmedTestModule } from '../../../test.module'; +import { MockActivatedRoute } from '../../../helpers/mock-route.service'; +import { ActivateService } from 'app/account/activate/activate.service'; +import { ActivateComponent } from 'app/account/activate/activate.component'; + +describe('Component Tests', () => { + describe('ActivateComponent', () => { + let comp: ActivateComponent; + + beforeEach(async(() => { + TestBed.configureTestingModule({ + imports: [EpasmedTestModule], + declarations: [ActivateComponent], + providers: [ + { + provide: ActivatedRoute, + useValue: new MockActivatedRoute({ key: 'ABC123' }), + }, + ], + }) + .overrideTemplate(ActivateComponent, '') + .compileComponents(); + })); + + beforeEach(() => { + const fixture = TestBed.createComponent(ActivateComponent); + comp = fixture.componentInstance; + }); + + it('calls activate.get with the key from params', inject( + [ActivateService], + fakeAsync((service: ActivateService) => { + spyOn(service, 'get').and.returnValue(of()); + + comp.ngOnInit(); + tick(); + + expect(service.get).toHaveBeenCalledWith('ABC123'); + }) + )); + + it('should set set success to true upon successful activation', inject( + [ActivateService], + fakeAsync((service: ActivateService) => { + spyOn(service, 'get').and.returnValue(of({})); + + comp.ngOnInit(); + tick(); + + expect(comp.error).toBe(false); + expect(comp.success).toBe(true); + }) + )); + + it('should set set error to true upon activation failure', inject( + [ActivateService], + fakeAsync((service: ActivateService) => { + spyOn(service, 'get').and.returnValue(throwError('ERROR')); + + comp.ngOnInit(); + tick(); + + expect(comp.error).toBe(true); + expect(comp.success).toBe(false); + }) + )); + }); +}); diff --git a/src/test/javascript/spec/app/account/password-reset/finish/password-reset-finish.component.spec.ts b/src/test/javascript/spec/app/account/password-reset/finish/password-reset-finish.component.spec.ts new file mode 100644 index 0000000..cc4e016 --- /dev/null +++ b/src/test/javascript/spec/app/account/password-reset/finish/password-reset-finish.component.spec.ts @@ -0,0 +1,102 @@ +import { ElementRef } from '@angular/core'; +import { ComponentFixture, TestBed, inject, tick, fakeAsync } from '@angular/core/testing'; +import { FormBuilder } from '@angular/forms'; +import { ActivatedRoute } from '@angular/router'; +import { of, throwError } from 'rxjs'; + +import { EpasmedTestModule } from '../../../../test.module'; +import { PasswordResetFinishComponent } from 'app/account/password-reset/finish/password-reset-finish.component'; +import { PasswordResetFinishService } from 'app/account/password-reset/finish/password-reset-finish.service'; +import { MockActivatedRoute } from '../../../../helpers/mock-route.service'; + +describe('Component Tests', () => { + describe('PasswordResetFinishComponent', () => { + let fixture: ComponentFixture; + let comp: PasswordResetFinishComponent; + + beforeEach(() => { + fixture = TestBed.configureTestingModule({ + imports: [EpasmedTestModule], + declarations: [PasswordResetFinishComponent], + providers: [ + FormBuilder, + { + provide: ActivatedRoute, + useValue: new MockActivatedRoute({ key: 'XYZPDQ' }), + }, + ], + }) + .overrideTemplate(PasswordResetFinishComponent, '') + .createComponent(PasswordResetFinishComponent); + }); + + beforeEach(() => { + fixture = TestBed.createComponent(PasswordResetFinishComponent); + comp = fixture.componentInstance; + comp.ngOnInit(); + }); + + it('should define its initial state', () => { + expect(comp.initialized).toBe(true); + expect(comp.key).toEqual('XYZPDQ'); + }); + + it('sets focus after the view has been initialized', () => { + const node = { + focus(): void {}, + }; + comp.newPassword = new ElementRef(node); + spyOn(node, 'focus'); + + comp.ngAfterViewInit(); + + expect(node.focus).toHaveBeenCalled(); + }); + + it('should ensure the two passwords entered match', () => { + comp.passwordForm.patchValue({ + newPassword: 'password', + confirmPassword: 'non-matching', + }); + + comp.finishReset(); + + expect(comp.doNotMatch).toBe(true); + }); + + it('should update success to true after resetting password', inject( + [PasswordResetFinishService], + fakeAsync((service: PasswordResetFinishService) => { + spyOn(service, 'save').and.returnValue(of({})); + comp.passwordForm.patchValue({ + newPassword: 'password', + confirmPassword: 'password', + }); + + comp.finishReset(); + tick(); + + expect(service.save).toHaveBeenCalledWith('XYZPDQ', 'password'); + expect(comp.success).toBe(true); + }) + )); + + it('should notify of generic error', inject( + [PasswordResetFinishService], + fakeAsync((service: PasswordResetFinishService) => { + spyOn(service, 'save').and.returnValue(throwError('ERROR')); + comp.passwordForm.patchValue({ + newPassword: 'password', + confirmPassword: 'password', + }); + + comp.finishReset(); + tick(); + + expect(service.save).toHaveBeenCalledWith('XYZPDQ', 'password'); + expect(comp.success).toBe(false); + expect(comp.error).toBe(true); + }) + )); + }); +}); diff --git a/src/test/javascript/spec/app/account/password-reset/init/password-reset-init.component.spec.ts b/src/test/javascript/spec/app/account/password-reset/init/password-reset-init.component.spec.ts new file mode 100644 index 0000000..2c6e2fd --- /dev/null +++ b/src/test/javascript/spec/app/account/password-reset/init/password-reset-init.component.spec.ts @@ -0,0 +1,66 @@ +import { ElementRef } from '@angular/core'; +import { ComponentFixture, TestBed, inject } from '@angular/core/testing'; +import { FormBuilder } from '@angular/forms'; +import { of, throwError } from 'rxjs'; + +import { EpasmedTestModule } from '../../../../test.module'; +import { PasswordResetInitComponent } from 'app/account/password-reset/init/password-reset-init.component'; +import { PasswordResetInitService } from 'app/account/password-reset/init/password-reset-init.service'; + +describe('Component Tests', () => { + describe('PasswordResetInitComponent', () => { + let fixture: ComponentFixture; + let comp: PasswordResetInitComponent; + + beforeEach(() => { + fixture = TestBed.configureTestingModule({ + imports: [EpasmedTestModule], + declarations: [PasswordResetInitComponent], + providers: [FormBuilder], + }) + .overrideTemplate(PasswordResetInitComponent, '') + .createComponent(PasswordResetInitComponent); + comp = fixture.componentInstance; + }); + + it('sets focus after the view has been initialized', () => { + const node = { + focus(): void {}, + }; + comp.email = new ElementRef(node); + spyOn(node, 'focus'); + + comp.ngAfterViewInit(); + + expect(node.focus).toHaveBeenCalled(); + }); + + it('notifies of success upon successful requestReset', inject([PasswordResetInitService], (service: PasswordResetInitService) => { + spyOn(service, 'save').and.returnValue(of({})); + comp.resetRequestForm.patchValue({ + email: 'user@domain.com', + }); + + comp.requestReset(); + + expect(service.save).toHaveBeenCalledWith('user@domain.com'); + expect(comp.success).toBe(true); + })); + + it('no notification of success upon error response', inject([PasswordResetInitService], (service: PasswordResetInitService) => { + spyOn(service, 'save').and.returnValue( + throwError({ + status: 503, + data: 'something else', + }) + ); + comp.resetRequestForm.patchValue({ + email: 'user@domain.com', + }); + comp.requestReset(); + + expect(service.save).toHaveBeenCalledWith('user@domain.com'); + expect(comp.success).toBe(false); + })); + }); +}); diff --git a/src/test/javascript/spec/app/account/password/password-strength-bar.component.spec.ts b/src/test/javascript/spec/app/account/password/password-strength-bar.component.spec.ts new file mode 100644 index 0000000..a5fdfea --- /dev/null +++ b/src/test/javascript/spec/app/account/password/password-strength-bar.component.spec.ts @@ -0,0 +1,48 @@ +import { ComponentFixture, TestBed, async } from '@angular/core/testing'; + +import { PasswordStrengthBarComponent } from 'app/account/password/password-strength-bar.component'; + +describe('Component Tests', () => { + describe('PasswordStrengthBarComponent', () => { + let comp: PasswordStrengthBarComponent; + let fixture: ComponentFixture; + + beforeEach(async(() => { + TestBed.configureTestingModule({ + declarations: [PasswordStrengthBarComponent], + }) + .overrideTemplate(PasswordStrengthBarComponent, '') + .compileComponents(); + })); + + beforeEach(() => { + fixture = TestBed.createComponent(PasswordStrengthBarComponent); + comp = fixture.componentInstance; + }); + + describe('PasswordStrengthBarComponents', () => { + it('should initialize with default values', () => { + expect(comp.measureStrength('')).toBe(0); + expect(comp.colors).toEqual(['#F00', '#F90', '#FF0', '#9F0', '#0F0']); + expect(comp.getColor(0).idx).toBe(1); + expect(comp.getColor(0).color).toBe(comp.colors[0]); + }); + + it('should increase strength upon password value change', () => { + expect(comp.measureStrength('')).toBe(0); + expect(comp.measureStrength('aa')).toBeGreaterThanOrEqual(comp.measureStrength('')); + expect(comp.measureStrength('aa^6')).toBeGreaterThanOrEqual(comp.measureStrength('aa')); + expect(comp.measureStrength('Aa090(**)')).toBeGreaterThanOrEqual(comp.measureStrength('aa^6')); + expect(comp.measureStrength('Aa090(**)+-07365')).toBeGreaterThanOrEqual(comp.measureStrength('Aa090(**)')); + }); + + it('should change the color based on strength', () => { + expect(comp.getColor(0).color).toBe(comp.colors[0]); + expect(comp.getColor(11).color).toBe(comp.colors[1]); + expect(comp.getColor(22).color).toBe(comp.colors[2]); + expect(comp.getColor(33).color).toBe(comp.colors[3]); + expect(comp.getColor(44).color).toBe(comp.colors[4]); + }); + }); + }); +}); diff --git a/src/test/javascript/spec/app/account/password/password.component.spec.ts b/src/test/javascript/spec/app/account/password/password.component.spec.ts new file mode 100644 index 0000000..c3d73b0 --- /dev/null +++ b/src/test/javascript/spec/app/account/password/password.component.spec.ts @@ -0,0 +1,102 @@ +import { ComponentFixture, TestBed, async } from '@angular/core/testing'; +import { HttpResponse } from '@angular/common/http'; +import { FormBuilder } from '@angular/forms'; +import { of, throwError } from 'rxjs'; + +import { EpasmedTestModule } from '../../../test.module'; +import { PasswordComponent } from 'app/account/password/password.component'; +import { PasswordService } from 'app/account/password/password.service'; + +describe('Component Tests', () => { + describe('PasswordComponent', () => { + let comp: PasswordComponent; + let fixture: ComponentFixture; + let service: PasswordService; + + beforeEach(async(() => { + TestBed.configureTestingModule({ + imports: [EpasmedTestModule], + declarations: [PasswordComponent], + providers: [FormBuilder], + }) + .overrideTemplate(PasswordComponent, '') + .compileComponents(); + })); + + beforeEach(() => { + fixture = TestBed.createComponent(PasswordComponent); + comp = fixture.componentInstance; + service = fixture.debugElement.injector.get(PasswordService); + }); + + it('should show error if passwords do not match', () => { + // GIVEN + comp.passwordForm.patchValue({ + newPassword: 'password1', + confirmPassword: 'password2', + }); + // WHEN + comp.changePassword(); + // THEN + expect(comp.doNotMatch).toBe(true); + expect(comp.error).toBe(false); + expect(comp.success).toBe(false); + }); + + it('should call Auth.changePassword when passwords match', () => { + // GIVEN + const passwordValues = { + currentPassword: 'oldPassword', + newPassword: 'myPassword', + }; + + spyOn(service, 'save').and.returnValue(of(new HttpResponse({ body: true }))); + + comp.passwordForm.patchValue({ + currentPassword: passwordValues.currentPassword, + newPassword: passwordValues.newPassword, + confirmPassword: passwordValues.newPassword, + }); + + // WHEN + comp.changePassword(); + + // THEN + expect(service.save).toHaveBeenCalledWith(passwordValues.newPassword, passwordValues.currentPassword); + }); + + it('should set success to true upon success', () => { + // GIVEN + spyOn(service, 'save').and.returnValue(of(new HttpResponse({ body: true }))); + comp.passwordForm.patchValue({ + newPassword: 'myPassword', + confirmPassword: 'myPassword', + }); + + // WHEN + comp.changePassword(); + + // THEN + expect(comp.doNotMatch).toBe(false); + expect(comp.error).toBe(false); + expect(comp.success).toBe(true); + }); + + it('should notify of error if change password fails', () => { + // GIVEN + spyOn(service, 'save').and.returnValue(throwError('ERROR')); + comp.passwordForm.patchValue({ + newPassword: 'myPassword', + confirmPassword: 'myPassword', + }); + + // WHEN + comp.changePassword(); + + // THEN + expect(comp.doNotMatch).toBe(false); + expect(comp.success).toBe(false); + expect(comp.error).toBe(true); + }); + }); +}); diff --git a/src/test/javascript/spec/app/account/register/register.component.spec.ts b/src/test/javascript/spec/app/account/register/register.component.spec.ts new file mode 100644 index 0000000..5e4a6fa --- /dev/null +++ b/src/test/javascript/spec/app/account/register/register.component.spec.ts @@ -0,0 +1,134 @@ +import { ComponentFixture, TestBed, async, inject, tick, fakeAsync } from '@angular/core/testing'; +import { FormBuilder } from '@angular/forms'; +import { of, throwError } from 'rxjs'; + +import { EpasmedTestModule } from '../../../test.module'; +import { EMAIL_ALREADY_USED_TYPE, LOGIN_ALREADY_USED_TYPE } from 'app/shared/constants/error.constants'; +import { RegisterService } from 'app/account/register/register.service'; +import { RegisterComponent } from 'app/account/register/register.component'; + +describe('Component Tests', () => { + describe('RegisterComponent', () => { + let fixture: ComponentFixture; + let comp: RegisterComponent; + + beforeEach(async(() => { + TestBed.configureTestingModule({ + imports: [EpasmedTestModule], + declarations: [RegisterComponent], + providers: [FormBuilder], + }) + .overrideTemplate(RegisterComponent, '') + .compileComponents(); + })); + + beforeEach(() => { + fixture = TestBed.createComponent(RegisterComponent); + comp = fixture.componentInstance; + }); + + it('should ensure the two passwords entered match', () => { + comp.registerForm.patchValue({ + password: 'password', + confirmPassword: 'non-matching', + }); + + comp.register(); + + expect(comp.doNotMatch).toBe(true); + }); + + it('should update success to true after creating an account', inject( + [RegisterService], + fakeAsync((service: RegisterService) => { + spyOn(service, 'save').and.returnValue(of({})); + comp.registerForm.patchValue({ + password: 'password', + confirmPassword: 'password', + }); + + comp.register(); + tick(); + + expect(service.save).toHaveBeenCalledWith({ + email: '', + password: 'password', + login: '', + langKey: 'en', + }); + expect(comp.success).toBe(true); + expect(comp.errorUserExists).toBe(false); + expect(comp.errorEmailExists).toBe(false); + expect(comp.error).toBe(false); + }) + )); + + it('should notify of user existence upon 400/login already in use', inject( + [RegisterService], + fakeAsync((service: RegisterService) => { + spyOn(service, 'save').and.returnValue( + throwError({ + status: 400, + error: { type: LOGIN_ALREADY_USED_TYPE }, + }) + ); + comp.registerForm.patchValue({ + password: 'password', + confirmPassword: 'password', + }); + + comp.register(); + tick(); + + expect(comp.errorUserExists).toBe(true); + expect(comp.errorEmailExists).toBe(false); + expect(comp.error).toBe(false); + }) + )); + + it('should notify of email existence upon 400/email address already in use', inject( + [RegisterService], + fakeAsync((service: RegisterService) => { + spyOn(service, 'save').and.returnValue( + throwError({ + status: 400, + error: { type: EMAIL_ALREADY_USED_TYPE }, + }) + ); + comp.registerForm.patchValue({ + password: 'password', + confirmPassword: 'password', + }); + + comp.register(); + tick(); + + expect(comp.errorEmailExists).toBe(true); + expect(comp.errorUserExists).toBe(false); + expect(comp.error).toBe(false); + }) + )); + + it('should notify of generic error', inject( + [RegisterService], + fakeAsync((service: RegisterService) => { + spyOn(service, 'save').and.returnValue( + throwError({ + status: 503, + }) + ); + comp.registerForm.patchValue({ + password: 'password', + confirmPassword: 'password', + }); + + comp.register(); + tick(); + + expect(comp.errorUserExists).toBe(false); + expect(comp.errorEmailExists).toBe(false); + expect(comp.error).toBe(true); + }) + )); + }); +}); diff --git a/src/test/javascript/spec/app/account/sessions/sessions.component.spec.ts b/src/test/javascript/spec/app/account/sessions/sessions.component.spec.ts new file mode 100644 index 0000000..b3e2cfd --- /dev/null +++ b/src/test/javascript/spec/app/account/sessions/sessions.component.spec.ts @@ -0,0 +1,112 @@ +import { ComponentFixture, TestBed, inject, tick, fakeAsync } from '@angular/core/testing'; +import { of, throwError } from 'rxjs'; + +import { EpasmedTestModule } from '../../../test.module'; +import { Session } from 'app/account/sessions/session.model'; +import { SessionsComponent } from 'app/account/sessions/sessions.component'; +import { SessionsService } from 'app/account/sessions/sessions.service'; +import { MockAccountService } from '../../../helpers/mock-account.service'; +import { AccountService } from 'app/core/auth/account.service'; + +describe('Component Tests', () => { + let sessions: Session[]; + let fixture: ComponentFixture; + let comp: SessionsComponent; + + describe('SessionsComponent', () => { + beforeEach(() => { + sessions = [new Session('xxxxxx==', new Date(2015, 10, 15), '0:0:0:0:0:0:0:1', 'Mozilla/5.0')]; + + fixture = TestBed.configureTestingModule({ + imports: [EpasmedTestModule], + declarations: [SessionsComponent], + }) + .overrideTemplate(SessionsComponent, '') + .createComponent(SessionsComponent); + comp = fixture.componentInstance; + }); + + it('should define its initial state', inject( + [AccountService, SessionsService], + fakeAsync((mockAccountService: MockAccountService, service: SessionsService) => { + mockAccountService.spy('identity').and.returnValue( + of({ + id: 'fuzzer', + }) + ); + spyOn(service, 'findAll').and.returnValue(of(sessions)); + + comp.ngOnInit(); + tick(); + + expect(mockAccountService.identitySpy).toHaveBeenCalled(); + expect(service.findAll).toHaveBeenCalled(); + expect(comp.success).toBe(false); + expect(comp.error).toBe(false); + expect(comp.account).toEqual({ + id: 'fuzzer', + }); + expect(comp.sessions).toEqual(sessions); + }) + )); + + it('should call delete on Sessions to invalidate a session', inject( + [AccountService, SessionsService], + fakeAsync((mockAccountService: MockAccountService, service: SessionsService) => { + mockAccountService.spy('identity').and.returnValue( + of({ + id: 'fuzzer', + }) + ); + spyOn(service, 'findAll').and.returnValue(of(sessions)); + spyOn(service, 'delete').and.returnValue(of({})); + + comp.ngOnInit(); + comp.invalidate('xyz'); + tick(); + + expect(service.delete).toHaveBeenCalledWith('xyz'); + }) + )); + + it('should call delete on Sessions and notify of error', inject( + [AccountService, SessionsService], + fakeAsync((mockAccountService: MockAccountService, service: SessionsService) => { + mockAccountService.spy('identity').and.returnValue( + of({ + id: 'fuzzer', + }) + ); + spyOn(service, 'findAll').and.returnValue(of(sessions)); + spyOn(service, 'delete').and.returnValue(throwError({})); + + comp.ngOnInit(); + comp.invalidate('xyz'); + tick(); + + expect(comp.success).toBe(false); + expect(comp.error).toBe(true); + }) + )); + + it('should call notify of success upon session invalidation', inject( + [AccountService, SessionsService], + fakeAsync((mockAccountService: MockAccountService, service: SessionsService) => { + mockAccountService.spy('identity').and.returnValue( + of({ + id: 'fuzzer', + }) + ); + spyOn(service, 'findAll').and.returnValue(of(sessions)); + spyOn(service, 'delete').and.returnValue(of({})); + + comp.ngOnInit(); + comp.invalidate('xyz'); + tick(); + + expect(comp.error).toBe(false); + expect(comp.success).toBe(true); + }) + )); + }); +}); diff --git a/src/test/javascript/spec/app/account/settings/settings.component.spec.ts b/src/test/javascript/spec/app/account/settings/settings.component.spec.ts new file mode 100644 index 0000000..5174142 --- /dev/null +++ b/src/test/javascript/spec/app/account/settings/settings.component.spec.ts @@ -0,0 +1,88 @@ +import { ComponentFixture, TestBed, async } from '@angular/core/testing'; +import { FormBuilder } from '@angular/forms'; +import { throwError, of } from 'rxjs'; + +import { EpasmedTestModule } from '../../../test.module'; +import { AccountService } from 'app/core/auth/account.service'; +import { Account } from 'app/core/user/account.model'; +import { SettingsComponent } from 'app/account/settings/settings.component'; +import { MockAccountService } from '../../../helpers/mock-account.service'; + +describe('Component Tests', () => { + describe('SettingsComponent', () => { + let comp: SettingsComponent; + let fixture: ComponentFixture; + let mockAuth: MockAccountService; + const accountValues: Account = { + firstName: 'John', + lastName: 'Doe', + activated: true, + email: 'john.doe@mail.com', + langKey: 'en', + login: 'john', + authorities: [], + imageUrl: '', + }; + + beforeEach(async(() => { + TestBed.configureTestingModule({ + imports: [EpasmedTestModule], + declarations: [SettingsComponent], + providers: [FormBuilder], + }) + .overrideTemplate(SettingsComponent, '') + .compileComponents(); + })); + + beforeEach(() => { + fixture = TestBed.createComponent(SettingsComponent); + comp = fixture.componentInstance; + mockAuth = TestBed.get(AccountService); + mockAuth.setIdentityResponse(accountValues); + }); + + it('should send the current identity upon save', () => { + // GIVEN + mockAuth.saveSpy.and.returnValue(of({})); + const settingsFormValues = { + firstName: 'John', + lastName: 'Doe', + email: 'john.doe@mail.com', + }; + + // WHEN + comp.ngOnInit(); + comp.save(); + + // THEN + expect(mockAuth.identitySpy).toHaveBeenCalled(); + expect(mockAuth.saveSpy).toHaveBeenCalledWith(accountValues); + expect(mockAuth.authenticateSpy).toHaveBeenCalledWith(accountValues); + expect(comp.settingsForm.value).toEqual(settingsFormValues); + }); + + it('should notify of success upon successful save', () => { + // GIVEN + mockAuth.saveSpy.and.returnValue(of({})); + + // WHEN + comp.ngOnInit(); + comp.save(); + + // THEN + expect(comp.success).toBe(true); + }); + + it('should notify of error upon failed save', () => { + // GIVEN + mockAuth.saveSpy.and.returnValue(throwError('ERROR')); + + // WHEN + comp.ngOnInit(); + comp.save(); + + // THEN + expect(comp.success).toBe(false); + }); + }); +}); diff --git a/src/test/javascript/spec/app/admin/audits/audits.component.spec.ts b/src/test/javascript/spec/app/admin/audits/audits.component.spec.ts new file mode 100644 index 0000000..ddcb281 --- /dev/null +++ b/src/test/javascript/spec/app/admin/audits/audits.component.spec.ts @@ -0,0 +1,204 @@ +import { ComponentFixture, TestBed, async } from '@angular/core/testing'; +import { HttpHeaders, HttpResponse } from '@angular/common/http'; +import { Router, ActivatedRoute } from '@angular/router'; +import { of } from 'rxjs'; +import { advanceTo } from 'jest-date-mock'; + +import { EpasmedTestModule } from '../../../test.module'; +import { AuditsComponent } from 'app/admin/audits/audits.component'; +import { AuditsService } from 'app/admin/audits/audits.service'; +import { Audit } from 'app/admin/audits/audit.model'; +import { ITEMS_PER_PAGE } from 'app/shared/constants/pagination.constants'; +import { MockRouter, MockActivatedRoute } from '../../../helpers/mock-route.service'; + +function build2DigitsDatePart(datePart: number): string { + return `0${datePart}`.slice(-2); +} + +function getDate(isToday = true): string { + let date: Date = new Date(); + if (isToday) { + // Today + 1 day - needed if the current day must be included + date.setDate(date.getDate() + 1); + } else { + // get last month + if (date.getMonth() === 0) { + date = new Date(date.getFullYear() - 1, 11, date.getDate()); + } else { + date = new Date(date.getFullYear(), date.getMonth() - 1, date.getDate()); + } + } + const monthString = build2DigitsDatePart(date.getMonth() + 1); + const dateString = build2DigitsDatePart(date.getDate()); + return `${date.getFullYear()}-${monthString}-${dateString}`; +} + +describe('Component Tests', () => { + describe('AuditsComponent', () => { + let comp: AuditsComponent; + let fixture: ComponentFixture; + let service: AuditsService; + let mockRouter: MockRouter; + let mockActivatedRoute: MockActivatedRoute; + + beforeEach(async(() => { + TestBed.configureTestingModule({ + imports: [EpasmedTestModule], + declarations: [AuditsComponent], + providers: [AuditsService], + }) + .overrideTemplate(AuditsComponent, '') + .compileComponents(); + })); + + beforeEach(() => { + fixture = TestBed.createComponent(AuditsComponent); + comp = fixture.componentInstance; + service = fixture.debugElement.injector.get(AuditsService); + mockRouter = TestBed.get(Router); + mockActivatedRoute = TestBed.get(ActivatedRoute); + }); + + describe('today function', () => { + it('should set toDate to current date', () => { + comp.ngOnInit(); + expect(comp.toDate).toBe(getDate()); + }); + + it('if current day is last day of month then should set toDate to first day of next month', () => { + advanceTo(new Date(2019, 0, 31, 0, 0, 0)); + comp.ngOnInit(); + expect(comp.toDate).toBe('2019-02-01'); + }); + + it('if current day is not last day of month then should set toDate to next day of current month', () => { + advanceTo(new Date(2019, 0, 27, 0, 0, 0)); + comp.ngOnInit(); + expect(comp.toDate).toBe('2019-01-28'); + }); + }); + + describe('previousMonth function', () => { + it('should set fromDate to previous month', () => { + comp.ngOnInit(); + expect(comp.fromDate).toBe(getDate(false)); + }); + + it('if current month is January then should set fromDate to previous year last month', () => { + advanceTo(new Date(2019, 0, 20, 0, 0, 0)); + comp.ngOnInit(); + expect(comp.fromDate).toBe('2018-12-20'); + }); + + it('if current month is not January then should set fromDate to current year previous month', () => { + advanceTo(new Date(2019, 1, 20, 0, 0, 0)); + comp.ngOnInit(); + expect(comp.fromDate).toBe('2019-01-20'); + }); + }); + + describe('By default, on init', () => { + it('should set all default values correctly', () => { + fixture.detectChanges(); + expect(comp.toDate).toBe(getDate()); + expect(comp.fromDate).toBe(getDate(false)); + expect(comp.itemsPerPage).toBe(ITEMS_PER_PAGE); + expect(comp.page).toBe(1); + expect(comp.ascending).toBe(false); + expect(comp.predicate).toBe('id'); + }); + }); + + describe('OnInit', () => { + it('Should call load all on init', () => { + // GIVEN + const headers = new HttpHeaders().append('X-Total-Count', '1'); + const audit = new Audit({ remoteAddress: '127.0.0.1', sessionId: '123' }, 'user', '20140101', 'AUTHENTICATION_SUCCESS'); + spyOn(service, 'query').and.returnValue( + of( + new HttpResponse({ + body: [audit], + headers, + }) + ) + ); + + // WHEN + comp.ngOnInit(); + + // THEN + expect(service.query).toHaveBeenCalledTimes(1); + expect(comp.audits && comp.audits[0]).toEqual(jasmine.objectContaining(audit)); + expect(comp.totalItems).toBe(1); + }); + }); + + describe('Create sort object', () => { + beforeEach(() => { + spyOn(service, 'query').and.returnValue(of(new HttpResponse({ body: null }))); + }); + + it('Should sort only by id asc', () => { + // GIVEN + mockActivatedRoute.setParameters({ + sort: 'id,desc', + }); + + // WHEN + comp.ngOnInit(); + + // THEN + expect(service.query).toBeCalledWith( + expect.objectContaining({ + sort: ['id,desc'], + }) + ); + }); + + it('Should sort by timestamp asc then by id', () => { + // GIVEN + mockActivatedRoute.setParameters({ + sort: 'timestamp,asc', + }); + + // WHEN + comp.ngOnInit(); + + // THEN + expect(service.query).toBeCalledWith( + expect.objectContaining({ + sort: ['timestamp,asc', 'id'], + }) + ); + }); + }); + + describe('transition', () => { + it('Should not query data if fromDate and toDate are empty', () => { + // GIVEN + comp.toDate = ''; + comp.fromDate = ''; + + // WHEN + comp.transition(); + + // THEN + expect(comp.canLoad()).toBe(false); + expect(mockRouter.navigateSpy).not.toBeCalled(); + }); + + it('Should query data if fromDate and toDate are not empty', () => { + // GIVEN + comp.toDate = getDate(); + comp.fromDate = getDate(false); + + // WHEN + comp.transition(); + + // THEN + expect(comp.canLoad()).toBe(true); + expect(mockRouter.navigateSpy).toBeCalled(); + }); + }); + }); +}); diff --git a/src/test/javascript/spec/app/admin/audits/audits.service.spec.ts b/src/test/javascript/spec/app/admin/audits/audits.service.spec.ts new file mode 100644 index 0000000..9f0dfc6 --- /dev/null +++ b/src/test/javascript/spec/app/admin/audits/audits.service.spec.ts @@ -0,0 +1,70 @@ +import { TestBed } from '@angular/core/testing'; +import { HttpClientTestingModule, HttpTestingController } from '@angular/common/http/testing'; +import { HttpResponse, HttpErrorResponse } from '@angular/common/http'; + +import { AuditsService, AuditsQuery } from 'app/admin/audits/audits.service'; +import { Audit } from 'app/admin/audits/audit.model'; +import { SERVER_API_URL } from 'app/app.constants'; + +describe('Service Tests', () => { + describe('Audits Service', () => { + let service: AuditsService; + let httpMock: HttpTestingController; + const fakeRequest: AuditsQuery = { page: 0, size: 0, sort: [], fromDate: '', toDate: '' }; + + beforeEach(() => { + TestBed.configureTestingModule({ + imports: [HttpClientTestingModule], + }); + + service = TestBed.get(AuditsService); + httpMock = TestBed.get(HttpTestingController); + }); + + afterEach(() => { + httpMock.verify(); + }); + + describe('Service methods', () => { + it('should call correct URL', () => { + service.query(fakeRequest).subscribe(); + + const req = httpMock.expectOne({ method: 'GET' }); + const resourceUrl = SERVER_API_URL + 'management/audits'; + expect(req.request.url).toEqual(resourceUrl); + }); + + it('should return Audits', () => { + let expectedResult: HttpResponse = new HttpResponse({ body: [] }); + const audit = new Audit({ remoteAddress: '127.0.0.1', sessionId: '123' }, 'user', '20140101', 'AUTHENTICATION_SUCCESS'); + + service.query(fakeRequest).subscribe(received => { + expectedResult = received; + }); + + const req = httpMock.expectOne({ method: 'GET' }); + req.flush([audit]); + let audits: Audit[] = []; + if (expectedResult.body !== null) { + audits = expectedResult.body; + } + expect(audits.length).toBe(1); + expect(audits[0]).toEqual(audit); + }); + + it('should propagate not found response', () => { + let expectedResult = 0; + service.query(fakeRequest).subscribe(null, (error: HttpErrorResponse) => { + expectedResult = error.status; + }); + + const req = httpMock.expectOne({ method: 'GET' }); + req.flush('Invalid request parameters', { + status: 404, + statusText: 'Bad Request', + }); + expect(expectedResult).toEqual(404); + }); + }); + }); +}); diff --git a/src/test/javascript/spec/app/admin/configuration/configuration.component.spec.ts b/src/test/javascript/spec/app/admin/configuration/configuration.component.spec.ts new file mode 100644 index 0000000..ac39ab5 --- /dev/null +++ b/src/test/javascript/spec/app/admin/configuration/configuration.component.spec.ts @@ -0,0 +1,68 @@ +import { ComponentFixture, TestBed, async } from '@angular/core/testing'; +import { of } from 'rxjs'; + +import { EpasmedTestModule } from '../../../test.module'; +import { ConfigurationComponent } from 'app/admin/configuration/configuration.component'; +import { ConfigurationService, Bean, PropertySource } from 'app/admin/configuration/configuration.service'; + +describe('Component Tests', () => { + describe('ConfigurationComponent', () => { + let comp: ConfigurationComponent; + let fixture: ComponentFixture; + let service: ConfigurationService; + + beforeEach(async(() => { + TestBed.configureTestingModule({ + imports: [EpasmedTestModule], + declarations: [ConfigurationComponent], + providers: [ConfigurationService], + }) + .overrideTemplate(ConfigurationComponent, '') + .compileComponents(); + })); + + beforeEach(() => { + fixture = TestBed.createComponent(ConfigurationComponent); + comp = fixture.componentInstance; + service = fixture.debugElement.injector.get(ConfigurationService); + }); + + describe('OnInit', () => { + it('Should call load all on init', () => { + // GIVEN + const beans: Bean[] = [ + { + prefix: 'jhipster', + properties: { + clientApp: { + name: 'jhipsterApp', + }, + }, + }, + ]; + const propertySources: PropertySource[] = [ + { + name: 'server.ports', + properties: { + 'local.server.port': { + value: '8080', + }, + }, + }, + ]; + spyOn(service, 'getBeans').and.returnValue(of(beans)); + spyOn(service, 'getPropertySources').and.returnValue(of(propertySources)); + + // WHEN + comp.ngOnInit(); + + // THEN + expect(service.getBeans).toHaveBeenCalled(); + expect(service.getPropertySources).toHaveBeenCalled(); + expect(comp.allBeans).toEqual(beans); + expect(comp.beans).toEqual(beans); + expect(comp.propertySources).toEqual(propertySources); + }); + }); + }); +}); diff --git a/src/test/javascript/spec/app/admin/configuration/configuration.service.spec.ts b/src/test/javascript/spec/app/admin/configuration/configuration.service.spec.ts new file mode 100644 index 0000000..0ec5162 --- /dev/null +++ b/src/test/javascript/spec/app/admin/configuration/configuration.service.spec.ts @@ -0,0 +1,81 @@ +import { TestBed } from '@angular/core/testing'; +import { HttpClientTestingModule, HttpTestingController } from '@angular/common/http/testing'; + +import { ConfigurationService, ConfigProps, Env, Bean, PropertySource } from 'app/admin/configuration/configuration.service'; +import { SERVER_API_URL } from 'app/app.constants'; + +describe('Service Tests', () => { + describe('Logs Service', () => { + let service: ConfigurationService; + let httpMock: HttpTestingController; + let expectedResult: Bean[] | PropertySource[] | null; + + beforeEach(() => { + TestBed.configureTestingModule({ + imports: [HttpClientTestingModule], + }); + + expectedResult = null; + service = TestBed.get(ConfigurationService); + httpMock = TestBed.get(HttpTestingController); + }); + + afterEach(() => { + httpMock.verify(); + }); + + describe('Service methods', () => { + it('should call correct URL', () => { + service.getBeans().subscribe(); + + const req = httpMock.expectOne({ method: 'GET' }); + const resourceUrl = SERVER_API_URL + 'management/configprops'; + expect(req.request.url).toEqual(resourceUrl); + }); + + it('should get the config', () => { + const bean: Bean = { + prefix: 'jhipster', + properties: { + clientApp: { + name: 'jhipsterApp', + }, + }, + }; + const configProps: ConfigProps = { + contexts: { + jhipster: { + beans: { + 'io.github.jhipster.config.JHipsterProperties': bean, + }, + }, + }, + }; + service.getBeans().subscribe(received => (expectedResult = received)); + + const req = httpMock.expectOne({ method: 'GET' }); + req.flush(configProps); + expect(expectedResult).toEqual([bean]); + }); + + it('should get the env', () => { + const propertySources: PropertySource[] = [ + { + name: 'server.ports', + properties: { + 'local.server.port': { + value: '8080', + }, + }, + }, + ]; + const env: Env = { propertySources }; + service.getPropertySources().subscribe(received => (expectedResult = received)); + + const req = httpMock.expectOne({ method: 'GET' }); + req.flush(env); + expect(expectedResult).toEqual(propertySources); + }); + }); + }); +}); diff --git a/src/test/javascript/spec/app/admin/health/health.component.spec.ts b/src/test/javascript/spec/app/admin/health/health.component.spec.ts new file mode 100644 index 0000000..f917c1a --- /dev/null +++ b/src/test/javascript/spec/app/admin/health/health.component.spec.ts @@ -0,0 +1,67 @@ +import { ComponentFixture, TestBed, async } from '@angular/core/testing'; +import { HttpErrorResponse } from '@angular/common/http'; +import { of, throwError } from 'rxjs'; + +import { EpasmedTestModule } from '../../../test.module'; +import { HealthComponent } from 'app/admin/health/health.component'; +import { HealthService, Health } from 'app/admin/health/health.service'; + +describe('Component Tests', () => { + describe('HealthComponent', () => { + let comp: HealthComponent; + let fixture: ComponentFixture; + let service: HealthService; + + beforeEach(async(() => { + TestBed.configureTestingModule({ + imports: [EpasmedTestModule], + declarations: [HealthComponent], + }) + .overrideTemplate(HealthComponent, '') + .compileComponents(); + })); + + beforeEach(() => { + fixture = TestBed.createComponent(HealthComponent); + comp = fixture.componentInstance; + service = fixture.debugElement.injector.get(HealthService); + }); + + describe('getBadgeClass', () => { + it('should get badge class', () => { + const upBadgeClass = comp.getBadgeClass('UP'); + const downBadgeClass = comp.getBadgeClass('DOWN'); + expect(upBadgeClass).toEqual('badge-success'); + expect(downBadgeClass).toEqual('badge-danger'); + }); + }); + + describe('refresh', () => { + it('should call refresh on init', () => { + // GIVEN + const health: Health = { status: 'UP', components: { mail: { status: 'UP', details: 'mailDetails' } } }; + spyOn(service, 'checkHealth').and.returnValue(of(health)); + + // WHEN + comp.ngOnInit(); + + // THEN + expect(service.checkHealth).toHaveBeenCalled(); + expect(comp.health).toEqual(health); + }); + + it('should handle a 503 on refreshing health data', () => { + // GIVEN + const health: Health = { status: 'DOWN', components: { mail: { status: 'DOWN', details: 'mailDetails' } } }; + spyOn(service, 'checkHealth').and.returnValue(throwError(new HttpErrorResponse({ status: 503, error: health }))); + + // WHEN + comp.refresh(); + + // THEN + expect(service.checkHealth).toHaveBeenCalled(); + expect(comp.health).toEqual(health); + }); + }); + }); +}); diff --git a/src/test/javascript/spec/app/admin/logs/logs.component.spec.ts b/src/test/javascript/spec/app/admin/logs/logs.component.spec.ts new file mode 100644 index 0000000..e001ffe --- /dev/null +++ b/src/test/javascript/spec/app/admin/logs/logs.component.spec.ts @@ -0,0 +1,85 @@ +import { ComponentFixture, TestBed, async } from '@angular/core/testing'; +import { of } from 'rxjs'; + +import { EpasmedTestModule } from '../../../test.module'; +import { LogsComponent } from 'app/admin/logs/logs.component'; +import { LogsService } from 'app/admin/logs/logs.service'; +import { Log } from 'app/admin/logs/log.model'; + +describe('Component Tests', () => { + describe('LogsComponent', () => { + let comp: LogsComponent; + let fixture: ComponentFixture; + let service: LogsService; + + beforeEach(async(() => { + TestBed.configureTestingModule({ + imports: [EpasmedTestModule], + declarations: [LogsComponent], + providers: [LogsService], + }) + .overrideTemplate(LogsComponent, '') + .compileComponents(); + })); + + beforeEach(() => { + fixture = TestBed.createComponent(LogsComponent); + comp = fixture.componentInstance; + service = fixture.debugElement.injector.get(LogsService); + }); + + describe('OnInit', () => { + it('should set all default values correctly', () => { + expect(comp.filter).toBe(''); + expect(comp.orderProp).toBe('name'); + expect(comp.reverse).toBe(false); + }); + + it('Should call load all on init', () => { + // GIVEN + const log = new Log('main', 'WARN'); + spyOn(service, 'findAll').and.returnValue( + of({ + loggers: { + main: { + effectiveLevel: 'WARN', + }, + }, + }) + ); + + // WHEN + comp.ngOnInit(); + + // THEN + expect(service.findAll).toHaveBeenCalled(); + expect(comp.loggers && comp.loggers[0]).toEqual(jasmine.objectContaining(log)); + }); + }); + + describe('change log level', () => { + it('should change log level correctly', () => { + // GIVEN + const log = new Log('main', 'ERROR'); + spyOn(service, 'changeLevel').and.returnValue(of({})); + spyOn(service, 'findAll').and.returnValue( + of({ + loggers: { + main: { + effectiveLevel: 'ERROR', + }, + }, + }) + ); + + // WHEN + comp.changeLevel('main', 'ERROR'); + + // THEN + expect(service.changeLevel).toHaveBeenCalled(); + expect(service.findAll).toHaveBeenCalled(); + expect(comp.loggers && comp.loggers[0]).toEqual(jasmine.objectContaining(log)); + }); + }); + }); +}); diff --git a/src/test/javascript/spec/app/admin/logs/logs.service.spec.ts b/src/test/javascript/spec/app/admin/logs/logs.service.spec.ts new file mode 100644 index 0000000..19e45c7 --- /dev/null +++ b/src/test/javascript/spec/app/admin/logs/logs.service.spec.ts @@ -0,0 +1,44 @@ +import { TestBed } from '@angular/core/testing'; +import { HttpClientTestingModule, HttpTestingController } from '@angular/common/http/testing'; + +import { LogsService } from 'app/admin/logs/logs.service'; +import { SERVER_API_URL } from 'app/app.constants'; + +describe('Service Tests', () => { + describe('Logs Service', () => { + let service: LogsService; + let httpMock: HttpTestingController; + + beforeEach(() => { + TestBed.configureTestingModule({ + imports: [HttpClientTestingModule], + }); + + service = TestBed.get(LogsService); + httpMock = TestBed.get(HttpTestingController); + }); + + afterEach(() => { + httpMock.verify(); + }); + + describe('Service methods', () => { + it('should call correct URL', () => { + service.findAll().subscribe(); + + const req = httpMock.expectOne({ method: 'GET' }); + const resourceUrl = SERVER_API_URL + 'management/loggers'; + expect(req.request.url).toEqual(resourceUrl); + }); + + it('should change log level', () => { + service.changeLevel('main', 'ERROR').subscribe(); + + const req = httpMock.expectOne({ method: 'POST' }); + const resourceUrl = SERVER_API_URL + 'management/loggers/main'; + expect(req.request.url).toEqual(resourceUrl); + expect(req.request.body).toEqual({ configuredLevel: 'ERROR' }); + }); + }); + }); +}); diff --git a/src/test/javascript/spec/app/admin/metrics/metrics.component.spec.ts b/src/test/javascript/spec/app/admin/metrics/metrics.component.spec.ts new file mode 100644 index 0000000..6b99078 --- /dev/null +++ b/src/test/javascript/spec/app/admin/metrics/metrics.component.spec.ts @@ -0,0 +1,54 @@ +import { ComponentFixture, TestBed, async } from '@angular/core/testing'; +import { of } from 'rxjs'; + +import { EpasmedTestModule } from '../../../test.module'; +import { MetricsComponent } from 'app/admin/metrics/metrics.component'; +import { MetricsService } from 'app/admin/metrics/metrics.service'; + +describe('Component Tests', () => { + describe('MetricsComponent', () => { + let comp: MetricsComponent; + let fixture: ComponentFixture; + let service: MetricsService; + + beforeEach(async(() => { + TestBed.configureTestingModule({ + imports: [EpasmedTestModule], + declarations: [MetricsComponent], + }) + .overrideTemplate(MetricsComponent, '') + .compileComponents(); + })); + + beforeEach(() => { + fixture = TestBed.createComponent(MetricsComponent); + comp = fixture.componentInstance; + service = fixture.debugElement.injector.get(MetricsService); + }); + + describe('refresh', () => { + it('should call refresh on init', () => { + // GIVEN + const response = { + timers: { + service: 'test', + unrelatedKey: 'test', + }, + gauges: { + 'jcache.statistics': { + value: 2, + }, + unrelatedKey: 'test', + }, + }; + spyOn(service, 'getMetrics').and.returnValue(of(response)); + + // WHEN + comp.ngOnInit(); + + // THEN + expect(service.getMetrics).toHaveBeenCalled(); + }); + }); + }); +}); diff --git a/src/test/javascript/spec/app/admin/metrics/metrics.service.spec.ts b/src/test/javascript/spec/app/admin/metrics/metrics.service.spec.ts new file mode 100644 index 0000000..472f9a6 --- /dev/null +++ b/src/test/javascript/spec/app/admin/metrics/metrics.service.spec.ts @@ -0,0 +1,68 @@ +import { TestBed } from '@angular/core/testing'; +import { HttpClientTestingModule, HttpTestingController } from '@angular/common/http/testing'; + +import { MetricsService, Metrics, ThreadDump } from 'app/admin/metrics/metrics.service'; +import { SERVER_API_URL } from 'app/app.constants'; + +describe('Service Tests', () => { + describe('Logs Service', () => { + let service: MetricsService; + let httpMock: HttpTestingController; + + beforeEach(() => { + TestBed.configureTestingModule({ + imports: [HttpClientTestingModule], + }); + service = TestBed.get(MetricsService); + httpMock = TestBed.get(HttpTestingController); + }); + + afterEach(() => { + httpMock.verify(); + }); + + describe('Service methods', () => { + it('should call correct URL', () => { + service.getMetrics().subscribe(); + + const req = httpMock.expectOne({ method: 'GET' }); + const resourceUrl = SERVER_API_URL + 'management/jhimetrics'; + expect(req.request.url).toEqual(resourceUrl); + }); + + it('should return Metrics', () => { + let expectedResult: Metrics | null = null; + const metrics: Metrics = { + jvm: {}, + 'http.server.requests': {}, + cache: {}, + services: {}, + databases: {}, + garbageCollector: {}, + processMetrics: {}, + }; + + service.getMetrics().subscribe(received => { + expectedResult = received; + }); + + const req = httpMock.expectOne({ method: 'GET' }); + req.flush(metrics); + expect(expectedResult).toEqual(metrics); + }); + + it('should return Thread Dump', () => { + let expectedResult: ThreadDump | null = null; + const dump: ThreadDump = { threads: [{ name: 'test1', threadState: 'RUNNABLE' }] }; + + service.threadDump().subscribe(received => { + expectedResult = received; + }); + + const req = httpMock.expectOne({ method: 'GET' }); + req.flush(dump); + expect(expectedResult).toEqual(dump); + }); + }); + }); +}); diff --git a/src/test/javascript/spec/app/admin/user-management/user-management-delete-dialog.component.spec.ts b/src/test/javascript/spec/app/admin/user-management/user-management-delete-dialog.component.spec.ts new file mode 100644 index 0000000..c54cd31 --- /dev/null +++ b/src/test/javascript/spec/app/admin/user-management/user-management-delete-dialog.component.spec.ts @@ -0,0 +1,56 @@ +import { ComponentFixture, TestBed, async, inject, fakeAsync, tick } from '@angular/core/testing'; +import { NgbActiveModal } from '@ng-bootstrap/ng-bootstrap'; +import { of } from 'rxjs'; +import { JhiEventManager } from 'ng-jhipster'; + +import { EpasmedTestModule } from '../../../test.module'; +import { MockEventManager } from '../../../helpers/mock-event-manager.service'; +import { MockActiveModal } from '../../../helpers/mock-active-modal.service'; +import { UserManagementDeleteDialogComponent } from 'app/admin/user-management/user-management-delete-dialog.component'; +import { UserService } from 'app/core/user/user.service'; + +describe('Component Tests', () => { + describe('User Management Delete Component', () => { + let comp: UserManagementDeleteDialogComponent; + let fixture: ComponentFixture; + let service: UserService; + let mockEventManager: MockEventManager; + let mockActiveModal: MockActiveModal; + + beforeEach(async(() => { + TestBed.configureTestingModule({ + imports: [EpasmedTestModule], + declarations: [UserManagementDeleteDialogComponent], + }) + .overrideTemplate(UserManagementDeleteDialogComponent, '') + .compileComponents(); + })); + + beforeEach(() => { + fixture = TestBed.createComponent(UserManagementDeleteDialogComponent); + comp = fixture.componentInstance; + service = fixture.debugElement.injector.get(UserService); + mockEventManager = TestBed.get(JhiEventManager); + mockActiveModal = TestBed.get(NgbActiveModal); + }); + + describe('confirmDelete', () => { + it('Should call delete service on confirmDelete', inject( + [], + fakeAsync(() => { + // GIVEN + spyOn(service, 'delete').and.returnValue(of({})); + + // WHEN + comp.confirmDelete('user'); + tick(); + + // THEN + expect(service.delete).toHaveBeenCalledWith('user'); + expect(mockActiveModal.closeSpy).toHaveBeenCalled(); + expect(mockEventManager.broadcastSpy).toHaveBeenCalled(); + }) + )); + }); + }); +}); diff --git a/src/test/javascript/spec/app/admin/user-management/user-management-detail.component.spec.ts b/src/test/javascript/spec/app/admin/user-management/user-management-detail.component.spec.ts new file mode 100644 index 0000000..c1e0bb0 --- /dev/null +++ b/src/test/javascript/spec/app/admin/user-management/user-management-detail.component.spec.ts @@ -0,0 +1,62 @@ +import { ComponentFixture, TestBed, async } from '@angular/core/testing'; +import { ActivatedRoute } from '@angular/router'; +import { of } from 'rxjs'; + +import { Authority } from 'app/shared/constants/authority.constants'; +import { EpasmedTestModule } from '../../../test.module'; +import { UserManagementDetailComponent } from 'app/admin/user-management/user-management-detail.component'; +import { User } from 'app/core/user/user.model'; + +describe('Component Tests', () => { + describe('User Management Detail Component', () => { + let comp: UserManagementDetailComponent; + let fixture: ComponentFixture; + const route: ActivatedRoute = ({ + data: of({ user: new User(1, 'user', 'first', 'last', 'first@last.com', true, 'en', [Authority.USER], 'admin') }), + } as any) as ActivatedRoute; + + beforeEach(async(() => { + TestBed.configureTestingModule({ + imports: [EpasmedTestModule], + declarations: [UserManagementDetailComponent], + providers: [ + { + provide: ActivatedRoute, + useValue: route, + }, + ], + }) + .overrideTemplate(UserManagementDetailComponent, '') + .compileComponents(); + })); + + beforeEach(() => { + fixture = TestBed.createComponent(UserManagementDetailComponent); + comp = fixture.componentInstance; + }); + + describe('OnInit', () => { + it('Should call load all on init', () => { + // GIVEN + + // WHEN + comp.ngOnInit(); + + // THEN + expect(comp.user).toEqual( + jasmine.objectContaining({ + id: 1, + login: 'user', + firstName: 'first', + lastName: 'last', + email: 'first@last.com', + activated: true, + langKey: 'en', + authorities: [Authority.USER], + createdBy: 'admin', + }) + ); + }); + }); + }); +}); diff --git a/src/test/javascript/spec/app/admin/user-management/user-management-update.component.spec.ts b/src/test/javascript/spec/app/admin/user-management/user-management-update.component.spec.ts new file mode 100644 index 0000000..343ce1f --- /dev/null +++ b/src/test/javascript/spec/app/admin/user-management/user-management-update.component.spec.ts @@ -0,0 +1,104 @@ +import { ComponentFixture, TestBed, async, inject, fakeAsync, tick } from '@angular/core/testing'; +import { HttpResponse } from '@angular/common/http'; +import { FormBuilder } from '@angular/forms'; +import { ActivatedRoute } from '@angular/router'; +import { of } from 'rxjs'; + +import { Authority } from 'app/shared/constants/authority.constants'; +import { EpasmedTestModule } from '../../../test.module'; +import { UserManagementUpdateComponent } from 'app/admin/user-management/user-management-update.component'; +import { UserService } from 'app/core/user/user.service'; +import { User } from 'app/core/user/user.model'; + +describe('Component Tests', () => { + describe('User Management Update Component', () => { + let comp: UserManagementUpdateComponent; + let fixture: ComponentFixture; + let service: UserService; + const route: ActivatedRoute = ({ + data: of({ user: new User(1, 'user', 'first', 'last', 'first@last.com', true, 'en', [Authority.USER], 'admin') }), + } as any) as ActivatedRoute; + + beforeEach(async(() => { + TestBed.configureTestingModule({ + imports: [EpasmedTestModule], + declarations: [UserManagementUpdateComponent], + providers: [ + FormBuilder, + { + provide: ActivatedRoute, + useValue: route, + }, + ], + }) + .overrideTemplate(UserManagementUpdateComponent, '') + .compileComponents(); + })); + + beforeEach(() => { + fixture = TestBed.createComponent(UserManagementUpdateComponent); + comp = fixture.componentInstance; + service = fixture.debugElement.injector.get(UserService); + }); + + describe('OnInit', () => { + it('Should load authorities and language on init', inject( + [], + fakeAsync(() => { + // GIVEN + spyOn(service, 'authorities').and.returnValue(of(['USER'])); + + // WHEN + comp.ngOnInit(); + + // THEN + expect(service.authorities).toHaveBeenCalled(); + expect(comp.authorities).toEqual(['USER']); + }) + )); + }); + + describe('save', () => { + it('Should call update service on save for existing user', inject( + [], + fakeAsync(() => { + // GIVEN + const entity = new User(123); + spyOn(service, 'update').and.returnValue( + of( + new HttpResponse({ + body: entity, + }) + ) + ); + comp.user = entity; + comp.editForm.patchValue({ id: entity.id }); + // WHEN + comp.save(); + tick(); // simulate async + + // THEN + expect(service.update).toHaveBeenCalledWith(entity); + expect(comp.isSaving).toEqual(false); + }) + )); + + it('Should call create service on save for new user', inject( + [], + fakeAsync(() => { + // GIVEN + const entity = new User(); + spyOn(service, 'create').and.returnValue(of(new HttpResponse({ body: entity }))); + comp.user = entity; + // WHEN + comp.save(); + tick(); // simulate async + + // THEN + expect(service.create).toHaveBeenCalledWith(entity); + expect(comp.isSaving).toEqual(false); + }) + )); + }); + }); +}); diff --git a/src/test/javascript/spec/app/admin/user-management/user-management.component.spec.ts b/src/test/javascript/spec/app/admin/user-management/user-management.component.spec.ts new file mode 100644 index 0000000..a3a8ab4 --- /dev/null +++ b/src/test/javascript/spec/app/admin/user-management/user-management.component.spec.ts @@ -0,0 +1,86 @@ +import { ComponentFixture, TestBed, async, inject, fakeAsync, tick } from '@angular/core/testing'; +import { HttpHeaders, HttpResponse } from '@angular/common/http'; +import { of } from 'rxjs'; + +import { EpasmedTestModule } from '../../../test.module'; +import { UserManagementComponent } from 'app/admin/user-management/user-management.component'; +import { UserService } from 'app/core/user/user.service'; +import { User } from 'app/core/user/user.model'; + +describe('Component Tests', () => { + describe('User Management Component', () => { + let comp: UserManagementComponent; + let fixture: ComponentFixture; + let service: UserService; + + beforeEach(async(() => { + TestBed.configureTestingModule({ + imports: [EpasmedTestModule], + declarations: [UserManagementComponent], + }) + .overrideTemplate(UserManagementComponent, '') + .compileComponents(); + })); + + beforeEach(() => { + fixture = TestBed.createComponent(UserManagementComponent); + comp = fixture.componentInstance; + service = fixture.debugElement.injector.get(UserService); + }); + + describe('OnInit', () => { + it('Should call load all on init', inject( + [], + fakeAsync(() => { + // GIVEN + const headers = new HttpHeaders().append('link', 'link;link'); + spyOn(service, 'query').and.returnValue( + of( + new HttpResponse({ + body: [new User(123)], + headers, + }) + ) + ); + + // WHEN + comp.ngOnInit(); + tick(); // simulate async + + // THEN + expect(service.query).toHaveBeenCalled(); + expect(comp.users && comp.users[0]).toEqual(jasmine.objectContaining({ id: 123 })); + }) + )); + }); + + describe('setActive', () => { + it('Should update user and call load all', inject( + [], + fakeAsync(() => { + // GIVEN + const headers = new HttpHeaders().append('link', 'link;link'); + const user = new User(123); + spyOn(service, 'query').and.returnValue( + of( + new HttpResponse({ + body: [user], + headers, + }) + ) + ); + spyOn(service, 'update').and.returnValue(of(new HttpResponse({ status: 200 }))); + + // WHEN + comp.setActive(user, true); + tick(); // simulate async + + // THEN + expect(service.update).toHaveBeenCalledWith({ ...user, activated: true }); + expect(service.query).toHaveBeenCalled(); + expect(comp.users && comp.users[0]).toEqual(jasmine.objectContaining({ id: 123 })); + }) + )); + }); + }); +}); diff --git a/src/test/javascript/spec/app/core/login/login-modal.service.spec.ts b/src/test/javascript/spec/app/core/login/login-modal.service.spec.ts new file mode 100644 index 0000000..d0d386a --- /dev/null +++ b/src/test/javascript/spec/app/core/login/login-modal.service.spec.ts @@ -0,0 +1,48 @@ +import { TestBed } from '@angular/core/testing'; +import { NgbModal } from '@ng-bootstrap/ng-bootstrap'; + +import { LoginModalService } from 'app/core/login/login-modal.service'; + +// Mock class for NgbModalRef +export class MockNgbModalRef { + result: Promise = new Promise(resolve => resolve('x')); +} + +describe('Service Tests', () => { + describe('Login Modal Service', () => { + let service: LoginModalService; + let modalService: NgbModal; + + beforeEach(() => { + service = TestBed.get(LoginModalService); + modalService = TestBed.get(NgbModal); + }); + + describe('Service methods', () => { + it('Should call open method for NgbModal when open method is called', () => { + // GIVEN + const mockModalRef: MockNgbModalRef = new MockNgbModalRef(); + spyOn(modalService, 'open').and.returnValue(mockModalRef); + + // WHEN + service.open(); + + // THEN + expect(modalService.open).toHaveBeenCalled(); + }); + + it('Should call open method for NgbModal one time when open method is called twice', () => { + // GIVEN + const mockModalRef: MockNgbModalRef = new MockNgbModalRef(); + spyOn(modalService, 'open').and.returnValue(mockModalRef); + + // WHEN + service.open(); + service.open(); + + // THEN + expect(modalService.open).toHaveBeenCalledTimes(1); + }); + }); + }); +}); diff --git a/src/test/javascript/spec/app/core/user/account.service.spec.ts b/src/test/javascript/spec/app/core/user/account.service.spec.ts new file mode 100644 index 0000000..c7e8858 --- /dev/null +++ b/src/test/javascript/spec/app/core/user/account.service.spec.ts @@ -0,0 +1,209 @@ +import { Router } from '@angular/router'; +import { HttpClientTestingModule, HttpTestingController } from '@angular/common/http/testing'; +import { TestBed } from '@angular/core/testing'; +import { JhiDateUtils } from 'ng-jhipster'; +import { NgxWebstorageModule } from 'ngx-webstorage'; + +import { SERVER_API_URL } from 'app/app.constants'; +import { AccountService } from 'app/core/auth/account.service'; +import { Account } from 'app/core/user/account.model'; +import { Authority } from 'app/shared/constants/authority.constants'; +import { StateStorageService } from 'app/core/auth/state-storage.service'; +import { MockRouter } from '../../../helpers/mock-route.service'; +import { MockStateStorageService } from '../../../helpers/mock-state-storage.service'; + +function accountWithAuthorities(authorities: string[]): Account { + return { + activated: true, + authorities, + email: '', + firstName: '', + langKey: '', + lastName: '', + login: '', + imageUrl: '', + }; +} + +describe('Service Tests', () => { + describe('Account Service', () => { + let service: AccountService; + let httpMock: HttpTestingController; + let storageService: MockStateStorageService; + let router: MockRouter; + + beforeEach(() => { + TestBed.configureTestingModule({ + imports: [HttpClientTestingModule, NgxWebstorageModule.forRoot()], + providers: [ + JhiDateUtils, + { + provide: StateStorageService, + useClass: MockStateStorageService, + }, + { + provide: Router, + useClass: MockRouter, + }, + ], + }); + + service = TestBed.get(AccountService); + httpMock = TestBed.get(HttpTestingController); + storageService = TestBed.get(StateStorageService); + router = TestBed.get(Router); + }); + + afterEach(() => { + httpMock.verify(); + }); + + describe('authenticate', () => { + it('authenticationState should emit null if input is null', () => { + // GIVEN + let userIdentity: Account | null = accountWithAuthorities([]); + service.getAuthenticationState().subscribe(account => (userIdentity = account)); + + // WHEN + service.authenticate(null); + + // THEN + expect(userIdentity).toBeNull(); + expect(service.isAuthenticated()).toBe(false); + }); + + it('authenticationState should emit the same account as was in input parameter', () => { + // GIVEN + const expectedResult = accountWithAuthorities([]); + let userIdentity: Account | null = null; + service.getAuthenticationState().subscribe(account => (userIdentity = account)); + + // WHEN + service.authenticate(expectedResult); + + // THEN + expect(userIdentity).toEqual(expectedResult); + expect(service.isAuthenticated()).toBe(true); + }); + }); + + describe('identity', () => { + it('should call /account if user is undefined', () => { + service.identity().subscribe(); + const req = httpMock.expectOne({ method: 'GET' }); + const resourceUrl = SERVER_API_URL + 'api/account'; + + expect(req.request.url).toEqual(`${resourceUrl}`); + }); + + it('should call /account only once if not logged out after first authentication and should call /account again if user has logged out', () => { + // Given the user is authenticated + service.identity().subscribe(); + httpMock.expectOne({ method: 'GET' }).flush({}); + + // When I call + service.identity().subscribe(); + + // Then there is no second request + httpMock.expectNone({ method: 'GET' }); + + // When I log out + service.authenticate(null); + // and then call + service.identity().subscribe(); + + // Then there is a new request + httpMock.expectOne({ method: 'GET' }); + }); + + describe('navigateToStoredUrl', () => { + it('should navigate to the previous stored url post successful authentication', () => { + // GIVEN + storageService.setResponse('admin/users?page=0'); + + // WHEN + service.identity().subscribe(); + httpMock.expectOne({ method: 'GET' }).flush({}); + + // THEN + expect(storageService.getUrlSpy).toHaveBeenCalledTimes(1); + expect(storageService.clearUrlSpy).toHaveBeenCalledTimes(1); + expect(router.navigateByUrlSpy).toHaveBeenCalledWith('admin/users?page=0'); + }); + + it('should not navigate to the previous stored url when authentication fails', () => { + // WHEN + service.identity().subscribe(); + httpMock.expectOne({ method: 'GET' }).error(new ErrorEvent('')); + + // THEN + expect(storageService.getUrlSpy).not.toHaveBeenCalled(); + expect(storageService.clearUrlSpy).not.toHaveBeenCalled(); + expect(router.navigateByUrlSpy).not.toHaveBeenCalled(); + }); + + it('should not navigate to the previous stored url when no such url exists post successful authentication', () => { + // GIVEN + storageService.setResponse(null); + + // WHEN + service.identity().subscribe(); + httpMock.expectOne({ method: 'GET' }).flush({}); + + // THEN + expect(storageService.getUrlSpy).toHaveBeenCalledTimes(1); + expect(storageService.clearUrlSpy).not.toHaveBeenCalled(); + expect(router.navigateByUrlSpy).not.toHaveBeenCalled(); + }); + }); + }); + + describe('hasAnyAuthority', () => { + describe('hasAnyAuthority string parameter', () => { + it('should return false if user is not logged', () => { + const hasAuthority = service.hasAnyAuthority(Authority.USER); + expect(hasAuthority).toBe(false); + }); + + it('should return false if user is logged and has not authority', () => { + service.authenticate(accountWithAuthorities([Authority.USER])); + + const hasAuthority = service.hasAnyAuthority(Authority.ADMIN); + + expect(hasAuthority).toBe(false); + }); + + it('should return true if user is logged and has authority', () => { + service.authenticate(accountWithAuthorities([Authority.USER])); + + const hasAuthority = service.hasAnyAuthority(Authority.USER); + + expect(hasAuthority).toBe(true); + }); + }); + + describe('hasAnyAuthority array parameter', () => { + it('should return false if user is not logged', () => { + const hasAuthority = service.hasAnyAuthority([Authority.USER]); + expect(hasAuthority).toBeFalsy(); + }); + + it('should return false if user is logged and has not authority', () => { + service.authenticate(accountWithAuthorities([Authority.USER])); + + const hasAuthority = service.hasAnyAuthority([Authority.ADMIN]); + + expect(hasAuthority).toBe(false); + }); + + it('should return true if user is logged and has authority', () => { + service.authenticate(accountWithAuthorities([Authority.USER])); + + const hasAuthority = service.hasAnyAuthority([Authority.USER, Authority.ADMIN]); + + expect(hasAuthority).toBe(true); + }); + }); + }); + }); +}); diff --git a/src/test/javascript/spec/app/core/user/user.service.spec.ts b/src/test/javascript/spec/app/core/user/user.service.spec.ts new file mode 100644 index 0000000..391b1e1 --- /dev/null +++ b/src/test/javascript/spec/app/core/user/user.service.spec.ts @@ -0,0 +1,79 @@ +import { TestBed } from '@angular/core/testing'; +import { HttpErrorResponse } from '@angular/common/http'; +import { HttpClientTestingModule, HttpTestingController } from '@angular/common/http/testing'; +import { JhiDateUtils } from 'ng-jhipster'; + +import { Authority } from 'app/shared/constants/authority.constants'; +import { UserService } from 'app/core/user/user.service'; +import { User } from 'app/core/user/user.model'; +import { SERVER_API_URL } from 'app/app.constants'; + +describe('Service Tests', () => { + describe('User Service', () => { + let service: UserService; + let httpMock: HttpTestingController; + + beforeEach(() => { + TestBed.configureTestingModule({ + imports: [HttpClientTestingModule], + providers: [JhiDateUtils], + }); + + service = TestBed.get(UserService); + httpMock = TestBed.get(HttpTestingController); + }); + + afterEach(() => { + httpMock.verify(); + }); + + describe('Service methods', () => { + it('should call correct URL', () => { + service.find('user').subscribe(); + + const req = httpMock.expectOne({ method: 'GET' }); + const resourceUrl = SERVER_API_URL + 'api/users'; + expect(req.request.url).toEqual(`${resourceUrl}/user`); + }); + + it('should return User', () => { + let expectedResult: string | undefined; + + service.find('user').subscribe(received => { + expectedResult = received.login; + }); + + const req = httpMock.expectOne({ method: 'GET' }); + req.flush(new User(1, 'user')); + expect(expectedResult).toEqual('user'); + }); + + it('should return Authorities', () => { + let expectedResult: string[] = []; + + service.authorities().subscribe(authorities => { + expectedResult = authorities; + }); + const req = httpMock.expectOne({ method: 'GET' }); + + req.flush([Authority.USER, Authority.ADMIN]); + expect(expectedResult).toEqual([Authority.USER, Authority.ADMIN]); + }); + + it('should propagate not found response', () => { + let expectedResult = 0; + + service.find('user').subscribe(null, (error: HttpErrorResponse) => { + expectedResult = error.status; + }); + + const req = httpMock.expectOne({ method: 'GET' }); + req.flush('Invalid request parameters', { + status: 404, + statusText: 'Bad Request', + }); + expect(expectedResult).toEqual(404); + }); + }); + }); +}); diff --git a/src/test/javascript/spec/app/home/home.component.spec.ts b/src/test/javascript/spec/app/home/home.component.spec.ts new file mode 100644 index 0000000..4fd496c --- /dev/null +++ b/src/test/javascript/spec/app/home/home.component.spec.ts @@ -0,0 +1,55 @@ +import { ComponentFixture, TestBed, async } from '@angular/core/testing'; + +import { EpasmedTestModule } from '../../test.module'; +import { HomeComponent } from 'app/home/home.component'; +import { AccountService } from 'app/core/auth/account.service'; +import { LoginModalService } from 'app/core/login/login-modal.service'; + +describe('Component Tests', () => { + describe('Home Component', () => { + let comp: HomeComponent; + let fixture: ComponentFixture; + let accountService: AccountService; + let loginModalService: LoginModalService; + + beforeEach(async(() => { + TestBed.configureTestingModule({ + imports: [EpasmedTestModule], + declarations: [HomeComponent], + }) + .overrideTemplate(HomeComponent, '') + .compileComponents(); + })); + + beforeEach(() => { + fixture = TestBed.createComponent(HomeComponent); + comp = fixture.componentInstance; + accountService = TestBed.get(AccountService); + loginModalService = TestBed.get(LoginModalService); + }); + + it('Should call accountService.getAuthenticationState on init', () => { + // WHEN + comp.ngOnInit(); + + // THEN + expect(accountService.getAuthenticationState).toHaveBeenCalled(); + }); + + it('Should call accountService.isAuthenticated when it checks authentication', () => { + // WHEN + comp.isAuthenticated(); + + // THEN + expect(accountService.isAuthenticated).toHaveBeenCalled(); + }); + + it('Should call loginModalService.open on login', () => { + // WHEN + comp.login(); + + // THEN + expect(loginModalService.open).toHaveBeenCalled(); + }); + }); +}); diff --git a/src/test/javascript/spec/app/layouts/main/main.component.spec.ts b/src/test/javascript/spec/app/layouts/main/main.component.spec.ts new file mode 100644 index 0000000..c97dc55 --- /dev/null +++ b/src/test/javascript/spec/app/layouts/main/main.component.spec.ts @@ -0,0 +1,108 @@ +import { async, ComponentFixture, TestBed } from '@angular/core/testing'; +import { Router, RouterEvent, NavigationEnd } from '@angular/router'; +import { Title } from '@angular/platform-browser'; +import { Subject } from 'rxjs'; + +import { MainComponent } from 'app/layouts/main/main.component'; +import { EpasmedTestModule } from '../../../test.module'; +import { MockRouter } from '../../../helpers/mock-route.service'; + +describe('Component Tests', () => { + describe('MainComponent', () => { + let comp: MainComponent; + let fixture: ComponentFixture; + let router: MockRouter; + const routerEventsSubject = new Subject(); + let titleService: Title; + + beforeEach(async(() => { + TestBed.configureTestingModule({ + imports: [EpasmedTestModule], + declarations: [MainComponent], + providers: [Title], + }) + .overrideTemplate(MainComponent, '') + .compileComponents(); + })); + + beforeEach(() => { + fixture = TestBed.createComponent(MainComponent); + comp = fixture.componentInstance; + router = TestBed.get(Router); + router.setEvents(routerEventsSubject.asObservable()); + titleService = TestBed.get(Title); + }); + + describe('page title', () => { + let routerState: any; + const defaultPageTitle = 'ePASMed'; + const parentRoutePageTitle = 'parentTitle'; + const childRoutePageTitle = 'childTitle'; + const navigationEnd = new NavigationEnd(1, '', ''); + + beforeEach(() => { + routerState = { snapshot: { root: {} } }; + router.setRouterState(routerState); + spyOn(titleService, 'setTitle'); + comp.ngOnInit(); + }); + + describe('navigation end', () => { + it('should set page title to default title if pageTitle is missing on routes', () => { + // WHEN + routerEventsSubject.next(navigationEnd); + + // THEN + expect(titleService.setTitle).toHaveBeenCalledWith(defaultPageTitle); + }); + + it('should set page title to root route pageTitle if there is no child routes', () => { + // GIVEN + routerState.snapshot.root.data = { pageTitle: parentRoutePageTitle }; + + // WHEN + routerEventsSubject.next(navigationEnd); + + // THEN + expect(titleService.setTitle).toHaveBeenCalledWith(parentRoutePageTitle); + }); + + it('should set page title to child route pageTitle if child routes exist and pageTitle is set for child route', () => { + // GIVEN + routerState.snapshot.root.data = { pageTitle: parentRoutePageTitle }; + routerState.snapshot.root.firstChild = { data: { pageTitle: childRoutePageTitle } }; + + // WHEN + routerEventsSubject.next(navigationEnd); + + // THEN + expect(titleService.setTitle).toHaveBeenCalledWith(childRoutePageTitle); + }); + + it('should set page title to parent route pageTitle if child routes exists but pageTitle is not set for child route data', () => { + // GIVEN + routerState.snapshot.root.data = { pageTitle: parentRoutePageTitle }; + routerState.snapshot.root.firstChild = { data: {} }; + + // WHEN + routerEventsSubject.next(navigationEnd); + + // THEN + expect(titleService.setTitle).toHaveBeenCalledWith(parentRoutePageTitle); + }); + + it('should set page title to parent route pageTitle if child routes exists but data is not set for child route', () => { + // GIVEN + routerState.snapshot.root.data = { pageTitle: parentRoutePageTitle }; + routerState.snapshot.root.firstChild = {}; + + // WHEN + routerEventsSubject.next(navigationEnd); + + // THEN + expect(titleService.setTitle).toHaveBeenCalledWith(parentRoutePageTitle); + }); + }); + }); + }); +}); diff --git a/src/test/javascript/spec/app/layouts/navbar/navbar.component.spec.ts b/src/test/javascript/spec/app/layouts/navbar/navbar.component.spec.ts new file mode 100644 index 0000000..3abe090 --- /dev/null +++ b/src/test/javascript/spec/app/layouts/navbar/navbar.component.spec.ts @@ -0,0 +1,52 @@ +import { ComponentFixture, TestBed, async } from '@angular/core/testing'; +import { of } from 'rxjs'; + +import { EpasmedTestModule } from '../../../test.module'; +import { ProfileInfo } from 'app/layouts/profiles/profile-info.model'; +import { NavbarComponent } from 'app/layouts/navbar/navbar.component'; +import { AccountService } from 'app/core/auth/account.service'; +import { ProfileService } from 'app/layouts/profiles/profile.service'; + +describe('Component Tests', () => { + describe('Navbar Component', () => { + let comp: NavbarComponent; + let fixture: ComponentFixture; + let accountService: AccountService; + let profileService: ProfileService; + + beforeEach(async(() => { + TestBed.configureTestingModule({ + imports: [EpasmedTestModule], + declarations: [NavbarComponent], + }) + .overrideTemplate(NavbarComponent, '') + .compileComponents(); + })); + + beforeEach(() => { + fixture = TestBed.createComponent(NavbarComponent); + comp = fixture.componentInstance; + accountService = TestBed.get(AccountService); + profileService = TestBed.get(ProfileService); + }); + + it('Should call profileService.getProfileInfo on init', () => { + // GIVEN + spyOn(profileService, 'getProfileInfo').and.returnValue(of(new ProfileInfo())); + + // WHEN + comp.ngOnInit(); + + // THEN + expect(profileService.getProfileInfo).toHaveBeenCalled(); + }); + + it('Should call accountService.isAuthenticated on authentication', () => { + // WHEN + comp.isAuthenticated(); + + // THEN + expect(accountService.isAuthenticated).toHaveBeenCalled(); + }); + }); +}); diff --git a/src/test/javascript/spec/app/shared/alert/alert-error.component.spec.ts b/src/test/javascript/spec/app/shared/alert/alert-error.component.spec.ts new file mode 100644 index 0000000..245b611 --- /dev/null +++ b/src/test/javascript/spec/app/shared/alert/alert-error.component.spec.ts @@ -0,0 +1,140 @@ +import { ComponentFixture, TestBed, async } from '@angular/core/testing'; +import { HttpErrorResponse, HttpHeaders } from '@angular/common/http'; +import { JhiAlertService, JhiEventManager } from 'ng-jhipster'; + +import { EpasmedTestModule } from '../../../test.module'; +import { AlertErrorComponent } from 'app/shared/alert/alert-error.component'; +import { MockAlertService } from '../../../helpers/mock-alert.service'; + +describe('Component Tests', () => { + describe('Alert Error Component', () => { + let comp: AlertErrorComponent; + let fixture: ComponentFixture; + let eventManager: JhiEventManager; + + beforeEach(async(() => { + TestBed.configureTestingModule({ + imports: [EpasmedTestModule], + declarations: [AlertErrorComponent], + providers: [ + JhiEventManager, + { + provide: JhiAlertService, + useClass: MockAlertService, + }, + ], + }) + .overrideTemplate(AlertErrorComponent, '') + .compileComponents(); + })); + + beforeEach(() => { + fixture = TestBed.createComponent(AlertErrorComponent); + comp = fixture.componentInstance; + eventManager = fixture.debugElement.injector.get(JhiEventManager); + }); + + describe('Error Handling', () => { + it('Should display an alert on status 0', () => { + // GIVEN + eventManager.broadcast({ name: 'epasmedApp.httpError', content: { status: 0 } }); + // THEN + expect(comp.alerts.length).toBe(1); + expect(comp.alerts[0].msg).toBe('Server not reachable'); + }); + + it('Should display an alert on status 404', () => { + // GIVEN + eventManager.broadcast({ name: 'epasmedApp.httpError', content: { status: 404 } }); + // THEN + expect(comp.alerts.length).toBe(1); + expect(comp.alerts[0].msg).toBe('Not found'); + }); + + it('Should display an alert on generic error', () => { + // GIVEN + eventManager.broadcast({ name: 'epasmedApp.httpError', content: { error: { message: 'Error Message' } } }); + eventManager.broadcast({ name: 'epasmedApp.httpError', content: { error: 'Second Error Message' } }); + // THEN + expect(comp.alerts.length).toBe(2); + expect(comp.alerts[0].msg).toBe('Error Message'); + expect(comp.alerts[1].msg).toBe('Second Error Message'); + }); + + it('Should display an alert on status 400 for generic error', () => { + // GIVEN + const response = new HttpErrorResponse({ + url: 'http://localhost:8080/api/foos', + headers: new HttpHeaders(), + status: 400, + statusText: 'Bad Request', + error: { + type: 'https://www.jhipster.tech/problem/constraint-violation', + title: 'Bad Request', + status: 400, + path: '/api/foos', + message: 'error.validation', + }, + }); + eventManager.broadcast({ name: 'epasmedApp.httpError', content: response }); + // THEN + expect(comp.alerts.length).toBe(1); + expect(comp.alerts[0].msg).toBe('error.validation'); + }); + + it('Should display an alert on status 400 for generic error without message', () => { + // GIVEN + const response = new HttpErrorResponse({ + url: 'http://localhost:8080/api/foos', + headers: new HttpHeaders(), + status: 400, + error: 'Bad Request', + }); + eventManager.broadcast({ name: 'epasmedApp.httpError', content: response }); + // THEN + expect(comp.alerts.length).toBe(1); + expect(comp.alerts[0].msg).toBe('Bad Request'); + }); + + it('Should display an alert on status 400 for invalid parameters', () => { + // GIVEN + const response = new HttpErrorResponse({ + url: 'http://localhost:8080/api/foos', + headers: new HttpHeaders(), + status: 400, + statusText: 'Bad Request', + error: { + type: 'https://www.jhipster.tech/problem/constraint-violation', + title: 'Method argument not valid', + status: 400, + path: '/api/foos', + message: 'error.validation', + fieldErrors: [{ objectName: 'foo', field: 'minField', message: 'Min' }], + }, + }); + eventManager.broadcast({ name: 'epasmedApp.httpError', content: response }); + // THEN + expect(comp.alerts.length).toBe(1); + expect(comp.alerts[0].msg).toBe('Error on field "MinField"'); + }); + + it('Should display an alert on status 400 for error headers', () => { + // GIVEN + const response = new HttpErrorResponse({ + url: 'http://localhost:8080/api/foos', + headers: new HttpHeaders().append('app-error', 'Error Message').append('app-params', 'foo'), + status: 400, + statusText: 'Bad Request', + error: { + status: 400, + message: 'error.validation', + }, + }); + eventManager.broadcast({ name: 'epasmedApp.httpError', content: response }); + // THEN + expect(comp.alerts.length).toBe(1); + expect(comp.alerts[0].msg).toBe('Error Message'); + }); + }); + }); +}); diff --git a/src/test/javascript/spec/app/shared/alert/alert.component.spec.ts b/src/test/javascript/spec/app/shared/alert/alert.component.spec.ts new file mode 100644 index 0000000..acdf7af --- /dev/null +++ b/src/test/javascript/spec/app/shared/alert/alert.component.spec.ts @@ -0,0 +1,44 @@ +import { ComponentFixture, TestBed, async } from '@angular/core/testing'; +import { JhiAlertService } from 'ng-jhipster'; + +import { EpasmedTestModule } from '../../../test.module'; +import { AlertComponent } from 'app/shared/alert/alert.component'; + +describe('Component Tests', () => { + describe('Alert Component', () => { + let comp: AlertComponent; + let fixture: ComponentFixture; + let alertService: JhiAlertService; + + beforeEach(async(() => { + TestBed.configureTestingModule({ + imports: [EpasmedTestModule], + declarations: [AlertComponent], + }) + .overrideTemplate(AlertComponent, '') + .compileComponents(); + })); + + beforeEach(() => { + fixture = TestBed.createComponent(AlertComponent); + comp = fixture.componentInstance; + alertService = TestBed.get(JhiAlertService); + }); + + it('Should call alertService.get on init', () => { + // WHEN + comp.ngOnInit(); + + // THEN + expect(alertService.get).toHaveBeenCalled(); + }); + + it('Should call alertService.clear on destroy', () => { + // WHEN + comp.ngOnDestroy(); + + // THEN + expect(alertService.clear).toHaveBeenCalled(); + }); + }); +}); diff --git a/src/test/javascript/spec/app/shared/login/login.component.spec.ts b/src/test/javascript/spec/app/shared/login/login.component.spec.ts new file mode 100644 index 0000000..3f7a5f0 --- /dev/null +++ b/src/test/javascript/spec/app/shared/login/login.component.spec.ts @@ -0,0 +1,116 @@ +import { ComponentFixture, TestBed, async, inject, fakeAsync, tick } from '@angular/core/testing'; +import { FormBuilder } from '@angular/forms'; +import { Router } from '@angular/router'; +import { NgbActiveModal } from '@ng-bootstrap/ng-bootstrap'; + +import { LoginService } from 'app/core/login/login.service'; +import { LoginModalComponent } from 'app/shared/login/login.component'; +import { EpasmedTestModule } from '../../../test.module'; +import { MockLoginService } from '../../../helpers/mock-login.service'; +import { MockRouter } from '../../../helpers/mock-route.service'; +import { MockActiveModal } from '../../../helpers/mock-active-modal.service'; + +describe('Component Tests', () => { + describe('LoginComponent', () => { + let comp: LoginModalComponent; + let fixture: ComponentFixture; + let mockLoginService: MockLoginService; + let mockRouter: MockRouter; + let mockActiveModal: MockActiveModal; + + beforeEach(async(() => { + TestBed.configureTestingModule({ + imports: [EpasmedTestModule], + declarations: [LoginModalComponent], + providers: [ + FormBuilder, + { + provide: LoginService, + useClass: MockLoginService, + }, + ], + }) + .overrideTemplate(LoginModalComponent, '') + .compileComponents(); + })); + + beforeEach(() => { + fixture = TestBed.createComponent(LoginModalComponent); + comp = fixture.componentInstance; + mockLoginService = TestBed.get(LoginService); + mockRouter = TestBed.get(Router); + mockActiveModal = TestBed.get(NgbActiveModal); + }); + + it('should authenticate the user', inject( + [], + fakeAsync(() => { + // GIVEN + const credentials = { + username: 'admin', + password: 'admin', + rememberMe: true, + }; + + comp.loginForm.patchValue({ + username: 'admin', + password: 'admin', + rememberMe: true, + }); + mockLoginService.setResponse({}); + mockRouter.url = '/admin/metrics'; + + // WHEN/ + comp.login(); + tick(); // simulate async + + // THEN + expect(comp.authenticationError).toEqual(false); + expect(mockActiveModal.closeSpy).toHaveBeenCalled(); + expect(mockLoginService.loginSpy).toHaveBeenCalledWith(credentials); + }) + )); + + it('should empty the credentials upon cancel', () => { + // GIVEN + comp.loginForm.patchValue({ + username: 'admin', + password: 'admin', + }); + + const expected = { + username: '', + password: '', + rememberMe: false, + }; + + // WHEN + comp.cancel(); + + // THEN + expect(comp.authenticationError).toEqual(false); + expect(comp.loginForm.get('username')!.value).toEqual(expected.username); + expect(comp.loginForm.get('password')!.value).toEqual(expected.password); + expect(comp.loginForm.get('rememberMe')!.value).toEqual(expected.rememberMe); + expect(mockActiveModal.dismissSpy).toHaveBeenCalledWith('cancel'); + }); + + it('should redirect user when register', () => { + // WHEN + comp.register(); + + // THEN + expect(mockActiveModal.dismissSpy).toHaveBeenCalledWith('to state register'); + expect(mockRouter.navigateSpy).toHaveBeenCalledWith(['/account/register']); + }); + + it('should redirect user when request password', () => { + // WHEN + comp.requestResetPassword(); + + // THEN + expect(mockActiveModal.dismissSpy).toHaveBeenCalledWith('to state requestReset'); + expect(mockRouter.navigateSpy).toHaveBeenCalledWith(['/account/reset', 'request']); + }); + }); +}); diff --git a/src/test/javascript/spec/helpers/mock-account.service.ts b/src/test/javascript/spec/helpers/mock-account.service.ts new file mode 100644 index 0000000..b75655c --- /dev/null +++ b/src/test/javascript/spec/helpers/mock-account.service.ts @@ -0,0 +1,31 @@ +import Spy = jasmine.Spy; +import { of } from 'rxjs'; + +import { SpyObject } from './spyobject'; +import { AccountService } from 'app/core/auth/account.service'; +import { Account } from 'app/core/user/account.model'; + +export class MockAccountService extends SpyObject { + getSpy: Spy; + saveSpy: Spy; + authenticateSpy: Spy; + identitySpy: Spy; + getAuthenticationStateSpy: Spy; + isAuthenticated: Spy; + + constructor() { + super(AccountService); + + this.getSpy = this.spy('get').andReturn(this); + this.saveSpy = this.spy('save').andReturn(this); + this.authenticateSpy = this.spy('authenticate').andReturn(this); + this.identitySpy = this.spy('identity').andReturn(of(null)); + this.getAuthenticationStateSpy = this.spy('getAuthenticationState').andReturn(of(null)); + this.isAuthenticated = this.spy('isAuthenticated').andReturn(true); + } + + setIdentityResponse(account: Account | null): void { + this.identitySpy = this.spy('identity').andReturn(of(account)); + this.getAuthenticationStateSpy = this.spy('getAuthenticationState').andReturn(of(account)); + } +} diff --git a/src/test/javascript/spec/helpers/mock-active-modal.service.ts b/src/test/javascript/spec/helpers/mock-active-modal.service.ts new file mode 100644 index 0000000..bd32d74 --- /dev/null +++ b/src/test/javascript/spec/helpers/mock-active-modal.service.ts @@ -0,0 +1,15 @@ +import Spy = jasmine.Spy; +import { NgbActiveModal } from '@ng-bootstrap/ng-bootstrap'; + +import { SpyObject } from './spyobject'; + +export class MockActiveModal extends SpyObject { + dismissSpy: Spy; + closeSpy: Spy; + + constructor() { + super(NgbActiveModal); + this.dismissSpy = this.spy('dismiss').andReturn(this); + this.closeSpy = this.spy('close').andReturn(this); + } +} diff --git a/src/test/javascript/spec/helpers/mock-alert.service.ts b/src/test/javascript/spec/helpers/mock-alert.service.ts new file mode 100644 index 0000000..2bf7724 --- /dev/null +++ b/src/test/javascript/spec/helpers/mock-alert.service.ts @@ -0,0 +1,13 @@ +import { JhiAlertService, JhiAlert } from 'ng-jhipster'; + +import { SpyObject } from './spyobject'; + +export class MockAlertService extends SpyObject { + constructor() { + super(JhiAlertService); + } + + addAlert(alertOptions: JhiAlert): JhiAlert { + return alertOptions; + } +} diff --git a/src/test/javascript/spec/helpers/mock-event-manager.service.ts b/src/test/javascript/spec/helpers/mock-event-manager.service.ts new file mode 100644 index 0000000..4a8f56a --- /dev/null +++ b/src/test/javascript/spec/helpers/mock-event-manager.service.ts @@ -0,0 +1,13 @@ +import Spy = jasmine.Spy; +import { JhiEventManager } from 'ng-jhipster'; + +import { SpyObject } from './spyobject'; + +export class MockEventManager extends SpyObject { + broadcastSpy: Spy; + + constructor() { + super(JhiEventManager); + this.broadcastSpy = this.spy('broadcast').andReturn(this); + } +} diff --git a/src/test/javascript/spec/helpers/mock-login-modal.service.ts b/src/test/javascript/spec/helpers/mock-login-modal.service.ts new file mode 100644 index 0000000..1c0f7c3 --- /dev/null +++ b/src/test/javascript/spec/helpers/mock-login-modal.service.ts @@ -0,0 +1,14 @@ +import Spy = jasmine.Spy; + +import { SpyObject } from './spyobject'; +import { LoginModalService } from 'app/core/login/login-modal.service'; + +export class MockLoginModalService extends SpyObject { + open: Spy; + + constructor() { + super(LoginModalService); + + this.open = this.spy('open'); + } +} diff --git a/src/test/javascript/spec/helpers/mock-login.service.ts b/src/test/javascript/spec/helpers/mock-login.service.ts new file mode 100644 index 0000000..a75a800 --- /dev/null +++ b/src/test/javascript/spec/helpers/mock-login.service.ts @@ -0,0 +1,27 @@ +import Spy = jasmine.Spy; +import { of } from 'rxjs'; + +import { SpyObject } from './spyobject'; +import { LoginService } from 'app/core/login/login.service'; + +export class MockLoginService extends SpyObject { + loginSpy: Spy; + logoutSpy: Spy; + registerSpy: Spy; + requestResetPasswordSpy: Spy; + cancelSpy: Spy; + + constructor() { + super(LoginService); + + this.loginSpy = this.spy('login').andReturn(of({})); + this.logoutSpy = this.spy('logout').andReturn(this); + this.registerSpy = this.spy('register').andReturn(this); + this.requestResetPasswordSpy = this.spy('requestResetPassword').andReturn(this); + this.cancelSpy = this.spy('cancel').andReturn(this); + } + + setResponse(json: any): void { + this.loginSpy = this.spy('login').andReturn(of(json)); + } +} diff --git a/src/test/javascript/spec/helpers/mock-route.service.ts b/src/test/javascript/spec/helpers/mock-route.service.ts new file mode 100644 index 0000000..c238688 --- /dev/null +++ b/src/test/javascript/spec/helpers/mock-route.service.ts @@ -0,0 +1,50 @@ +import Spy = jasmine.Spy; +import { ActivatedRoute, Router, RouterEvent, Data, Params } from '@angular/router'; +import { Observable, ReplaySubject } from 'rxjs'; + +import { SpyObject } from './spyobject'; + +export class MockActivatedRoute extends ActivatedRoute { + private queryParamsSubject = new ReplaySubject(); + private paramSubject = new ReplaySubject(); + private dataSubject = new ReplaySubject(); + + constructor(parameters: Params) { + super(); + this.queryParams = this.queryParamsSubject.asObservable(); + this.params = this.paramSubject.asObservable(); + this.data = this.dataSubject.asObservable(); + this.setParameters(parameters); + } + + setParameters(parameters: Params): void { + this.queryParamsSubject.next(parameters); + this.paramSubject.next(parameters); + this.dataSubject.next({ + ...parameters, + defaultSort: 'id,desc', + }); + } +} + +export class MockRouter extends SpyObject { + navigateSpy: Spy; + navigateByUrlSpy: Spy; + events: Observable | null = null; + routerState: any; + url = ''; + + constructor() { + super(Router); + this.navigateSpy = this.spy('navigate'); + this.navigateByUrlSpy = this.spy('navigateByUrl'); + } + + setEvents(events: Observable): void { + this.events = events; + } + + setRouterState(routerState: any): void { + this.routerState = routerState; + } +} diff --git a/src/test/javascript/spec/helpers/mock-state-storage.service.ts b/src/test/javascript/spec/helpers/mock-state-storage.service.ts new file mode 100644 index 0000000..1a23343 --- /dev/null +++ b/src/test/javascript/spec/helpers/mock-state-storage.service.ts @@ -0,0 +1,21 @@ +import Spy = jasmine.Spy; + +import { SpyObject } from './spyobject'; +import { StateStorageService } from 'app/core/auth/state-storage.service'; + +export class MockStateStorageService extends SpyObject { + getUrlSpy: Spy; + storeUrlSpy: Spy; + clearUrlSpy: Spy; + + constructor() { + super(StateStorageService); + this.getUrlSpy = this.spy('getUrl').andReturn(null); + this.storeUrlSpy = this.spy('storeUrl').andReturn(this); + this.clearUrlSpy = this.spy('clearUrl').andReturn(this); + } + + setResponse(previousUrl: string | null): void { + this.getUrlSpy = this.spy('getUrl').andReturn(previousUrl); + } +} diff --git a/src/test/javascript/spec/helpers/spyobject.ts b/src/test/javascript/spec/helpers/spyobject.ts new file mode 100644 index 0000000..eeb221a --- /dev/null +++ b/src/test/javascript/spec/helpers/spyobject.ts @@ -0,0 +1,48 @@ +export interface GuinessCompatibleSpy extends jasmine.Spy { + /** By chaining the spy with and.returnValue, all calls to the function will return a specific + * value. */ + andReturn(val: any): GuinessCompatibleSpy; + /** By chaining the spy with and.callFake, all calls to the spy will delegate to the supplied + * function. */ + andCallFake(fn: Function): GuinessCompatibleSpy; + /** removes all recorded calls */ + reset(): void; +} + +export class SpyObject { + constructor(type?: any) { + if (type) { + Object.keys(type.prototype).forEach(prop => { + let m = null; + try { + m = type.prototype[prop]; + } catch (e) { + // As we are creating spys for abstract classes, + // these classes might have getters that throw when they are accessed. + // As we are only auto creating spys for methods, this + // should not matter. + } + if (typeof m === 'function') { + this.spy(prop); + } + }); + } + } + + spy(name: string): GuinessCompatibleSpy { + if (!this[name]) { + this[name] = this.createGuinnessCompatibleSpy(name); + } + return this[name]; + } + + private createGuinnessCompatibleSpy(name: string): GuinessCompatibleSpy { + const newSpy: GuinessCompatibleSpy = jasmine.createSpy(name) as any; + newSpy.andCallFake = newSpy.and.callFake as any; + newSpy.andReturn = newSpy.and.returnValue as any; + newSpy.reset = newSpy.calls.reset as any; + // revisit return null here (previously needed for rtts_assert). + newSpy.and.returnValue(null); + return newSpy; + } +} diff --git a/src/test/javascript/spec/test.module.ts b/src/test/javascript/spec/test.module.ts new file mode 100644 index 0000000..d785099 --- /dev/null +++ b/src/test/javascript/spec/test.module.ts @@ -0,0 +1,67 @@ +import { DatePipe } from '@angular/common'; +import { ActivatedRoute, Router } from '@angular/router'; +import { NgModule } from '@angular/core'; +import { HttpClientTestingModule } from '@angular/common/http/testing'; +import { NgbActiveModal, NgbModal } from '@ng-bootstrap/ng-bootstrap'; +import { SessionStorageService, LocalStorageService } from 'ngx-webstorage'; +import { JhiDataUtils, JhiDateUtils, JhiEventManager, JhiAlertService, JhiParseLinks } from 'ng-jhipster'; + +import { AccountService } from 'app/core/auth/account.service'; +import { LoginModalService } from 'app/core/login/login-modal.service'; +import { MockLoginModalService } from './helpers/mock-login-modal.service'; +import { MockAccountService } from './helpers/mock-account.service'; +import { MockActivatedRoute, MockRouter } from './helpers/mock-route.service'; +import { MockActiveModal } from './helpers/mock-active-modal.service'; +import { MockAlertService } from './helpers/mock-alert.service'; +import { MockEventManager } from './helpers/mock-event-manager.service'; + +@NgModule({ + providers: [ + DatePipe, + JhiDataUtils, + JhiDateUtils, + JhiParseLinks, + { + provide: JhiEventManager, + useClass: MockEventManager, + }, + { + provide: NgbActiveModal, + useClass: MockActiveModal, + }, + { + provide: ActivatedRoute, + useValue: new MockActivatedRoute({ id: 123 }), + }, + { + provide: Router, + useClass: MockRouter, + }, + { + provide: AccountService, + useClass: MockAccountService, + }, + { + provide: LoginModalService, + useClass: MockLoginModalService, + }, + { + provide: JhiAlertService, + useClass: MockAlertService, + }, + { + provide: NgbModal, + useValue: null, + }, + { + provide: SessionStorageService, + useValue: null, + }, + { + provide: LocalStorageService, + useValue: null, + }, + ], + imports: [HttpClientTestingModule], +}) +export class EpasmedTestModule {} diff --git a/src/test/resources/config/application-testcontainers.yml b/src/test/resources/config/application-testcontainers.yml new file mode 100644 index 0000000..6e3b9b5 --- /dev/null +++ b/src/test/resources/config/application-testcontainers.yml @@ -0,0 +1,22 @@ +# =================================================================== +# Spring Boot configuration. +# +# This configuration is used for unit/integration tests with testcontainers database containers. +# +# To activate this configuration launch integration tests with the 'testcontainers' profile +# +# More information on database containers: https://www.testcontainers.org/modules/databases/ +# =================================================================== + +spring: + datasource: + type: com.zaxxer.hikari.HikariDataSource + driver-class-name: org.testcontainers.jdbc.ContainerDatabaseDriver + url: jdbc:tc:postgresql:latest:///epasmed?TC_TMPFS=/testtmpfs:rw + username: epasmed + password: + hikari: + poolName: Hikari + auto-commit: false + jpa: + database-platform: io.github.jhipster.domain.util.FixedPostgreSQL10Dialect diff --git a/src/test/resources/config/application.yml b/src/test/resources/config/application.yml new file mode 100644 index 0000000..e739a3d --- /dev/null +++ b/src/test/resources/config/application.yml @@ -0,0 +1,126 @@ +# =================================================================== +# Spring Boot configuration. +# +# This configuration is used for unit/integration tests. +# +# More information on profiles: https://www.jhipster.tech/profiles/ +# More information on configuration properties: https://www.jhipster.tech/common-application-properties/ +# =================================================================== + +# =================================================================== +# Standard Spring Boot properties. +# Full reference is available at: +# http://docs.spring.io/spring-boot/docs/current/reference/html/common-application-properties.html +# =================================================================== +logging: + level: + ROOT: DEBUG + io.github.jhipster: DEBUG + it.cnr.isti.epasmed: DEBUG + +spring: + profiles: + # Uncomment the following line to enable tests against production database type rather than H2, using Testcontainers + #active: testcontainer + application: + name: epasmed + jackson: + serialization: + write-durations-as-timestamps: false + mail: + host: localhost + main: + allow-bean-definition-overriding: true + messages: + basename: i18n/messages + task: + execution: + thread-name-prefix: epasmed-task- + pool: + core-size: 1 + max-size: 50 + queue-capacity: 10000 + scheduling: + thread-name-prefix: epasmed-scheduling- + pool: + size: 1 + thymeleaf: + mode: HTML + +server: + port: 10344 + address: localhost + +# =================================================================== +# JHipster specific properties +# +# Full reference is available at: https://www.jhipster.tech/common-application-properties/ +# =================================================================== + +jhipster: + clientApp: + name: 'epasmedApp' + logging: + # To test json console appender + use-json-format: false + logstash: + enabled: false + host: localhost + port: 5000 + queue-size: 512 + mail: + from: test@localhost + base-url: http://127.0.0.1:8080 + security: + remember-me: + # security key (this key should be unique for your application, and kept secret) + key: + metrics: + logs: # Reports metrics in the logs + enabled: true + report-frequency: 60 # in seconds + +# =================================================================== +# Application specific properties +# Add your own application properties here, see the ApplicationProperties class +# to have type-safe configuration, like in the JHipsterProperties above +# +# More documentation is available at: +# https://www.jhipster.tech/common-application-properties/ +# =================================================================== + +application: + datasource-epasmed: + datasource: + #type: com.zaxxer.hikari.HikariDataSource + url: jdbc:h2:file:./target/h2db/db/epasmed;DB_CLOSE_DELAY=-1 + username: epasmed + password: + hikari: + poolName: epasmedPool + auto-commit: false + maximum-pool-size: 10 + liquibase: + # Remove 'faker' if you do not want the sample data to be loaded automatically + contexts: dev, faker + change-log: classpath:config/liquibase/epasmed/master.xml + enabled: true + + datasource-sistemainformativo: + datasource: + url: jdbc:postgresql://sistemainformativo-dev.isti.cnr.it:5432/sistemainformativo + driver-class-name: org.postgresql.Driver + username: epasmed + password: + hikari: + poolName: sistemainformativoPool + auto-commit: false + maximum-pool-size: 10 + + datasource-epas-rest: + #rest-url: https://epas-demo.devel.iit.cnr.it/rest + rest-url: https://epas.isti.cnr.it/rest + rest-username1: isti_registry_manager + rest-password1: + rest-username2: isti_person_day_reader + rest-password2: diff --git a/src/test/resources/i18n/messages_en.properties b/src/test/resources/i18n/messages_en.properties new file mode 100644 index 0000000..f19db86 --- /dev/null +++ b/src/test/resources/i18n/messages_en.properties @@ -0,0 +1 @@ +email.test.title=test title diff --git a/src/test/resources/it/cnr/isti/epasmed/web/rest/epas/affiliazioniRepartiISTI.csv b/src/test/resources/it/cnr/isti/epasmed/web/rest/epas/affiliazioniRepartiISTI.csv new file mode 100644 index 0000000..478ded7 --- /dev/null +++ b/src/test/resources/it/cnr/isti/epasmed/web/rest/epas/affiliazioniRepartiISTI.csv @@ -0,0 +1,445 @@ +"rel_pers_reparto_codice","t_reparto_codice","t_personale_codice","t_personale_codice_fiscale","rel_pers_reparto_data_inizio","rel_pers_reparto_data_fine","rel_pers_reparto_perc" +1,27,1,"BCCFCM82C17A285V","2012-02-15 00:00:00","2016-10-02 00:00:00",100 +3,11,13,"BLTPLA71L20F052H","2012-02-21 00:00:00","2016-12-15 00:00:00",100 +19,11,13,"BLTPLA71L20F052H","2010-02-01 00:00:00","2012-01-31 00:00:00",100 +20,11,13,"BLTPLA71L20F052H","2016-12-16 00:00:00","2019-12-31 00:00:00",100 +31,26,20,"BNTFNC82S30F861R","2010-02-08 00:00:00","2017-04-07 00:00:00",100 +37,11,25,"BRDLSS82D69C415P","2012-09-01 00:00:00","2015-06-14 00:00:00",100 +42,11,29,"BRRFNC71M61G702S","2009-02-16 00:00:00","2011-05-31 00:00:00",50 +43,16,29,"BRRFNC71M61G702S","2011-01-25 00:00:00","2011-05-31 00:00:00",50 +46,27,31,"BRSPLA78S27G702C","2010-07-19 00:00:00","2012-10-15 00:00:00",100 +47,27,31,"BRSPLA78S27G702C","2012-10-16 00:00:00",,100 +55,19,37,"BRTNTN60R56E974Z","1987-09-01 00:00:00","2012-02-29 00:00:00",100 +56,20,37,"BRTNTN60R56E974Z","2012-03-01 00:00:00",,100 +62,11,42,"BSOCHR64R51Z110U","2011-01-21 00:00:00","2018-11-29 00:00:00",90 +63,16,42,"BSOCHR64R51Z110U","2011-01-25 00:00:00","2018-11-29 00:00:00",10 +74,26,51,"CGNPLA68M27B509P","1996-01-01 00:00:00",,100 +77,3,54,"CHRSVN66S21Z133B","2008-04-14 00:00:00","2012-02-29 00:00:00",100 +78,20,54,"CHRSVN66S21Z133B","2012-03-01 00:00:00",,100 +80,19,56,"CLBNNL81C05H224R","2011-05-02 00:00:00","2012-02-29 00:00:00",100 +81,20,56,"CLBNNL81C05H224R","2012-03-01 00:00:00","2018-12-26 00:00:00",100 +82,26,57,"CLLMRC75D24F952O","2008-04-14 00:00:00",,100 +84,22,59,"CLNSRA74S49A485U","2009-12-28 00:00:00",,100 +87,1,62,"CMPGRL78L31Z315T","2009-10-01 00:00:00",,100 +88,13,63,"CMPPNG66B61G702M","2008-04-14 00:00:00",,100 +89,11,64,"CNCCSR65S20E058O","2009-02-16 00:00:00","2020-09-24 00:00:00",100 +91,11,66,"CNDLRD76B13A399L","2011-09-16 00:00:00","2020-09-24 00:00:00",100 +92,14,67,"CNIRCR68T30E715W","2011-06-01 00:00:00",,100 +96,8,71,"CPPMSM69P25C773E","2009-02-16 00:00:00",,100 +108,11,78,"CROGPL80H25F839W","2012-11-01 00:00:00","2013-09-01 00:00:00",100 +113,26,82,"CRSMSM74A10H980B","2011-03-30 00:00:00","2012-02-29 00:00:00",100 +114,26,82,"CRSMSM74A10H980B","2012-03-01 00:00:00",,100 +119,27,87,"CSSPTR79C05G273U","2012-11-01 00:00:00","2018-12-26 00:00:00",100 +120,11,88,"CSTDTL57L65A864M","1988-06-11 00:00:00","2020-09-24 00:00:00",100 +123,11,91,"DBLFNC76M42Z110M","2010-01-11 00:00:00","2012-01-10 00:00:00",100 +124,11,91,"DBLFNC76M42Z110M","2012-02-01 00:00:00","2012-10-15 00:00:00",100 +125,11,91,"DBLFNC76M42Z110M","2012-10-16 00:00:00","2020-09-24 00:00:00",100 +126,26,92,"DBNMRC79R10A089M","2009-04-21 00:00:00","2011-12-20 00:00:00",100 +127,26,92,"DBNMRC79R10A089M","2012-01-10 00:00:00","2017-10-09 00:00:00",100 +129,5,94,"DCTRRT60E54G843L","2012-04-19 00:00:00",,0 +130,24,94,"DCTRRT60E54G843L","1988-06-01 00:00:00",,100 +136,3,99,"DGNFCT60R44C633V","1988-12-30 00:00:00","2012-02-29 00:00:00",100 +137,20,99,"DGNFCT60R44C633V","2012-03-01 00:00:00",,100 +138,18,100,"DLCRSR65T46G702U","2008-03-10 00:00:00","2012-01-09 00:00:00",100 +139,18,100,"DLCRSR65T46G702U","2012-01-30 00:00:00","2018-02-11 00:00:00",100 +149,16,107,"DMTFNC72S46H793U","2011-01-25 00:00:00","2012-03-07 00:00:00",10 +150,16,107,"DMTFNC72S46H793U","2012-03-28 00:00:00","2018-11-29 00:00:00",10 +151,26,107,"DMTFNC72S46H793U","2010-04-08 00:00:00","2012-03-07 00:00:00",90 +152,26,107,"DMTFNC72S46H793U","2012-03-28 00:00:00","2018-11-29 00:00:00",90 +153,9,108,"DNGCRN60L45L331L","1988-11-16 00:00:00","2014-03-02 00:00:00",100 +171,8,122,"DZZPRZ79S23B832W","2011-02-25 00:00:00","2013-02-24 00:00:00",100 +173,16,124,"FEOGTT76M57G702C","2011-01-25 00:00:00",,100 +176,14,127,"FLCBNL67C67G702J","2008-04-14 00:00:00",,100 +177,9,128,"FLCCHR79A50A657F","2011-01-20 00:00:00","2018-12-26 00:00:00",90 +178,16,128,"FLCCHR79A50A657F","2011-01-25 00:00:00","2018-12-26 00:00:00",10 +179,11,129,"FLCFRZ75H11G843N","2011-09-16 00:00:00","2020-09-24 00:00:00",100 +181,27,130,"FLCLNI64A62Z133E","2010-01-01 00:00:00","2018-12-26 00:00:00",100 +184,2,133,"FNTNRC56D03F935I","2015-11-04 00:00:00","2019-11-03 00:00:00",0 +185,25,133,"FNTNRC56D03F935I","1983-03-01 00:00:00",,100 +186,27,134,"FRFFNC64C13D086Z","2010-07-05 00:00:00","2013-07-04 00:00:00",100 +190,4,137,"FRRLSS82A18I726H","2011-06-14 00:00:00","2013-02-03 00:00:00",100 +191,1,138,"FRRMNL83M56G702N","2009-03-02 00:00:00","2012-11-05 00:00:00",100 +192,13,138,"FRRMNL83M56G702N","2012-11-06 00:00:00",,100 +199,22,144,"FSCGPP62L26H224R","2010-03-01 00:00:00","2012-04-29 00:00:00",100 +200,22,144,"FSCGPP62L26H224R","2012-04-30 00:00:00","2014-11-29 00:00:00",100 +208,7,149,"GHNGPP81S05B354D","2011-05-23 00:00:00","2013-05-22 00:00:00",100 +212,7,152,"GLSGLI83E10E379Y","2011-09-01 00:00:00","2018-12-26 00:00:00",100 +215,11,155,"GNNCLD68A12D960O","2008-04-14 00:00:00","2020-09-24 00:00:00",100 +216,12,156,"GNNFNC59P25G702N","1984-01-01 00:00:00","2015-03-31 00:00:00",100 +217,9,157,"GNNFSC58R44C303V","1990-07-01 00:00:00",,100 +218,18,158,"GNNSLV61T47G702O","1987-11-01 00:00:00",,100 +220,26,160,"GNVFBA71B10L424Q","2008-04-14 00:00:00",,100 +223,22,162,"GRGDNL79T61G482Z","2012-03-01 00:00:00","2018-06-14 00:00:00",100 +224,27,163,"GRLMHL81L07G628D","2010-11-15 00:00:00","2011-11-14 00:00:00",100 +225,27,163,"GRLMHL81L07G628D","2011-11-23 00:00:00","2017-05-22 00:00:00",100 +228,10,165,"GRRMRA75B63F839N","2011-03-01 00:00:00","2011-11-30 00:00:00",100 +229,10,165,"GRRMRA75B63F839N","2011-12-01 00:00:00",,100 +230,1,166,"GRSFBA82B13G702C","2011-04-15 00:00:00","2018-12-26 00:00:00",100 +232,27,168,"GTTLRT77R01D969N","2010-07-19 00:00:00",,100 +233,22,169,"KRGRNN69S00X000X","2001-12-31 00:00:00",,100 +249,11,184,"LLELCU79T12F870I","2011-06-01 00:00:00","2016-06-30 00:00:00",100 +251,23,186,"LMAGPP68M29G702D","2007-12-01 00:00:00",,100 +252,15,187,"LMBGNN59T09G702K","1986-11-17 00:00:00",,95 +253,25,187,"LMBGNN59T09G702K","1986-11-17 00:00:00",,5 +254,8,188,"LMBSFN76C46I045D","2010-09-03 00:00:00","2015-12-02 00:00:00",100 +258,19,192,"LNTFNC77P67B771D","2011-05-02 00:00:00","2012-02-29 00:00:00",100 +259,20,192,"LNTFNC77P67B771D","2012-03-01 00:00:00","2018-11-29 00:00:00",100 +260,7,193,"LPRBBR72T48E715G","2007-05-29 00:00:00","2020-05-07 00:00:00",100 +264,2,197,"LTLDGI59B09H224P","2008-12-01 00:00:00","2015-11-03 00:00:00",0 +265,4,197,"LTLDGI59B09H224P","1986-04-01 00:00:00",,100 +269,2,201,"MGHCRL56C02E783X","2008-12-01 00:00:00","2015-11-03 00:00:00",0 +270,11,201,"MGHCRL56C02E783X","1984-02-01 00:00:00","2020-09-24 00:00:00",100 +272,22,204,"MGRMSM66R01E715C","2010-10-18 00:00:00","2018-11-29 00:00:00",100 +276,5,208,"MLSDNL59T56E281S","2012-04-19 00:00:00","2018-11-29 00:00:00",0 +277,16,208,"MLSDNL59T56E281S","2011-01-25 00:00:00","2015-10-31 00:00:00",50 +278,19,208,"MLSDNL59T56E281S","2011-01-24 00:00:00","2012-02-29 00:00:00",50 +279,20,208,"MLSDNL59T56E281S","2012-03-01 00:00:00","2015-10-31 00:00:00",50 +284,13,211,"MNCLCU87M13G702W","2011-02-16 00:00:00",,100 +287,14,213,"MNDTZN68M67I422Z","2009-11-02 00:00:00",,100 +290,11,215,"MNGPLA70T22D612X","2011-09-16 00:00:00","2020-09-24 00:00:00",100 +300,19,223,"MRCDEA72M50B455T","2010-01-15 00:00:00","2012-02-29 00:00:00",100 +301,20,223,"MRCDEA72M50B455T","2012-03-01 00:00:00",,100 +311,22,230,"MRNDVD77B04E801X","2010-12-15 00:00:00",,100 +312,7,231,"MROGLI68H29E625F","2011-12-16 00:00:00",,100 +315,22,234,"MRTMSM67R30G702R","1997-03-17 00:00:00",,100 +318,4,236,"MSSMKI65D58Z126H","2008-04-14 00:00:00",,100 +319,22,237,"MSTDLG59R56E463J","1987-12-01 00:00:00",,100 +322,11,240,"MTAGPP68D15D976W","2001-12-21 00:00:00","2020-09-24 00:00:00",100 +324,4,242,"MZZFNC58T25E715W","1988-12-01 00:00:00",,100 +329,9,247,"NNNMRC73M11Z130N","2010-02-01 00:00:00",,100 +330,8,248,"NRDFNC80E05G491G","2008-06-01 00:00:00","2012-09-30 00:00:00",100 +331,8,248,"NRDFNC80E05G491G","2012-10-01 00:00:00","2016-12-29 00:00:00",100 +332,11,249,"NRDLSN75B22E625N","2009-03-02 00:00:00","2012-01-01 00:00:00",100 +333,11,249,"NRDLSN75B22E625N","2012-01-23 00:00:00","2018-12-26 00:00:00",100 +334,21,250,"NSLLCN57P27I608Z","1985-03-01 00:00:00",,100 +342,14,256,"PCCMRA63M50G702W","1988-12-30 00:00:00",,100 +343,11,257,"PCCTMS69D11G702V","2011-09-21 00:00:00","2016-12-15 00:00:00",95 +344,25,257,"PCCTMS69D11G702V","2011-09-21 00:00:00","2016-12-15 00:00:00",5 +346,10,259,"PDVCST61L63E625M","1987-08-01 00:00:00",,100 +349,5,261,"PGNPQL68A17F839V","2012-04-19 00:00:00",,0 +350,11,261,"PGNPQL68A17F839V","2009-02-16 00:00:00","2020-09-24 00:00:00",100 +351,6,262,"PLLLDN59D65D086U","2006-06-01 00:00:00","2015-02-28 00:00:00",100 +358,13,267,"PLTSRN73A70G702U","2009-02-16 00:00:00",,100 +359,26,268,"PNCFRC72B27L840F","2010-02-01 00:00:00",,100 +362,26,271,"PNGPLA69A10E506F","2010-09-01 00:00:00",,100 +372,21,280,"PRDCMN59P55B455B","2001-09-05 00:00:00",,100 +373,22,281,"PRDFNC77D55G702B","2010-03-01 00:00:00","2012-02-29 00:00:00",100 +374,22,281,"PRDFNC77D55G702B","2012-03-05 00:00:00","2018-12-26 00:00:00",100 +376,22,283,"PRDPLA72C02I608N","2001-12-28 00:00:00",,100 +379,8,286,"PRGRFL62A28A757Y","1988-11-16 00:00:00",,100 +380,22,287,"PRIGRL74E31G491R","2010-07-01 00:00:00","2012-12-31 00:00:00",100 +383,13,290,"PRTLSN69L49G702V","2010-02-01 00:00:00",,100 +389,7,295,"PTRFBA60P05L628J","1986-04-01 00:00:00",,100 +390,27,296,"PTRFNC63D04H224K","2001-09-04 00:00:00",,100 +393,1,298,"PVNLCU65P53F839H","1995-12-01 00:00:00",,100 +402,13,303,"RCCGNN58S30G702P","1988-12-31 00:00:00",,100 +415,9,315,"RNSCHR68R57L781C","2008-04-14 00:00:00","2014-10-05 00:00:00",100 +417,9,317,"RNZSVT76R29H163J","2010-07-01 00:00:00",,100 +426,11,323,"RTNMHL76B25A390M","2011-08-22 00:00:00","2018-01-31 00:00:00",100 +427,17,324,"RVLCLD56M41L219R","2011-01-25 00:00:00",,100 +428,11,325,"SBSFRZ60B15C372J","1988-04-01 00:00:00","2020-09-24 00:00:00",100 +431,26,328,"SCPRRT60L20H501Z","1986-12-01 00:00:00",,100 +432,22,329,"SCZNDR66L09E625Y","2012-08-01 00:00:00",,100 +435,11,331,"SLENDR77T26G702W","2011-09-16 00:00:00","2020-09-24 00:00:00",100 +437,22,333,"SLRMNL58S04A494N","1987-09-01 00:00:00",,100 +438,8,334,"SLVFRZ75R10A561S","2009-02-16 00:00:00",,100 +450,7,342,"SNTCML69R48F205S","2008-04-14 00:00:00",,100 +457,11,349,"SSNMSM82C05E625D","2010-04-15 00:00:00","2011-12-14 00:00:00",100 +458,11,349,"SSNMSM82C05E625D","2011-12-15 00:00:00","2016-06-30 00:00:00",100 +459,11,350,"STRMRT65P29Z133D","2001-09-17 00:00:00","2020-09-24 00:00:00",100 +461,11,352,"SVNPQL55S26L505N","2001-10-18 00:00:00","2020-09-24 00:00:00",100 +466,22,356,"TMPMRC81H17G702E","2011-02-16 00:00:00",,100 +471,22,359,"TNZNNA57M53E463D","1984-02-01 00:00:00",,100 +473,4,361,"TRBMCH72R07Z126L","2010-02-01 00:00:00",,100 +478,23,364,"TRNGLC68B27L117E","2009-02-11 00:00:00","2012-10-29 00:00:00",100 +479,23,364,"TRNGLC68B27L117E","2012-10-30 00:00:00",,100 +482,11,367,"TRPLCU65S05H501F","2012-02-15 00:00:00","2013-02-14 00:00:00",100 +483,9,368,"TRSRRT79M08M082U","2011-09-16 00:00:00",,100 +484,4,369,"TRVMLS69M70D612T","2011-08-01 00:00:00","2018-12-26 00:00:00",100 +492,5,376,"VLPFRC79R18G843Z","2012-04-19 00:00:00",,0 +493,25,376,"VLPFRC79R18G843Z","2011-02-16 00:00:00",,100 +507,5,388,"ZPPFNC56P29G702L","2012-04-19 00:00:00","2016-06-14 00:00:00",100 +508,11,388,"ZPPFNC56P29G702L","2011-09-21 00:00:00","2016-06-14 00:00:00",100 +511,1,391,"PNTMRC77T05G702B","2012-12-10 00:00:00","2016-11-30 00:00:00",100 +514,11,393,"DLLNDR68H16L833Z","2013-01-02 00:00:00","2014-02-02 00:00:00",100 +517,22,287,"PRIGRL74E31G491R","2012-12-31 00:00:00",,100 +518,8,122,"DZZPRZ79S23B832W","2012-12-31 00:00:00",,100 +520,7,149,"GHNGPP81S05B354D","2013-01-11 00:00:00",,100 +522,27,134,"FRFFNC64C13D086Z","2013-01-04 00:00:00",,100 +536,8,75,"CRLMNL81M13E463L","2013-02-01 00:00:00","2016-11-01 00:00:00",100 +542,11,404,"VRSLDN72D68I119V","2013-02-04 00:00:00","2016-03-03 00:00:00",100 +543,4,137,"FRRLSS82A18I726H","2013-02-04 00:00:00","2014-11-02 00:00:00",100 +564,11,367,"TRPLCU65S05H501F","2012-02-15 00:00:00","2016-02-24 00:00:00",100 +574,4,424,"CNCVCN77S01F052K","2013-04-22 00:00:00","2018-12-26 00:00:00",100 +601,11,78,"CROGPL80H25F839W","2013-09-02 00:00:00","2018-12-26 00:00:00",100 +602,11,269,"PNCGCR75S04L103G","2013-10-01 00:00:00","2017-10-16 00:00:00",100 +626,9,135,"FRLBBR76T43E625E","2013-12-02 00:00:00","2020-11-30 00:00:00",100 +647,11,393,"DLLNDR68H16L833Z","2014-02-03 00:00:00","2018-12-26 00:00:00",100 +648,11,269,"PNCGCR75S04L103G","2013-10-01 00:00:00","2014-09-30 00:00:00",100 +655,23,50,"CCOLSN70P02C351P","2014-01-03 00:00:00",,100 +657,27,108,"DNGCRN60L45L331L","2014-03-03 00:00:00","2015-12-31 00:00:00",100 +665,23,41,"BSCSLL74R54H163P","2014-05-05 00:00:00","2018-12-26 00:00:00",100 +672,22,307,"RGGMRC81H12L833E","2014-06-16 00:00:00","2018-12-26 00:00:00",100 +673,22,310,"RGHMRC75M21E625Z","2014-06-16 00:00:00","2018-12-26 00:00:00",100 +679,27,263,"PLMFPP81S18C975U","2014-07-01 00:00:00","2017-08-30 00:00:00",100 +706,8,315,"RNSCHR68R57L781C","2014-10-06 00:00:00",,100 +711,4,137,"FRRLSS82A18I726H","2014-11-03 00:00:00","2018-12-26 00:00:00",100 +727,22,511,"CRBNDR79C08G702H","2014-12-01 00:00:00","2017-11-07 00:00:00",100 +740,22,144,"FSCGPP62L26H224R","2015-01-08 00:00:00","2015-12-31 00:00:00",100 +749,26,265,"PLMGPL84C10I549Y","2015-01-26 00:00:00","2017-05-21 00:00:00",100 +750,4,345,"SPGGGR81B09I119V","2015-02-02 00:00:00","2018-12-26 00:00:00",50 +770,27,262,"PLLLDN59D65D086U","2015-03-01 00:00:00",,100 +779,28,156,"GNNFNC59P25G702N","2015-04-01 00:00:00","2021-01-31 00:00:00",100 +788,7,212,"MNCMRC82E25E004N","2015-05-04 00:00:00","2018-05-03 00:00:00",100 +799,11,172,"LBRSDR82P20I199Q","2015-06-10 00:00:00","2018-06-09 00:00:00",100 +804,11,25,"BRDLSS82D69C415P","2015-06-15 00:00:00","2018-06-14 00:00:00",100 +818,9,148,"GBRLNZ84R01G628X","2015-08-01 00:00:00","2017-12-26 00:00:00",100 +842,2,59,"CLNSRA74S49A485U","2015-11-04 00:00:00","2019-11-03 00:00:00",0 +843,2,129,"FLCFRZ75H11G843N","2015-11-04 00:00:00","2019-11-03 00:00:00",0 +844,2,165,"GRRMRA75B63F839N","2015-11-04 00:00:00","2019-11-03 00:00:00",0 +845,2,317,"RNZSVT76R29H163J","2015-11-04 00:00:00","2019-11-03 00:00:00",0 +846,2,361,"TRBMCH72R07Z126L","2015-11-04 00:00:00","2019-11-03 00:00:00",0 +856,11,397,"MLNNNA84B67A485C","2015-12-15 00:00:00","2018-02-14 00:00:00",100 +858,11,214,"MNGFNC83E18A512N","2016-01-04 00:00:00","2018-12-26 00:00:00",100 +866,2,328,"SCPRRT60L20H501Z","2019-04-01 00:00:00","2024-12-04 00:00:00",0 +867,16,208,"MLSDNL59T56E281S","2015-11-01 00:00:00","2018-11-29 00:00:00",50 +870,13,208,"MLSDNL59T56E281S","2015-11-01 00:00:00","2018-11-29 00:00:00",50 +878,22,144,"FSCGPP62L26H224R","2016-02-08 00:00:00","2018-12-26 00:00:00",100 +884,18,108,"DNGCRN60L45L331L","2016-01-01 00:00:00",,100 +890,11,367,"TRPLCU65S05H501F","2016-02-25 00:00:00","2018-12-26 00:00:00",100 +893,11,404,"VRSLDN72D68I119V","2016-03-04 00:00:00","2018-12-26 00:00:00",100 +900,11,141,"FRSLCU80H29D403A","2016-03-14 00:00:00","2018-12-26 00:00:00",100 +919,11,379,"VRACDF82P22H926O","2016-05-16 00:00:00","2018-12-26 00:00:00",100 +929,9,464,"RMNVTR73C05H224X","2016-06-15 00:00:00","2018-12-26 00:00:00",100 +932,11,184,"LLELCU79T12F870I","2016-07-01 00:00:00","2018-12-26 00:00:00",100 +935,11,349,"SSNMSM82C05E625D","2016-07-01 00:00:00","2018-12-26 00:00:00",100 +936,5,388,"ZPPFNC56P29G702L","2016-06-15 00:00:00","2018-12-26 00:00:00",10 +937,5,76,"CRLRRT81E29L628N","2016-06-15 00:00:00","2017-06-14 00:00:00",100 +939,10,495,"PLLDNL76D26E715E","2016-07-15 00:00:00","2016-10-23 00:00:00",100 +940,11,76,"CRLRRT81E29L628N","2016-06-15 00:00:00","2018-12-26 00:00:00",100 +959,27,1,"BCCFCM82C17A285V","2016-10-03 00:00:00","2018-12-26 00:00:00",100 +968,10,495,"PLLDNL76D26E715E","2016-10-24 00:00:00","2018-11-29 00:00:00",100 +970,8,75,"CRLMNL81M13E463L","2016-11-02 00:00:00","2018-12-26 00:00:00",100 +971,27,436,"LRSDVD86P28E463U","2016-11-02 00:00:00","2018-12-26 00:00:00",100 +985,13,391,"PNTMRC77T05G702B","2016-12-01 00:00:00",,100 +991,11,13,"BLTPLA71L20F052H","2020-01-01 00:00:00","2020-09-24 00:00:00",100 +992,11,257,"PCCTMS69D11G702V","2016-12-16 00:00:00","2020-09-24 00:00:00",100 +993,25,257,"PCCTMS69D11G702V","2016-12-16 00:00:00",,5 +994,25,623,"BRNPLA73T13G843E","2017-01-02 00:00:00","2019-12-09 00:00:00",50 +998,27,623,"BRNPLA73T13G843E","2017-01-02 00:00:00","2019-12-09 00:00:00",50 +999,8,248,"NRDFNC80E05G491G","2016-12-30 00:00:00",,100 +1009,11,341,"SNBFBA84A05B354L","2017-05-02 00:00:00","2018-05-01 00:00:00",100 +1041,11,388,"ZPPFNC56P29G702L","2016-06-15 00:00:00","2018-12-26 00:00:00",90 +1060,27,318,"RSSDRA76D18E625D","2017-04-03 00:00:00","2018-12-26 00:00:00",100 +1064,26,20,"BNTFNC82S30F861R","2017-04-10 00:00:00","2018-12-26 00:00:00",100 +1069,11,341,"SNBFBA84A05B354L","2017-05-02 00:00:00","2018-12-26 00:00:00",100 +1070,26,207,"MLMLGU85H08C352Q","2017-05-02 00:00:00","2019-05-01 00:00:00",100 +1079,26,265,"PLMGPL84C10I549Y","2017-05-22 00:00:00","2018-12-26 00:00:00",100 +1082,27,163,"GRLMHL81L07G628D","2017-05-23 00:00:00","2018-05-22 00:00:00",100 +1087,22,291,"PSCMNT81R46D862B","2017-06-05 00:00:00","2018-12-26 00:00:00",100 +1112,27,263,"PLMFPP81S18C975U","2017-08-31 00:00:00","2018-12-26 00:00:00",100 +1123,11,38,"BRTVNT80R55G702I","2017-09-11 00:00:00","2018-12-26 00:00:00",100 +1135,11,269,"PNCGCR75S04L103G","2017-10-17 00:00:00","2018-12-26 00:00:00",100 +1136,25,449,"LPRGPP86D30D423E","2017-10-17 00:00:00","2018-12-26 00:00:00",100 +1144,9,278,"PPPLCU84E05H703D","2017-11-06 00:00:00","2019-06-30 00:00:00",100 +1146,11,227,"MRFLND85P11Z131I","2017-11-02 00:00:00","2018-12-26 00:00:00",100 +1148,22,511,"CRBNDR79C08G702H","2017-11-08 00:00:00","2018-12-26 00:00:00",100 +1163,11,370,"TZRCLD83E02B354K","2017-12-01 00:00:00","2018-12-26 00:00:00",100 +1164,11,615,"BGLMRM73P70D583A","2017-12-01 00:00:00","2018-12-26 00:00:00",100 +1180,23,345,"SPGGGR81B09I119V","2015-02-02 00:00:00","2018-12-26 00:00:00",50 +1181,9,148,"GBRLNZ84R01G628X","2017-12-28 00:00:00","2018-01-31 00:00:00",100 +1207,11,323,"RTNMHL76B25A390M","2018-02-01 00:00:00","2020-09-24 00:00:00",100 +1213,29,100,"DLCRSR65T46G702U","2018-02-12 00:00:00","2018-12-26 00:00:00",100 +1214,29,100,"DLCRSR65T46G702U","2018-02-12 12:24:01","2018-12-26 00:00:00",100 +1215,11,453,"VDCLCU87A64D086T","2018-02-15 00:00:00","2019-06-30 00:00:00",100 +1216,18,397,"MLNNNA84B67A485C","2018-02-15 00:00:00","2018-12-26 00:00:00",100 +1219,26,714,"MNTLSN90B02H856X","2018-02-21 00:00:00","2019-03-20 00:00:00",100 +1235,22,431,"CDACLD78D61G687Y","2018-04-03 00:00:00","2019-09-29 00:00:00",100 +1236,26,389,"PTNMRC81L10A269S","2018-03-01 00:00:00","2019-09-29 00:00:00",100 +1250,7,212,"MNCMRC82E25E004N","2018-05-04 00:00:00","2018-12-26 00:00:00",100 +1257,11,172,"LBRSDR82P20I199Q","2018-06-11 00:00:00","2018-12-26 00:00:00",100 +1259,26,162,"GRGDNL79T61G482Z","2018-06-15 00:00:00",,100 +1260,11,25,"BRDLSS82D69C415P","2018-06-15 00:00:00","2018-12-26 00:00:00",100 +1273,8,222,"MNTCST85P59Z129X","2018-09-10 00:00:00","2019-06-30 00:00:00",100 +1281,16,29,"BRRFNC71M61G702S","2011-06-01 00:00:00",,100 +1299,10,495,"PLLDNL76D26E715E","2018-11-30 00:00:00",,100 +1300,20,192,"LNTFNC77P67B771D","2018-11-30 00:00:00",,100 +1301,13,208,"MLSDNL59T56E281S","2018-11-30 00:00:00","2020-02-29 00:00:00",50 +1302,16,208,"MLSDNL59T56E281S","2018-11-30 00:00:00","2020-02-29 00:00:00",50 +1303,11,42,"BSOCHR64R51Z110U","2018-11-30 00:00:00","2020-09-24 00:00:00",90 +1304,16,42,"BSOCHR64R51Z110U","2018-11-30 00:00:00",,10 +1305,22,204,"MGRMSM66R01E715C","2018-11-30 00:00:00",,100 +1306,16,107,"DMTFNC72S46H793U","2018-11-30 00:00:00","2020-09-20 00:00:00",10 +1307,26,107,"DMTFNC72S46H793U","2018-11-30 00:00:00","2020-09-20 00:00:00",90 +1311,8,366,"TRNSVT83D12E396M","2018-02-15 00:00:00","2019-02-14 00:00:00",100 +1316,9,319,"RSSGLI84R06E202M","2018-11-26 00:00:00","2018-12-27 00:00:00",100 +1320,11,92,"DBNMRC79R10A089M","2018-11-30 00:00:00","2020-09-24 00:00:00",100 +1330,11,370,"TZRCLD83E02B354K","2018-12-27 00:00:00","2020-09-24 00:00:00",100 +1331,11,615,"BGLMRM73P70D583A","2018-12-27 00:00:00","2020-09-24 00:00:00",100 +1332,27,1,"BCCFCM82C17A285V","2018-12-27 00:00:00",,100 +1333,11,25,"BRDLSS82D69C415P","2018-12-27 00:00:00","2020-09-24 00:00:00",100 +1334,11,38,"BRTVNT80R55G702I","2018-12-27 00:00:00","2020-09-24 00:00:00",100 +1335,22,511,"CRBNDR79C08G702H","2018-12-27 00:00:00",,100 +1336,8,75,"CRLMNL81M13E463L","2018-12-27 00:00:00",,100 +1337,27,87,"CSSPTR79C05G273U","2018-12-27 00:00:00",,100 +1338,11,76,"CRLRRT81E29L628N","2018-12-27 00:00:00","2020-09-24 00:00:00",100 +1339,11,141,"FRSLCU80H29D403A","2018-12-27 00:00:00","2020-09-24 00:00:00",100 +1340,11,172,"LBRSDR82P20I199Q","2018-12-27 00:00:00","2020-09-24 00:00:00",100 +1341,27,436,"LRSDVD86P28E463U","2018-12-27 00:00:00",,100 +1342,25,449,"LPRGPP86D30D423E","2018-12-27 00:00:00",,100 +1343,11,214,"MNGFNC83E18A512N","2018-12-27 00:00:00","2020-09-24 00:00:00",100 +1344,7,212,"MNCMRC82E25E004N","2018-12-27 00:00:00",,100 +1345,18,397,"MLNNNA84B67A485C","2018-12-27 00:00:00",,100 +1346,11,227,"MRFLND85P11Z131I","2018-12-27 00:00:00","2020-09-24 00:00:00",100 +1347,26,265,"PLMGPL84C10I549Y","2018-12-27 00:00:00",,100 +1348,22,291,"PSCMNT81R46D862B","2018-12-27 00:00:00",,100 +1349,9,464,"RMNVTR73C05H224X","2018-12-27 00:00:00",,100 +1350,27,318,"RSSDRA76D18E625D","2018-12-27 00:00:00",,100 +1351,11,341,"SNBFBA84A05B354L","2018-12-27 00:00:00","2020-09-24 00:00:00",100 +1352,23,345,"SPGGGR81B09I119V","2018-12-27 00:00:00",,50 +1353,11,379,"VRACDF82P22H926O","2018-12-27 00:00:00","2020-09-24 00:00:00",100 +1354,11,349,"SSNMSM82C05E625D","2018-12-27 00:00:00","2020-09-24 00:00:00",100 +1355,26,20,"BNTFNC82S30F861R","2018-12-27 00:00:00",,100 +1356,23,41,"BSCSLL74R54H163P","2018-12-27 00:00:00","2019-10-22 00:00:00",100 +1357,20,56,"CLBNNL81C05H224R","2018-12-27 00:00:00",,100 +1358,4,424,"CNCVCN77S01F052K","2018-12-27 00:00:00",,100 +1359,11,78,"CROGPL80H25F839W","2018-12-27 00:00:00","2020-09-24 00:00:00",100 +1360,11,393,"DLLNDR68H16L833Z","2018-12-27 00:00:00","2020-09-24 00:00:00",100 +1361,29,100,"DLCRSR65T46G702U","2018-12-27 00:00:00",,100 +1362,17,130,"FLCLNI64A62Z133E","2018-12-27 00:00:00","2020-01-07 00:00:00",50 +1363,4,137,"FRRLSS82A18I726H","2018-12-27 00:00:00",,100 +1364,22,144,"FSCGPP62L26H224R","2018-12-27 00:00:00",,100 +1365,7,152,"GLSGLI83E10E379Y","2018-12-27 00:00:00","2020-05-14 00:00:00",100 +1366,1,166,"GRSFBA82B13G702C","2018-12-27 00:00:00",,100 +1367,11,184,"LLELCU79T12F870I","2018-12-27 00:00:00","2020-09-24 00:00:00",100 +1368,11,269,"PNCGCR75S04L103G","2018-12-27 00:00:00","2020-09-24 00:00:00",100 +1369,22,281,"PRDFNC77D55G702B","2018-12-27 00:00:00",,100 +1370,22,307,"RGGMRC81H12L833E","2018-12-27 00:00:00",,100 +1371,22,310,"RGHMRC75M21E625Z","2018-12-27 00:00:00",,100 +1372,4,369,"TRVMLS69M70D612T","2018-12-27 00:00:00","2020-02-29 00:00:00",100 +1373,11,367,"TRPLCU65S05H501F","2018-12-27 00:00:00","2020-09-24 00:00:00",100 +1374,11,388,"ZPPFNC56P29G702L","2018-12-27 00:00:00","2020-09-24 00:00:00",100 +1375,11,404,"VRSLDN72D68I119V","2018-12-27 00:00:00","2019-10-22 00:00:00",100 +1376,9,128,"FLCCHR79A50A657F","2018-12-27 00:00:00","2019-10-22 00:00:00",90 +1377,16,128,"FLCCHR79A50A657F","2018-12-27 00:00:00","2020-09-20 00:00:00",10 +1378,11,249,"NRDLSN75B22E625N","2018-12-27 00:00:00","2020-09-24 00:00:00",100 +1381,18,188,"LMBSFN76C46I045D","2018-11-30 00:00:00",,100 +1382,27,163,"GRLMHL81L07G628D","2018-12-27 00:00:00",,100 +1387,9,148,"GBRLNZ84R01G628X","2018-12-27 00:00:00","2019-10-22 00:00:00",100 +1388,27,263,"PLMFPP81S18C975U","2018-12-27 00:00:00",,100 +1389,9,319,"RSSGLI84R06E202M","2018-12-28 00:00:00",,100 +1406,4,345,"SPGGGR81B09I119V","2018-12-27 00:00:00",,50 +1416,27,130,"FLCLNI64A62Z133E","2018-12-27 00:00:00","2020-01-07 00:00:00",50 +1417,30,261,"PGNPQL68A17F839V","2019-01-02 00:00:00",,0 +1418,30,349,"SSNMSM82C05E625D","2019-01-02 00:00:00",,0 +1419,30,66,"CNDLRD76B13A399L","2019-01-02 00:00:00",,0 +1421,30,88,"CSTDTL57L65A864M","2019-01-02 00:00:00",,0 +1422,30,76,"CRLRRT81E29L628N","2019-01-02 00:00:00",,0 +1423,30,78,"CROGPL80H25F839W","2019-01-02 00:00:00",,0 +1424,30,393,"DLLNDR68H16L833Z","2019-01-02 00:00:00",,0 +1425,30,141,"FRSLCU80H29D403A","2019-01-02 00:00:00",,0 +1426,30,184,"LLELCU79T12F870I","2019-01-02 00:00:00",,0 +1427,30,214,"MNGFNC83E18A512N","2019-01-02 00:00:00",,0 +1428,30,269,"PNCGCR75S04L103G","2019-01-02 00:00:00",,0 +1429,30,341,"SNBFBA84A05B354L","2019-01-02 00:00:00",,0 +1430,31,257,"PCCTMS69D11G702V","2018-09-12 00:00:00",,0 +1431,31,91,"DBLFNC76M42Z110M","2018-09-12 00:00:00",,0 +1432,31,367,"TRPLCU65S05H501F","2018-09-12 00:00:00","2020-07-02 00:00:00",0 +1433,31,152,"GLSGLI83E10E379Y","2018-09-12 00:00:00","2020-07-02 00:00:00",0 +1434,31,393,"DLLNDR68H16L833Z","2018-09-12 00:00:00",,0 +1438,8,366,"TRNSVT83D12E396M","2019-02-15 00:00:00","2020-06-30 00:00:00",100 +1445,30,257,"PCCTMS69D11G702V","2019-01-02 00:00:00",,0 +1453,26,714,"MNTLSN90B02H856X","2019-03-21 00:00:00","2019-06-01 00:00:00",100 +1459,26,351,"STTLNE73R57I726Q","2019-04-15 00:00:00","2019-09-29 00:00:00",100 +1460,2,168,"GTTLRT77R01D969N","2019-04-01 00:00:00","2019-11-03 00:00:00",0 +1464,11,219,"MNNNDR84C27G702H","2019-04-15 00:00:00","2019-10-08 00:00:00",100 +1471,26,207,"MLMLGU85H08C352Q","2019-05-02 00:00:00","2019-09-04 00:00:00",100 +1493,9,278,"PPPLCU84E05H703D","2019-07-01 00:00:00",,100 +1510,27,472,"CRVNNN87D20D423F","2019-08-30 00:00:00","2021-12-31 00:00:00",100 +1518,26,207,"MLMLGU85H08C352Q","2019-09-05 00:00:00","2020-06-30 00:00:00",100 +1524,26,389,"PTNMRC81L10A269S","2019-09-30 00:00:00",,100 +1525,22,431,"CDACLD78D61G687Y","2019-09-30 00:00:00",,100 +1532,11,219,"MNNNDR84C27G702H","2019-10-09 00:00:00","2020-09-24 00:00:00",100 +1540,7,531,"PRVPVN84M42Z224D","2019-11-01 00:00:00","2020-03-31 00:00:00",100 +1542,26,714,"MNTLSN90B02H856X","2019-11-04 00:00:00","2021-11-03 00:00:00",100 +1544,9,167,"GRSVLR77L18B832G","2019-11-05 00:00:00","2020-11-30 00:00:00",100 +1554,23,41,"BSCSLL74R54H163P","2019-10-23 00:00:00",,50 +1555,14,41,"BSCSLL74R54H163P","2019-10-23 00:00:00",,50 +1556,9,128,"FLCCHR79A50A657F","2019-10-23 00:00:00","2020-09-20 00:00:00",40 +1557,13,128,"FLCCHR79A50A657F","2019-10-23 00:00:00",,50 +1558,11,404,"VRSLDN72D68I119V","2019-10-23 00:00:00","2020-09-24 00:00:00",50 +1559,13,404,"VRSLDN72D68I119V","2019-10-23 00:00:00",,50 +1562,8,148,"GBRLNZ84R01G628X","2019-10-23 00:00:00",,100 +1577,27,623,"BRNPLA73T13G843E","2019-12-10 00:00:00",,100 +1586,8,222,"MNTCST85P59Z129X","2019-07-01 00:00:00",,100 +1606,11,691,"LZZMME81E62E463W","2019-12-03 00:00:00","2020-09-24 00:00:00",100 +1628,11,453,"VDCLCU87A64D086T","2019-06-01 00:00:00","2020-09-24 00:00:00",100 +1635,26,351,"STTLNE73R57I726Q","2019-09-30 00:00:00",,100 +1640,7,531,"PRVPVN84M42Z224D","2020-04-01 00:00:00","2021-03-31 00:00:00",100 +1653,8,366,"TRNSVT83D12E396M","2020-07-01 00:00:00",,100 +1654,11,85,"CSNGNN79R29G843T","2020-07-01 00:00:00","2020-09-24 00:00:00",100 +1655,16,208,"MLSDNL59T56E281S","2020-03-01 00:00:00",,100 +1656,22,193,"LPRBBR72T48E715G","2020-05-08 00:00:00",,100 +1657,17,130,"FLCLNI64A62Z133E","2020-01-08 00:00:00",,100 +1658,13,152,"GLSGLI83E10E379Y","2020-05-15 00:00:00",,50 +1659,22,152,"GLSGLI83E10E379Y","2020-05-15 00:00:00",,50 +1660,26,207,"MLMLGU85H08C352Q","2020-07-01 00:00:00",,100 +1680,22,518,"LNEGPP74E22L113C","2020-07-30 00:00:00",,100 +1681,2,240,"MTAGPP68D15D976W","2019-12-04 00:00:00","2023-12-03 00:00:00",0 +1682,2,127,"FLCBNL67C67G702J","2019-12-04 00:00:00","2023-12-03 00:00:00",0 +1683,2,162,"GRGDNL79T61G482Z","2019-12-04 00:00:00","2023-12-03 00:00:00",0 +1684,2,248,"NRDFNC80E05G491G","2019-12-04 00:00:00","2023-12-03 00:00:00",0 +1685,2,261,"PGNPQL68A17F839V","2019-12-04 00:00:00","2023-12-03 00:00:00",0 +1686,2,283,"PRDPLA72C02I608N","2019-12-04 00:00:00","2023-12-03 00:00:00",0 +1687,2,345,"SPGGGR81B09I119V","2019-12-04 00:00:00","2023-12-03 00:00:00",0 +1690,13,369,"TRVMLS69M70D612T","2020-03-01 00:00:00",,50 +1691,4,369,"TRVMLS69M70D612T","2020-03-01 00:00:00",,50 +1692,14,128,"FLCCHR79A50A657F","2020-09-21 00:00:00",,50 +1693,26,107,"DMTFNC72S46H793U","2020-09-21 00:00:00",,50 +1694,14,107,"DMTFNC72S46H793U","2020-09-21 00:00:00",,50 +1778,32,240,"MTAGPP68D15D976W","2020-09-25 00:00:00",,100 +1779,33,323,"RTNMHL76B25A390M","2020-09-25 00:00:00",,100 +1780,33,349,"SSNMSM82C05E625D","2020-09-25 00:00:00",,100 +1781,33,370,"TZRCLD83E02B354K","2020-09-25 00:00:00",,100 +1782,33,615,"BGLMRM73P70D583A","2020-09-25 00:00:00",,100 +1784,33,25,"BRDLSS82D69C415P","2020-09-25 00:00:00",,100 +1785,32,38,"BRTVNT80R55G702I","2020-09-25 00:00:00",,100 +1788,32,13,"BLTPLA71L20F052H","2020-09-25 00:00:00",,100 +1789,33,42,"BSOCHR64R51Z110U","2020-09-25 00:00:00",,90 +1790,33,66,"CNDLRD76B13A399L","2020-09-25 00:00:00",,100 +1794,33,85,"CSNGNN79R29G843T","2020-09-25 00:00:00",,100 +1795,33,88,"CSTDTL57L65A864M","2020-09-25 00:00:00",,100 +1798,33,76,"CRLRRT81E29L628N","2020-09-25 00:00:00",,100 +1799,32,64,"CNCCSR65S20E058O","2020-09-25 00:00:00",,100 +1802,33,78,"CROGPL80H25F839W","2020-09-25 00:00:00",,100 +1804,33,91,"DBLFNC76M42Z110M","2020-09-25 00:00:00",,100 +1805,33,393,"DLLNDR68H16L833Z","2020-09-25 00:00:00",,100 +1806,32,92,"DBNMRC79R10A089M","2020-09-25 00:00:00",,100 +1807,32,331,"SLENDR77T26G702W","2020-09-25 00:00:00",,100 +1808,32,129,"FLCFRZ75H11G843N","2020-09-25 00:00:00",,100 +1809,33,141,"FRSLCU80H29D403A","2020-09-25 00:00:00",,100 +1810,32,155,"GNNCLD68A12D960O","2020-09-25 00:00:00",,100 +1811,33,172,"LBRSDR82P20I199Q","2020-09-25 00:00:00",,100 +1813,33,691,"LZZMME81E62E463W","2020-09-25 00:00:00","2021-02-28 00:00:00",100 +1814,33,184,"LLELCU79T12F870I","2020-09-25 00:00:00",,100 +1816,33,215,"MNGPLA70T22D612X","2020-09-25 00:00:00",,100 +1817,33,214,"MNGFNC83E18A512N","2020-09-25 00:00:00",,100 +1818,33,219,"MNNNDR84C27G702H","2020-09-25 00:00:00","2022-01-08 00:00:00",100 +1820,32,201,"MGHCRL56C02E783X","2020-09-25 00:00:00",,100 +1824,32,227,"MRFLND85P11Z131I","2020-09-25 00:00:00",,100 +1826,32,249,"NRDLSN75B22E625N","2020-09-25 00:00:00",,100 +1828,33,261,"PGNPQL68A17F839V","2020-09-25 00:00:00",,100 +1829,33,269,"PNCGCR75S04L103G","2020-09-25 00:00:00",,100 +1833,33,257,"PCCTMS69D11G702V","2020-09-25 00:00:00",,100 +1836,32,352,"SVNPQL55S26L505N","2020-09-25 00:00:00",,100 +1837,32,325,"SBSFRZ60B15C372J","2020-09-25 00:00:00",,100 +1838,33,341,"SNBFBA84A05B354L","2020-09-25 00:00:00",,100 +1839,33,350,"STRMRT65P29Z133D","2020-09-25 00:00:00",,100 +1841,32,367,"TRPLCU65S05H501F","2020-09-25 00:00:00",,100 +1842,32,453,"VDCLCU87A64D086T","2020-09-25 00:00:00",,100 +1843,32,379,"VRACDF82P22H926O","2020-09-25 00:00:00",,100 +1844,33,404,"VRSLDN72D68I119V","2020-09-25 00:00:00",,50 +1845,33,388,"ZPPFNC56P29G702L","2020-09-25 00:00:00",,100 +1853,9,167,"GRSVLR77L18B832G","2020-12-01 00:00:00",,100 +1854,14,135,"FRLBBR76T43E625E","2020-12-01 00:00:00",,100 +1917,23,156,"GNNFNC59P25G702N","2021-02-01 00:00:00",,100 diff --git a/src/test/resources/it/cnr/isti/epasmed/web/rest/epas/affiliazioniRepartiISTIRimanenti.csv b/src/test/resources/it/cnr/isti/epasmed/web/rest/epas/affiliazioniRepartiISTIRimanenti.csv new file mode 100644 index 0000000..87f661e --- /dev/null +++ b/src/test/resources/it/cnr/isti/epasmed/web/rest/epas/affiliazioniRepartiISTIRimanenti.csv @@ -0,0 +1,71 @@ +"rel_pers_reparto_codice","t_reparto_codice","t_personale_codice","t_personale_codice_fiscale","rel_pers_reparto_data_inizio","rel_pers_reparto_data_fine","rel_pers_reparto_perc" +1628,11,453,"VDCLCU87A64D086T","2019-06-01 00:00:00","2020-09-24 00:00:00",100 +1635,26,351,"STTLNE73R57I726Q","2019-09-30 00:00:00",,100 +1640,7,531,"PRVPVN84M42Z224D","2020-04-01 00:00:00","2021-03-31 00:00:00",100 +1653,8,366,"TRNSVT83D12E396M","2020-07-01 00:00:00",,100 +1654,11,85,"CSNGNN79R29G843T","2020-07-01 00:00:00","2020-09-24 00:00:00",100 +1655,16,208,"MLSDNL59T56E281S","2020-03-01 00:00:00",,100 +1656,22,193,"LPRBBR72T48E715G","2020-05-08 00:00:00",,100 +1657,17,130,"FLCLNI64A62Z133E","2020-01-08 00:00:00",,100 +1658,13,152,"GLSGLI83E10E379Y","2020-05-15 00:00:00",,50 +1659,22,152,"GLSGLI83E10E379Y","2020-05-15 00:00:00",,50 +1660,26,207,"MLMLGU85H08C352Q","2020-07-01 00:00:00",,100 +1680,22,518,"LNEGPP74E22L113C","2020-07-30 00:00:00",,100 +1681,2,240,"MTAGPP68D15D976W","2019-12-04 00:00:00","2023-12-03 00:00:00",0 +1682,2,127,"FLCBNL67C67G702J","2019-12-04 00:00:00","2023-12-03 00:00:00",0 +1683,2,162,"GRGDNL79T61G482Z","2019-12-04 00:00:00","2023-12-03 00:00:00",0 +1684,2,248,"NRDFNC80E05G491G","2019-12-04 00:00:00","2023-12-03 00:00:00",0 +1685,2,261,"PGNPQL68A17F839V","2019-12-04 00:00:00","2023-12-03 00:00:00",0 +1686,2,283,"PRDPLA72C02I608N","2019-12-04 00:00:00","2023-12-03 00:00:00",0 +1687,2,345,"SPGGGR81B09I119V","2019-12-04 00:00:00","2023-12-03 00:00:00",0 +1690,13,369,"TRVMLS69M70D612T","2020-03-01 00:00:00",,50 +1691,4,369,"TRVMLS69M70D612T","2020-03-01 00:00:00",,50 +1692,14,128,"FLCCHR79A50A657F","2020-09-21 00:00:00",,50 +1693,26,107,"DMTFNC72S46H793U","2020-09-21 00:00:00",,50 +1694,14,107,"DMTFNC72S46H793U","2020-09-21 00:00:00",,50 +1778,32,240,"MTAGPP68D15D976W","2020-09-25 00:00:00",,100 +1779,33,323,"RTNMHL76B25A390M","2020-09-25 00:00:00",,100 +1780,33,349,"SSNMSM82C05E625D","2020-09-25 00:00:00",,100 +1781,33,370,"TZRCLD83E02B354K","2020-09-25 00:00:00",,100 +1782,33,615,"BGLMRM73P70D583A","2020-09-25 00:00:00",,100 +1784,33,25,"BRDLSS82D69C415P","2020-09-25 00:00:00",,100 +1785,32,38,"BRTVNT80R55G702I","2020-09-25 00:00:00",,100 +1788,32,13,"BLTPLA71L20F052H","2020-09-25 00:00:00",,100 +1789,33,42,"BSOCHR64R51Z110U","2020-09-25 00:00:00",,90 +1790,33,66,"CNDLRD76B13A399L","2020-09-25 00:00:00",,100 +1794,33,85,"CSNGNN79R29G843T","2020-09-25 00:00:00",,100 +1795,33,88,"CSTDTL57L65A864M","2020-09-25 00:00:00",,100 +1798,33,76,"CRLRRT81E29L628N","2020-09-25 00:00:00",,100 +1799,32,64,"CNCCSR65S20E058O","2020-09-25 00:00:00",,100 +1802,33,78,"CROGPL80H25F839W","2020-09-25 00:00:00",,100 +1804,33,91,"DBLFNC76M42Z110M","2020-09-25 00:00:00",,100 +1805,33,393,"DLLNDR68H16L833Z","2020-09-25 00:00:00",,100 +1806,32,92,"DBNMRC79R10A089M","2020-09-25 00:00:00",,100 +1807,32,331,"SLENDR77T26G702W","2020-09-25 00:00:00",,100 +1808,32,129,"FLCFRZ75H11G843N","2020-09-25 00:00:00",,100 +1809,33,141,"FRSLCU80H29D403A","2020-09-25 00:00:00",,100 +1810,32,155,"GNNCLD68A12D960O","2020-09-25 00:00:00",,100 +1811,33,172,"LBRSDR82P20I199Q","2020-09-25 00:00:00",,100 +1813,33,691,"LZZMME81E62E463W","2020-09-25 00:00:00","2021-02-28 00:00:00",100 +1814,33,184,"LLELCU79T12F870I","2020-09-25 00:00:00",,100 +1816,33,215,"MNGPLA70T22D612X","2020-09-25 00:00:00",,100 +1817,33,214,"MNGFNC83E18A512N","2020-09-25 00:00:00",,100 +1818,33,219,"MNNNDR84C27G702H","2020-09-25 00:00:00","2022-01-08 00:00:00",100 +1820,32,201,"MGHCRL56C02E783X","2020-09-25 00:00:00",,100 +1824,32,227,"MRFLND85P11Z131I","2020-09-25 00:00:00",,100 +1826,32,249,"NRDLSN75B22E625N","2020-09-25 00:00:00",,100 +1828,33,261,"PGNPQL68A17F839V","2020-09-25 00:00:00",,100 +1829,33,269,"PNCGCR75S04L103G","2020-09-25 00:00:00",,100 +1833,33,257,"PCCTMS69D11G702V","2020-09-25 00:00:00",,100 +1836,32,352,"SVNPQL55S26L505N","2020-09-25 00:00:00",,100 +1837,32,325,"SBSFRZ60B15C372J","2020-09-25 00:00:00",,100 +1838,33,341,"SNBFBA84A05B354L","2020-09-25 00:00:00",,100 +1839,33,350,"STRMRT65P29Z133D","2020-09-25 00:00:00",,100 +1841,32,367,"TRPLCU65S05H501F","2020-09-25 00:00:00",,100 +1842,32,453,"VDCLCU87A64D086T","2020-09-25 00:00:00",,100 +1843,32,379,"VRACDF82P22H926O","2020-09-25 00:00:00",,100 +1844,33,404,"VRSLDN72D68I119V","2020-09-25 00:00:00",,50 +1845,33,388,"ZPPFNC56P29G702L","2020-09-25 00:00:00",,100 +1853,9,167,"GRSVLR77L18B832G","2020-12-01 00:00:00",,100 +1854,14,135,"FRLBBR76T43E625E","2020-12-01 00:00:00",,100 +1917,23,156,"GNNFNC59P25G702N","2021-02-01 00:00:00",,100 diff --git a/src/test/resources/it/cnr/isti/epasmed/web/rest/epas/personale.csv b/src/test/resources/it/cnr/isti/epasmed/web/rest/epas/personale.csv new file mode 100644 index 0000000..af64caa --- /dev/null +++ b/src/test/resources/it/cnr/isti/epasmed/web/rest/epas/personale.csv @@ -0,0 +1,247 @@ +"t_personale_codice","t_personale_cognome","t_personale_nome","t_personale_matricola","t_personale_codice_fiscale","t_personale_telefono","t_personale_data_nascita" +170,"Aloia","Nicola",1315,"LAONCL52L30F784C","050 6212997","1952-07-30 00:00:00" +240,"Amato","Giuseppe",1688,"MTAGPP68D15D976W","050 6212906","1968-04-15 00:00:00" +213,"Amendola","Tiziana",11529,"MNDTZN68M67I422Z","050 6212812","1968-08-27 00:00:00" +243,"Andreini","Caterina",,"NDRCRN84L69G843O",,"1984-07-29 00:00:00" +250,"Anselmo","Luciano",1456,"NSLLCN57P27I608Z","050 6212952","1957-09-27 00:00:00" +756,"Arcari","Sara",,"RCRSRA93R17B832V","050 6213374","1993-10-07 00:00:00" +618,"Argirò","Catia",,"RGRCTA89A53L063A",,"1989-01-13 00:00:00" +323,"Artini","Michele",14242,"RTNMHL76B25A390M","050 6212802","1976-02-25 00:00:00" +349,"Assante","Massimiliano",14540,"SSNMSM82C05E625D","050 6212908","1982-03-05 00:00:00" +370,"Atzori","Claudio",17560,"TZRCLD83E02B354K","050 6212899","1983-05-02 00:00:00" +1,"Bacco","Felice Manlio",17245,"BCCFCM82C17A285V","050 6212887","1982-03-17 00:00:00" +615,"Baglioni","Miriam",17559,"BGLMRM73P70D583A","050 6212899","1973-09-30 00:00:00" +20,"Banterle","Francesco",11665,"BNTFNC82S30F861R","050 6212940","1982-11-30 00:00:00" +26,"Baraglia","Ranieri",2863,"BRGRNR49H19L702U","050 6212994","1949-06-19 00:00:00" +25,"Bardi","Alessia",15923,"BRDLSS82D69C415P","050 6212899","1982-04-29 00:00:00" +623,"Baronti","Paolo",14351,"BRNPLA73T13G843E","050 6218208","1973-12-13 00:00:00" +31,"Barsocchi","Paolo",11902,"BRSPLA78S27G702C","050 6212965","1978-11-27 00:00:00" +32,"Barsocchi","Sergio",2868,"BRSSRG52E10L850T","050 6212935","1952-05-10 00:00:00" +603,"Barsotti","Alessia",,"BRSLSS90P42G702B",,"1990-09-02 00:00:00" +38,"Bartalesi Lenzi","Valentina",17481,"BRTVNT80R55G702I","050 6213143","1980-10-15 00:00:00" +33,"Bartolini","Cesare",14336,"BRTCSR77R27G687D","050 6213467","1977-10-27 00:00:00" +516,"Basile","Davide",18872,"BSLDVD84L24H163S","050 6212976","1984-07-24 00:00:00" +471,"Bedini","Chiara",,"BDNCHR87A64F023M",,"1987-01-24 00:00:00" +37,"Bertolino","Antonia",28070,"BRTNTN60R56E974Z","050 6212914","1960-10-16 00:00:00" +7,"Biagioni","Stefania",2577,"BGNSFN51A45B832A","050 6212900","1951-01-05 00:00:00" +18,"Binante","Vincenzo",12798,"BNNVCN76R15C352G","050 6212932","1976-10-15 00:00:00" +41,"Biscoglio","Isabella",15610,"BSCSLL74R54H163P","050 6212042","1974-10-14 00:00:00" +13,"Bolettieri","Paolo",11074,"BLTPLA71L20F052H","050 6212968","1971-07-20 00:00:00" +11,"Bolognesi","Tommaso",2362,"BLGTMS52R26D548E","050 6212797","1952-10-26 00:00:00" +29,"Borri","Francesca",9344,"BRRFNC71M61G702S","050 6212966","1971-08-21 00:00:00" +42,"Bosio","Catherine",11440,"BSOCHR64R51Z110U","050 6213470","1964-10-11 00:00:00" +56,"Calabrò","Antonello",12859,"CLBNNL81C05H224R","050 6213463","1981-03-05 00:00:00" +57,"Callieri","Marco",9388,"CLLMRC75D24F952O","050 6212608","1975-04-24 00:00:00" +62,"Campone","Gabriele",11501,"CMPGRL78L31Z315T","050 6212009","1978-07-31 00:00:00" +66,"Candela","Leonardo",10378,"CNDLRD76B13A399L","050 6212039","1976-02-13 00:00:00" +511,"Carboni","Andrea",17548,"CRBNDR79C08G702H","050 6213144","1979-03-08 00:00:00" +73,"Cardillo","Andrea",3478,"CRDNDR50D27E233U","050 6212957","1950-04-27 00:00:00" +75,"Carlini","Emanuele",17278,"CRLMNL81M13E463L","050 6213011","1981-08-13 00:00:00" +85,"Casini","Giovanni",20584,"CSNGNN79R29G843T","050 6213115","1979-10-29 00:00:00" +87,"Cassarà","Pietro",17242,"CSSPTR79C05G273U","050 6212844","1979-03-05 00:00:00" +88,"Castelli","Donatella",29247,"CSTDTL57L65A864M","050 6212902","1957-07-25 00:00:00" +431,"Caudai","Claudia",17673,"CDACLD78D61G687Y","050 6212962","1978-04-21 00:00:00" +63,"Cempini","Pierangela",1770,"CMPPNG66B61G702M","050 6212206","1966-02-21 00:00:00" +54,"Chiaradonna","Silvano",6919,"CHRSVN66S21Z133B","050 6213005","1966-11-21 00:00:00" +424,"Ciancia","Vincenzo",15294,"CNCVCN77S01F052K","050 6212795","1977-11-01 00:00:00" +51,"Cignoni","Paolo",1717,"CGNPLA68M27B509P","050 6212926","1968-08-27 00:00:00" +67,"Cini","Riccardo",12902,"CNIRCR68T30E715W","050 6212193","1968-12-30 00:00:00" +76,"Cirillo","Roberto",17180,"CRLRRT81E29L628N","050 6212978","1981-05-29 00:00:00" +50,"Coco","Alessandro",11298,"CCOLSN70P02C351P","050 6212928","1970-09-02 00:00:00" +59,"Colantonio","Sara",10888,"CLNSRA74S49A485U","050 6213141","1974-11-09 00:00:00" +60,"Coltelli","Primo",29223,"CLTPRM55L11E625S","050 6213061","1955-07-11 00:00:00" +64,"Concordia","Cesare",6828,"CNCCSR65S20E058O","050 6212970","1965-11-20 00:00:00" +71,"Coppola","Massimo",9585,"CPPMSM69P25C773E","050 6212992","1969-09-25 00:00:00" +78,"Coro","Gianpaolo",15381,"CROGPL80H25F839W","050 6218210","1980-06-25 00:00:00" +82,"Corsini","Massimiliano",10951,"CRSMSM74A10H980B","050 6212925","1974-01-10 00:00:00" +472,"Crivello","Antonino",18727,"CRVNNN87D20D423F","050 6213003","1987-04-20 00:00:00" +93,"D'Acunto","Mario",14613,"DCNMRA66M20F224I","050 6213121","1966-08-20 00:00:00" +108,"D'Angelo","Caterina",4853,"DNGCRN60L45L331L","050 6212960","1960-07-05 00:00:00" +115,"Da Riva","Monja",14536,"DRVMNJ78B51C957U","050 6213139","1978-02-11 00:00:00" +122,"Dazzi","Patrizio",11507,"DZZPRZ79S23B832W","050 6213074","1979-11-23 00:00:00" +109,"De Angelis","Guglielmo",10947,"DNGGLL78M01I838S","050 6213468","1978-08-01 00:00:00" +98,"De Faveri","Federico",12989,"DFVFRC81D29G113X","050 6213473","1981-04-29 00:00:00" +107,"De Mitry","Francesca",11750,"DMTFNC72S46H793U","050 6212975","1972-11-06 00:00:00" +91,"Debole","Franca",10358,"DBLFNC76M42Z110M","050 6212888","1976-08-02 00:00:00" +393,"Dell'Amico","Andrea",15206,"DLLNDR68H16L833Z","050 6213139","1968-06-16 00:00:00" +101,"Dellepiane","Matteo",10367,"DLLMTT79A03D969W","050 6212925","1979-01-03 00:00:00" +100,"Deluca","Rosaria",8272,"DLCRSR65T46G702U","050 6212941","1965-12-06 00:00:00" +92,"Di Benedetto","Marco",11360,"DBNMRC79R10A089M","39 3388482883","1979-10-10 00:00:00" +99,"Di Giandomenico","Felicita",4873,"DGNFCT60R44C633V","050 6212904","1960-10-04 00:00:00" +755,"Di Marco","Giovanna",,"DMRGNN93D58U708E","050 6213374","1983-04-18 00:00:00" +818,"Di Mauro","Gianmarco",,"DMRGMR95A17A028F","050 6213374","1995-01-17 00:00:00" +94,"Diciotti","Roberta",4438,"DCTRRT60E54G843L","050 6212883","1960-05-14 00:00:00" +331,"Esuli","Andrea",11607,"SLENDR77T26G702W","050 6212895","1977-12-26 00:00:00" +123,"Fabbrini","Fabrizio",6431,"FBBFRZ49S22E202Z","050 6212915","1949-11-22 00:00:00" +126,"Fagni","Tiziano",15411,"FGNTZN75L19G713R","050 6212905","1975-07-19 00:00:00" +127,"Falchi","Brunella",5673,"FLCBNL67C67G702J","050 6212880","1967-03-27 00:00:00" +128,"Falchi","Chiara",12714,"FLCCHR79A50A657F","050 6213000","1979-01-10 00:00:00" +129,"Falchi","Fabrizio",11508,"FLCFRZ75H11G843N","050 6212911","1975-06-11 00:00:00" +130,"Falconetti","Liana",11977,"FLCLNI64A62Z133E","050 6218209","1964-01-22 00:00:00" +133,"Fantini","Enrico",6017,"FNTNRC56D03F935I","050 6213131","1956-04-03 00:00:00" +124,"Feo","Giuditta Moly",12899,"FEOGTT76M57G702C","050 6212954","1976-08-17 00:00:00" +138,"Ferrante","Manuela",11326,"FRRMNL83M56G702N","050 6212796","1983-08-16 00:00:00" +137,"Ferrari","Alessio",15226,"FRRLSS82A18I726H","050 6218211","1982-01-18 00:00:00" +139,"Ferro","Erina",6441,"FRRRNE51S45G702U","050 6213070","1951-11-05 00:00:00" +136,"Fieroni","Chiara",,"FRNCHR86R44G702E",,"1986-10-04 00:00:00" +678,"Fortunato","Margherita",,"FRTMGH90A67E734A",,"1990-01-27 00:00:00" +141,"Frosini","Luca",17127,"FRSLCU80H29D403A","050 6218203","1980-06-29 00:00:00" +134,"Furfari","Francesco",10379,"FRFFNC64C13D086Z","0587 628141","1964-03-13 00:00:00" +135,"Furletti","Barbara",15454,"FRLBBR76T43E625E","050 6212882","1976-12-03 00:00:00" +144,"Fusco","Giuseppe",5769,"FSCGPP62L26H224R","050 6212035","1962-07-26 00:00:00" +148,"Gabrielli","Lorenzo",17579,"GBRLNZ84R01G628X","050 6212967","1984-10-01 00:00:00" +152,"Galesi","Giulio",14248,"GLSGLI83E10E379Y","050 6213470","1983-05-10 00:00:00" +680,"Gambacorta","Simona",,"GMBSMN90L58A662V",,"1990-07-18 00:00:00" +815,"Gambino","Vittoria",,"GMBVTR91C42B429S","050 6213374","1991-03-02 00:00:00" +160,"Ganovelli","Fabio",9395,"GNVFBA71B10L424Q","00 3284148124","1971-02-10 00:00:00" +156,"Gennai","Francesco",7209,"GNNFNC59P25G702N","050 6212592","1959-09-25 00:00:00" +155,"Gennaro","Claudio",7849,"GNNCLD68A12D960O","050 6213077","1968-01-12 00:00:00" +149,"Ghiani","Giuseppe",10892,"GHNGPP81S05B354D","050 6213112","1981-11-05 00:00:00" +154,"Gianfaldoni","Antonella",7932,"GNFNNL53H43G702B","050 6212122","1953-06-03 00:00:00" +158,"Giannini","Silvia",7455,"GNNSLV61T47G702O","050 6212121","1961-12-07 00:00:00" +157,"Giannotti","Fosca",7601,"GNNFSC58R44C303V","050 6212999","1958-10-04 00:00:00" +162,"Giorgi","Daniela",11697,"GRGDNL79T61G482Z","050 6212608","1979-12-21 00:00:00" +165,"Girardi","Maria",12757,"GRRMRA75B63F839N","050 6213075","1975-02-23 00:00:00" +163,"Girolami","Michele",14516,"GRLMHL81L07G628D","050 6212950","1981-07-07 00:00:00" +159,"Gnesi","Stefania",7405,"GNSSFN54L61E625P","050 6212918","1954-07-21 00:00:00" +168,"Gotta","Alberto",11903,"GTTLRT77R01D969N","050 6212053","1977-10-01 00:00:00" +166,"Grassini","Fabio",12849,"GRSFBA82B13G702C","050 6212169","1982-02-13 00:00:00" +167,"Grossi","Valerio",18828,"GRSVLR77L18B832G","050 6213006","1977-07-18 00:00:00" +410,"Guidotti","Riccardo",17806,"GDTRCR88A11G716J","050 6213004","1988-01-11 00:00:00" +169,"Kuruoglu","Ercan Engin",9339,"KRGRNN69S00X000X","050 6213128","1969-11-02 00:00:00" +172,"La Bruzzo","Sandro Fabrizio",15926,"LBRSDR82P20I199Q","050 6212802","1982-09-20 00:00:00" +436,"La Rosa","Davide",17292,"LRSDVD86P28E463U","050 6212958","1986-09-28 00:00:00" +609,"Labernarda","Rossana",,"LBRRSN88B62D122Z",,"1988-02-22 00:00:00" +186,"Lami","Giuseppe",9343,"LMAGPP68M29G702D","050 6213493","1968-08-29 00:00:00" +197,"Latella","Diego",12148,"LTLDGI59B09H224P","050 6212982","1959-02-09 00:00:00" +816,"Laurini","Laura",,"LRNLRA92L55D086C","050 6213374","1992-07-15 00:00:00" +691,"Lazzeri","Emma",17801,"LZZMME81E62E463W",,"1981-05-22 00:00:00" +184,"Lelii","Lucio",12904,"LLELCU79T12F870I","050 6213473","1979-12-12 00:00:00" +390,"Lenzi","Stefano",15209,"LNZSFN80R24G702U","050 6212844","1980-10-24 00:00:00" +518,"Leone","Giuseppe Riccardo",20759,"LNEGPP74E22L113C","050 6212962","1974-05-22 00:00:00" +193,"Leporini","Barbara",10374,"LPRBBR72T48E715G","050 6212034","1972-12-08 00:00:00" +449,"Lipari","Giuseppe",17516,"LPRGPP86D30D423E","050 6213135","1986-04-30 00:00:00" +176,"Locuratolo","Elvira Immacolata",12448,"LCRLRM52T47F104F","050 6212895","1952-12-07 00:00:00" +180,"Lofrese","Elena",12368,"LFRLNE53D51A048A","050 6212977","1953-04-11 00:00:00" +187,"Lombardi","Giovanni",12162,"LMBGNN59T09G702K","050 6212006","1959-12-09 00:00:00" +188,"Lombardi","Stefania",10775,"LMBSFN76C46I045D","050 6212273","1976-03-06 00:00:00" +192,"Lonetti","Francesca",12861,"LNTFNC77P67B771D","050 6213468","1977-09-27 00:00:00" +196,"Lorefice","Antonino",,"LRFNNN84H25H163Z",,"1984-06-25 00:00:00" +173,"Lucchese","Claudio",10588,"LCCCLD80D19F206P","050 6212468","1980-04-19 00:00:00" +416,"Magnani","Matteo",15289,"MGNMTT79R01C573W","050 6213073","1979-10-01 00:00:00" +204,"Magrini","Massimo",11178,"MGRMSM66R01E715C","050 6213144","1966-10-01 00:00:00" +207,"Malomo","Luigi",17431,"MLMLGU85H08C352Q","050 6212980","1985-06-08 00:00:00" +212,"Manca","Marco",15890,"MNCMRC82E25E004N","050 6213127","1982-05-25 00:00:00" +210,"Mancini","Carmine",13590,"MNCCMN57E20I301M","050 6212956","1957-05-20 00:00:00" +211,"Mancini","Luca",12751,"MNCLCU87M13G702W","050 6212955","1987-08-13 00:00:00" +215,"Manghi","Paolo",11027,"MNGPLA70T22D612X","050 6212038","1970-12-22 00:00:00" +214,"Mangiacrapa","Francesco",17057,"MNGFNC83E18A512N","050 6212908","1983-05-18 00:00:00" +219,"Mannocci","Andrea",18819,"MNNNDR84C27G702H","050 6212899","1984-03-27 00:00:00" +223,"Marchetti","Eda",10407,"MRCDEA72M50B455T","050 6213467","1972-08-10 00:00:00" +234,"Martinelli","Massimo",5253,"MRTMSM67R30G702R","050 6212803","1967-10-30 00:00:00" +236,"Massink","Mieke",7758,"MSSMKI65D58Z126H","050 6212981","1965-04-18 00:00:00" +396,"Mavilia","Fabio",18728,"MVLFBA86L18G702V","050 6213003","1986-07-18 00:00:00" +242,"Mazzanti","Franco",39290,"MZZFNC58T25E715W","050 6212919","1958-12-25 00:00:00" +201,"Meghini","Carlo",39029,"MGHCRL56C02E783X","050 6212893","1956-03-02 00:00:00" +217,"Mennucci Bernardini","Manuela",13906,"MNNMNL53B41G702M","050 6212876","1953-02-01 00:00:00" +228,"Miori","Vittorio",13594,"MRIVTR57L06L378H","050 6213007","1957-07-06 00:00:00" +754,"Mirabelli","Ileana Rosa",,"MRBLRS90B44E919X","050 6213374","1990-02-04 00:00:00" +397,"Molino","Anna",17056,"MLNNNA84B67A485C","050 6212120","1984-02-27 00:00:00" +221,"Montani","Claudio",13406,"MNTCLD53E02B893Q","050 6212923","1953-05-02 00:00:00" +469,"Montesi","Sara",,"MNTSRA86M47G843P",,"1986-08-07 00:00:00" +227,"Moreo Fernandez","Alejandro David",17540,"MRFLND85P11Z131I","050 6212890","1985-09-11 00:00:00" +231,"Mori","Giulio",14547,"MROGLI68H29E625F","050 6213010","1968-06-29 00:00:00" +230,"Moroni","Davide",10889,"MRNDVD77B04E801X","050 6213130","1977-02-04 00:00:00" +208,"Mulas","Daniela",10757,"MLSDNL59T56E281S","050 6213466","1959-12-16 00:00:00" +222,"Muntean","Cristina Ioana",17756,"MNTCST85P59Z129X","050 6213014","1985-09-19 00:00:00" +714,"Muntoni","Alessandro",17631,"MNTLSN90B02H856X","050 6212921","1990-02-02 00:00:00" +237,"Musto","Daniela",39205,"MSTDLG59R56E463J","050 6212896","1959-10-16 00:00:00" +247,"Nanni","Mirco",10315,"NNNMRC73M11Z130N","050 6212843","1973-08-11 00:00:00" +249,"Nardi","Alessandro",11331,"NRDLSN75B22E625N","050 6213470","1975-02-22 00:00:00" +248,"Nardini","Franco Maria",14935,"NRDFNC80E05G491G","050 6212969","1980-05-05 00:00:00" +259,"Padovani","Cristina",16985,"PDVCST61L63E625M","050 6212951","1961-07-23 00:00:00" +261,"Pagano","Pasquale",8128,"PGNPQL68A17F839V","050 6212891","1968-01-17 00:00:00" +265,"Palma","Gianpaolo",15815,"PLMGPL84C10I549Y","050 6212921","1984-03-10 00:00:00" +263,"Palumbo","Filippo",15654,"PLMFPP81S18C975U","050 6212988","1981-11-18 00:00:00" +269,"Panichi","Giancarlo",15409,"PNCGCR75S04L103G","050 6213473","1975-11-04 00:00:00" +267,"Paoletti","Serena",9578,"PLTSRN73A70G702U","050 6212885","1973-01-30 00:00:00" +278,"Pappalardo","Luca",17541,"PPPLCU84E05H703D","050 6213013","1984-05-05 00:00:00" +283,"Paradisi","Paolo",8972,"PRDPLA72C02I608N","050 6213129","1972-03-02 00:00:00" +282,"Pardi","Luciano",16676,"PRDLCN51D13G702J","050 6213132","1951-04-13 00:00:00" +602,"Pardini","Andrea",,"PRDNDR93L29G843V",,"1993-07-29 00:00:00" +280,"Pardini","Carmen",373,"PRDCMN59P55B455B","050 6212987","1959-09-15 00:00:00" +281,"Pardini","Francesca",14716,"PRDFNC77D55G702B","050 6213146","1977-04-15 00:00:00" +531,"Parvin","Parvaneh",20565,"PRVPVN84M42Z224D",,"1984-08-02 00:00:00" +291,"Pascali","Maria Antonietta",17443,"PSCMNT81R46D862B","050 6213126","1981-10-06 00:00:00" +292,"Pasquinelli","Giuseppe",16734,"PSQGPP48M30M126P","050 6212950","1948-08-30 00:00:00" +293,"Pastoris","Claudia",16770,"PSTCLD52R64D612C","050 6212875","1952-10-24 00:00:00" +295,"Paternò","Fabio",16556,"PTRFBA60P05L628J","050 6213066","1960-09-05 00:00:00" +298,"Pavoni","Lucia",1640,"PVNLCU65P53F839H","050 6212012","1965-09-13 00:00:00" +495,"Pellegrini","Daniele",17266,"PLLDNL76D26E715E","050 6212948","1976-04-26 00:00:00" +286,"Perego","Raffaele",42067,"PRGRFL62A28A757Y","050 6212993","1962-01-28 00:00:00" +757,"Piano","Federica",,"PNIFRC91M62B519H","050 6213374","1991-08-22 00:00:00" +256,"Picchi","Maria",42118,"PCCMRA63M50G702W","050 6213149","1963-08-10 00:00:00" +257,"Piccioli","Tommaso",11391,"PCCTMS69D11G702V","050 6213139","1969-04-11 00:00:00" +287,"Pieri","Gabriele",10868,"PRIGRL74E31G491R","050 6213120","1974-05-31 00:00:00" +290,"Pierotti","Alessandra",5482,"PRTLSN69L49G702V","050 6212877","1969-07-09 00:00:00" +297,"Pietroni","Nico",11967,"PTRNCI78R20I726E","050 6212608","1978-10-20 00:00:00" +262,"Pillitteri","Loredana",381,"PLLLDN59D65D086U","050 6212944","1959-04-25 00:00:00" +271,"Pingi","Paolo",9972,"PNGPLA69A10E506F","050 6213116","1969-01-10 00:00:00" +601,"Pisano","Francesca",,"PSNFNC87R53G702M",,"1987-10-13 00:00:00" +266,"Polini","Andrea",14943,"PLNNDR73P21H769R","050 6212795","1973-09-21 00:00:00" +268,"Ponchio","Federico",10104,"PNCFRC72B27L840F","050 6212925","1972-02-27 00:00:00" +470,"Ponti","Roberto",,"PNTRRT84T21G113X",,"1984-12-21 00:00:00" +279,"Porcelli","Margherita",14715,"PRCMGH82D50D612N","050 6212931","1982-04-10 00:00:00" +389,"Potenziani","Marco",17658,"PTNMRC81L10A269S","050 6218212","1981-07-10 00:00:00" +296,"Potortì","Francesco",968,"PTRFNC63D04H224K","050 6213058","1963-04-04 00:00:00" +616,"Pozza","Stefano",17277,"PZZSFN86A20A459D","050 6212799","1986-01-20 00:00:00" +255,"Puccini","Jonathan",12771,"PCCJTH72E22G702Z","050 6213135","1972-05-22 00:00:00" +391,"Puntoni","Marco",15192,"PNTMRC77T05G702B","050 6212875","1977-12-05 00:00:00" +302,"Rabitti","Fausto",18021,"RBTFST52E04F257U","050 6212897","1952-05-04 00:00:00" +316,"Ranzuglia","Guido",12814,"RNZGDU77D16L191O","050 6212608","1977-04-16 00:00:00" +324,"Raviolo","Claudia",89,"RVLCLD56M41L219R","050 6212403","1956-08-01 00:00:00" +307,"Reggiannini","Marco",15642,"RGGMRC81H12L833E","050 6213469","1981-06-12 00:00:00" +315,"Renso","Chiara",7670,"RNSCHR68R57L781C","050 6213001","1968-10-17 00:00:00" +301,"Ribolini","Alberto",18552,"RBLLRT55E27B832V","050 6212807","1955-05-27 00:00:00" +303,"Ricci","Giovanni",18626,"RCCGNN58S30G702P","050 6212874","1958-11-30 00:00:00" +310,"Righi","Marco",15643,"RGHMRC75M21E625Z","050 6213138","1975-08-21 00:00:00" +317,"Rinzivillo","Salvatore",11512,"RNZSVT76R29H163J","050 6213073","1976-10-29 00:00:00" +638,"Robol","Leonardo",17419,"RBLLRD88E02H612A","050 6212799","1988-05-02 00:00:00" +464,"Romano","Vittorio Enrico Carlo",17181,"RMNVTR73C05H224X","050 6212996","1973-03-05 00:00:00" +319,"Rossetti","Giulio",17514,"RSSGLI84R06E202M","050 6212997","1984-10-06 00:00:00" +318,"Russo","Dario",17415,"RSSDRA76D18E625D","050 6212961","1976-04-18 00:00:00" +333,"Salerno","Emanuele",19208,"SLRMNL58S04A494N","050 6213137","1958-11-04 00:00:00" +337,"Salvetti","Ovidio",19053,"SLVVDO51P18A944Q","050 6213124","1951-09-18 00:00:00" +344,"Santoro","Antonino",19771,"SNTNNN50H22D423N","050 6212986","1950-06-22 00:00:00" +342,"Santoro","Carmelina",5500,"SNTCML69R48F205S","050 6213053","1969-10-08 00:00:00" +352,"Savino","Pasquale",1712,"SVNPQL55S26L505N","050 6212898","1955-11-26 00:00:00" +328,"Scopigno","Roberto",19458,"SCPRRT60L20H501Z","050 6212929","1960-07-20 00:00:00" +329,"Scozzari","Andrea",9492,"SCZNDR66L09E625Y","050 6212338","1966-07-09 00:00:00" +325,"Sebastiani","Fabrizio",19855,"SBSFRZ60B15C372J","050 6212892","1960-02-15 00:00:00" +334,"Silvestri","Fabrizio",9847,"SLVFRZ75R10A561S","050 6213011","1975-10-10 00:00:00" +341,"Sinibaldi","Fabio",17426,"SNBFBA84A05B354L","050 6213071","1984-01-05 00:00:00" +351,"Siotto","Eliana",18779,"STTLNE73R57I726Q",,"1973-10-17 00:00:00" +345,"Spagnolo","Giorgio Oronzo",15822,"SPGGGR81B09I119V","050 6212806","1981-02-09 00:00:00" +350,"Straccia","Umberto",625,"STRMRT65P29Z133D","050 6212894","1965-09-29 00:00:00" +356,"Tampucci","Marco",12770,"TMPMRC81H17G702E","050 6213122","1981-06-17 00:00:00" +360,"Tarabella","Leonello",20383,"TRBLLL48E10D730F","050 6213331","1948-05-10 00:00:00" +361,"ter Beek","Maurice Henri",10361,"TRBMCH72R07Z126L","050 6213471","1972-10-07 00:00:00" +817,"Tomasi","Elisabetta",,"TMSLBT95T55G843G","050 6213374","1995-12-15 00:00:00" +359,"Tonazzini","Anna",20066,"TNZNNA57M53E463D","050 6213136","1957-08-13 00:00:00" +358,"Tonellotto","Nicola",10362,"TNLNCL75R20E463P","050 6212967","1975-10-20 00:00:00" +468,"Torracca","Barbara",,"TRRBBR86T44I449V",,"1986-12-04 00:00:00" +371,"Tozzi","Francesca",,"TZZFNC84C49Z336K",,"1984-03-09 00:00:00" +372,"Tozzi","Sabrina",14439,"TZZSRN71P61L702F","050 6213560","1971-09-21 00:00:00" +366,"Trani","Salvatore",17897,"TRNSVT83D12E396M",,"1983-04-12 00:00:00" +368,"Trasarti","Roberto",14301,"TRSRRT79M08M082U","050 6213006","1979-08-08 00:00:00" +364,"Trentanni","Gianluca",11305,"TRNGLC68B27L117E","050 6212922","1968-02-27 00:00:00" +369,"Trivella","Maria Luisa Rita",12998,"TRVMLS69M70D612T","050 6212918","1969-08-30 00:00:00" +367,"Trupiano","Luca",15251,"TRPLCU65S05H501F","050 6212942","1965-11-05 00:00:00" +453,"Vadicamo","Lucia",17950,"VDCLCU87A64D086T","050 6213068","1987-01-24 00:00:00" +679,"Vagli","Letizia",,"VGLLTZ93H65C236Z",,"1993-06-25 00:00:00" +379,"Vairo","Claudio Francesco",17161,"VRACDF82P22H926O","050 6213064","1982-09-22 00:00:00" +381,"Vassale","Gian Franco",22388,"VSSGFR50H01L833Q","050 6212941","1950-06-01 00:00:00" +404,"Versienti","Loredana",10929,"VRSLDN72D68I119V","050 6213464","1972-04-28 00:00:00" +375,"Villa","Paolo",14835,"VLLPLA79C03F704J","050 6213121","1979-03-03 00:00:00" +376,"Volpini","Federico",12767,"VLPFRC79R18G843Z","050 6213140","1979-10-18 00:00:00" +388,"Zoppi","Franco",11958,"ZPPFNC56P29G702L","050 6212973","1956-09-29 00:00:00" diff --git a/src/test/resources/it/cnr/isti/epasmed/web/rest/epas/posizioniISTI.csv b/src/test/resources/it/cnr/isti/epasmed/web/rest/epas/posizioniISTI.csv new file mode 100644 index 0000000..e194859 --- /dev/null +++ b/src/test/resources/it/cnr/isti/epasmed/web/rest/epas/posizioniISTI.csv @@ -0,0 +1,167 @@ +"id","dal","al","livello","cf","cognome","nome","flag_del","id_flusso","data_mod" +"2459","2018-12-27","","3","BCCFCM82C17A285V","Bacco","Felice Manlio","0","98954","2019-12-05 16:48:53" +"2460","2018-12-27","","3","BGLMRM73P70D583A","Baglioni","Miriam","0","78937","2018-12-06 15:07:13" +"2483","2018-12-27","","3","BNTFNC82S30F861R","Banterle","Francesco","0","78937","2018-12-06 16:36:58" +"2461","2018-12-27","","3","BRDLSS82D69C415P","Bardi","Alessia","0","79319","2018-12-07 12:46:22" +"1945","2011-10-03","2017-01-01","6","BRNPLA73T13G843E","Baronti","Paolo","1","34182","2017-01-13 08:35:11" +"2423","2017-01-01","","5","BRRFNC71M61G702S","Borri","Francesca","0","76030","2018-11-08 10:41:05" +"2765","2020-06-01","","2","BRSPLA78S27G702C","Barsocchi","Paolo","0","106709","2020-05-29 14:08:43" +"82","2004-01-02","","1","BRTNTN60R56E974Z","Bertolino","Antonia","0","20185","2015-11-26 14:47:20" +"2462","2018-12-27","","3","BRTVNT80R55G702I","Bartalesi Lenzi","Valentina","0","78937","2018-12-06 15:13:33" +"2484","2018-12-27","","6","BSCSLL74R54H163P","Biscoglio","Isabella","0","78937","2018-12-06 16:38:09" +"2683","2019-11-28","","3","BSLDVD84L24H163S","Basile","Davide","0","99025","2019-12-06 17:34:12" +"2429","2018-11-30","","6","BSOCHR64R51Z110U","Bosio","Catherine","0","76490","2018-11-14 12:17:07" +"1190","2014-01-03","","6","CCOLSN70P02C351P","Coco","Alessandro","0","5321","2014-01-03 10:20:50" +"2639","2019-09-30","","3","CDACLD78D61G687Y","Caudai","Claudia","0","95695","2019-10-07 16:52:00" +"2062","2010-01-01","","1","CGNPLA68M27B509P","Cignoni","Paolo","0","49300","2017-06-30 13:16:51" +"105","2008-04-14","","3","CHRSVN66S21Z133B","Chiaradonna","Silvano","0","770","2012-11-29 12:47:13" +"2485","2018-12-27","","6","CLBNNL81C05H224R","Calabrò","Antonello","0","78937","2018-12-06 16:39:13" +"2766","2020-06-01","","2","CLLMRC75D24F952O","Callieri","Marco","0","106709","2020-05-29 14:07:23" +"1189","2013-12-31","","3","CLNSRA74S49A485U","Colantonio","Sara","0","54011","2017-12-05 16:37:31" +"123","2009-10-01","2020-11-01","8","CMPGRL78L31Z315T","Campone","Gabriele","0","117555","2021-01-20 15:17:56" +"2421","2018-01-01","","5","CMPPNG66B61G702M","Cempini","Pierangela","0","75951","2018-11-07 17:34:44" +"126","2009-02-16","","3","CNCCSR65S20E058O","Concordia","Cesare","0","55795","2018-01-18 15:39:57" +"2486","2018-12-27","","3","CNCVCN77S01F052K","Ciancia","Vincenzo","0","78937","2018-12-06 16:44:45" +"2420","2018-01-01","","6","CNIRCR68T30E715W","Cini","Riccardo","0","75951","2018-11-07 17:07:53" +"2463","2018-12-27","","3","CRBNDR79C08G702H","Carboni","Andrea","0","78937","2018-12-06 15:15:22" +"2464","2018-12-27","","3","CRLMNL81M13E463L","Carlini","Emanuele","0","78937","2018-12-06 15:43:20" +"2466","2018-12-27","","3","CRLRRT81E29L628N","Cirillo","Roberto","0","78937","2018-12-06 15:49:05" +"2487","2018-12-27","","3","CROGPL80H25F839W","Coro","Gianpaolo","0","78937","2018-12-06 16:45:53" +"2767","2020-06-01","","2","CRSMSM74A10H980B","Corsini","Massimiliano","0","106709","2020-05-29 14:10:27" +"2620","2019-08-30","2020-08-29","3","CRVNNN87D20D423F","Crivello","Antonino","0","117057","2021-01-08 10:52:32" +"2780","2020-07-01","","3","CSNGNN79R29G843T","Casini","Giovanni","0","109539","2020-07-28 10:41:08" +"2465","2018-12-27","","3","CSSPTR79C05G273U","Cassarà","Pietro","0","99159","2019-12-09 14:06:25" +"2061","2010-01-01","","1","CSTDTL57L65A864M","Castelli","Donatella","0","48168","2017-05-23 15:34:33" +"183","2012-10-16","","3","DBLFNC76M42Z110M","Debole","Franca","0","58598","2018-03-07 16:53:23" +"2433","2018-11-30","","3","DBNMRC79R10A089M","Di Benedetto","Marco","0","76533","2018-11-15 14:56:55" +"190","2006-01-01","","4","DCTRRT60E54G843L","Diciotti","Roberta","0","770","2012-11-29 12:47:13" +"199","2001-12-31","","2","DGNFCT60R44C633V","Di Giandomenico","Felicita","0","115128","2020-11-12 15:57:18" +"2489","2018-12-27","","6","DLCRSR65T46G702U","Deluca","Rosaria","0","78937","2018-12-06 16:47:51" +"2488","2018-12-27","","6","DLLNDR68H16L833Z","Dell'Amico","Andrea","0","78937","2018-12-06 16:46:48" +"2432","2018-11-30","","7","DMTFNC72S46H793U","De Mitry","Francesca","0","76490","2018-11-14 12:48:03" +"219","2006-01-01","","4","DNGCRN60L45L331L","D'Angelo","Caterina","0","770","2012-11-29 12:47:13" +"219","2006-01-01","","4","DNGCRN60L45L331L","D'Angelo","Caterina Maria Paola","0","770","2012-11-29 12:47:13" +"240","2011-06-01","","6","FEOGTT76M57G702C","Feo","Giuditta Moly","0","770","2012-11-29 12:47:13" +"2419","2018-01-01","","5","FLCBNL67C67G702J","Falchi","Brunella","0","75951","2018-11-07 17:06:19" +"2504","2018-12-27","","7","FLCCHR79A50A657F","Falchi","Chiara","0","79319","2018-12-07 09:27:41" +"246","2011-09-16","","3","FLCFRZ75H11G843N","Falchi","Fabrizio","0","114111","2020-10-26 12:15:24" +"2490","2018-12-27","","7","FLCLNI64A62Z133E","Falconetti","Liana","0","78937","2018-12-06 16:50:45" +"255","2005-01-01","","4","FNTNRC56D03F935I","Fantini","Enrico","0","770","2012-11-29 12:47:13" +"891","2013-01-04","","3","FRFFNC64C13D086Z","Furfari","Francesco","0","48826","2017-06-09 10:47:20" +"2841","2020-12-01","","3","FRLBBR76T43E625E","Furletti","Barbara","0","115692","2020-11-27 11:17:38" +"2491","2018-12-27","","3","FRRLSS82A18I726H","Ferrari","Alessio","0","78937","2018-12-06 16:53:48" +"263","2009-03-02","","7","FRRMNL83M56G702N","Ferrante","Manuela","0","770","2012-11-29 12:47:13" +"2467","2018-12-27","","3","FRSLCU80H29D403A","Frosini","Luca","0","78937","2018-12-06 15:50:58" +"2492","2018-12-27","","6","FSCGPP62L26H224R","Fusco","Giuseppe","0","78937","2018-12-06 16:54:58" +"2513","2018-12-27","","3","GBRLNZ84R01G628X","Gabrielli","Lorenzo","0","80536","2018-12-20 08:23:15" +"2493","2018-12-27","","6","GLSGLI83E10E379Y","Galesi","Giulio","0","78937","2018-12-06 16:56:44" +"300","2005-01-01","","4","GNNFNC59P25G702N","Gennai","Francesco","0","770","2012-11-29 12:47:13" +"303","2007-01-01","","1","GNNFSC58R44C303V","Giannotti","Fosca","0","12689","2015-01-21 10:46:58" +"307","2006-01-01","","4","GNNSLV61T47G702O","Giannini","Silvia","0","770","2012-11-29 12:47:13" +"2769","2020-06-01","","2","GNVFBA71B10L424Q","Ganovelli","Fabio","0","106709","2020-05-29 14:15:42" +"2770","2020-06-01","","2","GRGDNL79T61G482Z","Giorgi","Daniela","0","106709","2020-05-29 14:17:18" +"2696","2019-12-31","","3","GRLMHL81L07G628D","Girolami","Michele","0","100094","2019-12-31 09:51:29" +"325","2011-12-01","","3","GRRMRA75B63F839N","Girardi","Maria","0","115029","2020-11-11 12:31:28" +"2494","2018-12-27","2020-11-01","8","GRSFBA82B13G702C","Grassini","Fabio","0","117555","2021-01-20 15:15:07" +"2840","2020-12-01","","3","GRSVLR77L18B832G","Grossi","Valerio","0","115692","2020-11-27 11:14:07" +"329","2012-01-16","","3","GTTLRT77R01D969N","Gotta","Alberto","0","33830","2017-01-09 08:52:58" +"2774","2020-06-01","","1","KRGRNN69S00X000X","Kuruoglu","Ercan Engin","0","106709","2020-05-29 14:23:10" +"2468","2018-12-27","","6","LBRSDR82P20I199Q","La Bruzzo","Sandro Fabrizio","0","78937","2018-12-06 15:52:58" +"2495","2018-12-27","","3","LLELCU79T12F870I","Lelii","Lucio","0","78937","2018-12-06 17:00:53" +"363","2007-12-01","","3","LMAGPP68M29G702D","Lami","Giuseppe","0","12471","2015-01-16 15:44:26" +"367","2006-01-01","","4","LMBGNN59T09G702K","Lombardi","Giovanni","0","770","2012-11-29 12:47:13" +"2435","2018-11-30","","7","LMBSFN76C46I045D","Lombardi","Stefania","0","76699","2018-11-19 12:21:33" +"2802","2020-07-30","","3","LNEGPP74E22L113C","Leone","Giuseppe Riccardo","0","109662","2020-07-30 17:27:01" +"2428","2018-11-30","","3","LNTFNC77P67B771D","Lonetti","Francesca","0","76490","2018-11-14 11:46:42" +"376","2007-05-29","","3","LPRBBR72T48E715G","Leporini","Barbara","0","770","2012-11-29 12:47:13" +"2470","2018-12-27","","8","LPRGPP86D30D423E","Lipari","Giuseppe","0","78937","2018-12-06 16:01:40" +"2469","2018-12-27","","3","LRSDVD86P28E463U","La Rosa","Davide","0","98954","2019-12-05 16:54:58" +"384","2001-12-31","","2","LTLDGI59B09H224P","Latella","Diego","0","4975","2013-11-04 13:19:23" +"2671","2019-12-03","2020-12-02","3","LZZMME81E62E463W","Lazzeri","Emma","0","111598","2020-09-23 15:18:33" +"391","2001-12-28","","2","MGHCRL56C02E783X","Meghini","Carlo","0","14329","2015-03-05 10:16:01" +"2431","2018-11-30","","3","MGRMSM66R01E715C","Magrini","Massimo","0","84734","2019-02-21 10:40:56" +"2781","2020-07-01","","3","MLMLGU85H08C352Q","Malomo","Luigi","0","107486","2020-06-11 17:22:23" +"2473","2018-12-27","","6","MLNNNA84B67A485C","Molino","Anna","0","78937","2018-12-06 16:17:58" +"2430","2018-11-30","","7","MLSDNL59T56E281S","Mulas","Daniela","0","76490","2018-11-14 12:20:42" +"412","2011-02-16","","6","MNCLCU87M13G702W","Mancini","Luca","0","770","2012-11-29 12:47:13" +"2472","2018-12-27","","3","MNCMRC82E25E004N","Manca","Marco","0","78937","2018-12-06 16:16:08" +"1998","2017-03-01","","7","MNDTZN68M67I422Z","Amendola","Tiziana","0","39476","2017-02-27 15:21:18" +"2471","2018-12-27","","3","MNGFNC83E18A512N","Mangiacrapa","Francesco","0","78937","2018-12-06 16:16:52" +"418","2011-09-16","","3","MNGPLA70T22D612X","Manghi","Paolo","0","14480","2015-03-12 11:16:57" +"2645","2019-10-09","2020-10-08","3","MNNNDR84C27G702H","Mannocci","Andrea","0","119071","2021-02-11 15:35:26" +"2600","2019-07-01","","3","MNTCST85P59Z129X","Muntean","Cristina","0","91752","2019-06-26 19:10:35" +"2600","2019-07-01","","3","MNTCST85P59Z129X","Muntean","Cristina Ioana","0","91752","2019-06-26 19:10:35" +"2656","2019-11-04","2020-11-03","3","MNTLSN90B02H856X","Muntoni","Alessandro","0","111598","2020-09-23 15:03:23" +"434","2011-12-16","","3","MRCDEA72M50B455T","Marchetti","Eda","0","12689","2015-01-21 12:08:27" +"2474","2018-12-27","","3","MRFLND85P11Z131I","Moreo Fernandez","Alejandro David","0","78937","2018-12-06 16:19:26" +"2474","2018-12-27","","3","MRFLND85P11Z131I","Moreo Fernandez","Alejandro","0","78937","2018-12-06 16:19:26" +"441","2007-01-01","","2","MRIVTR57L06L378H","Miori","Vittorio","0","12967","2015-01-27 12:26:11" +"443","2010-12-15","","3","MRNDVD77B04E801X","Moroni","Davide","0","53967","2017-12-04 10:42:19" +"444","2011-12-16","","3","MROGLI68H29E625F","Mori","Giulio","0","56191","2018-01-24 11:39:11" +"2258","2017-01-01","","5","MRTMSM67R30G702R","Martinelli","Massimo","0","56011","2018-01-19 08:25:58" +"451","2008-04-14","","3","MSSMKI65D58Z126H","Massink","Mieke","0","93856","2019-08-22 13:49:22" +"453","1989-07-01","","3","MSTDLG59R56E463J","Musto","Daniela","0","770","2012-11-29 12:47:13" +"2280","2010-01-01","","2","MTAGPP68D15D976W","Amato","Giuseppe","0","57873","2018-02-21 15:12:46" +"461","2005-12-31","","2","MZZFNC58T25E715W","Mazzanti","Franco","0","117057","2021-01-08 17:12:31" +"2771","2020-06-01","","2","NNNMRC73M11Z130N","Nanni","Mirco","0","106709","2020-05-29 14:18:38" +"2505","2018-12-27","","6","NRDLSN75B22E625N","Nardi","Alessandro","0","79319","2018-12-07 09:40:25" +"2736","2020-03-02","","1","NSLLCN57P27I608Z","Anselmo","Luciano","0","102011","2020-02-11 14:48:19" +"485","2006-01-01","","5","PCCMRA63M50G702W","Picchi","Maria","0","770","2012-11-29 12:47:13" +"493","2001-12-31","","1","PDVCST61L63E625M","Padovani","Cristina","0","12967","2015-01-27 08:53:11" +"499","2009-02-16","","2","PGNPQL68A17F839V","Pagano","Pasquale","0","57027","2018-02-08 10:51:12" +"2427","2018-11-30","","3","PLLDNL76D26E715E","Pellegrini","Daniele","0","115029","2020-11-11 13:20:46" +"500","2006-06-01","","3","PLLLDN59D65D086U","Pillitteri","Loredana","0","93976","2019-08-27 10:37:45" +"2514","2018-12-27","","3","PLMFPP81S18C975U","Palumbo","Filippo","0","80739","2018-12-27 07:55:47" +"2475","2018-12-27","","3","PLMGPL84C10I549Y","Palma","Gianpaolo","0","78937","2018-12-06 16:25:38" +"512","2010-01-01","","7","PLTSRN73A70G702U","Paoletti","Serena","0","770","2012-11-29 12:47:13" +"513","2010-02-01","","3","PNCFRC72B27L840F","Ponchio","Federico","0","14092","2015-02-26 14:41:32" +"2496","2018-12-27","","6","PNCGCR75S04L103G","Panichi","Giancarlo","0","78937","2018-12-06 17:02:31" +"516","2010-09-01","","3","PNGPLA69A10E506F","Pingi","Paolo","0","12581","2015-01-19 09:18:23" +"858","2012-12-10","","6","PNTMRC77T05G702B","Puntoni","Marco","0","1545","2013-01-29 15:52:34" +"2601","2019-07-01","","3","PPPLCU84E05H703D","Pappalardo","Luca","0","91752","2019-06-26 19:12:33" +"529","2001-09-05","","3","PRDCMN59P55B455B","Pardini","Carmen","0","12581","2015-01-19 12:25:01" +"2497","2018-12-27","","6","PRDFNC77D55G702B","Pardini","Francesca","0","78937","2018-12-06 17:03:28" +"2908","2020-01-01","","2","PRDPLA72C02I608N","Paradisi","Paolo","0","118650","2021-02-05 14:45:29" +"2737","2020-02-17","","1","PRGRFL62A28A757Y","Perego","Raffaele","0","102011","2020-02-11 14:53:38" +"881","2012-12-31","","3","PRIGRL74E31G491R","Pieri","Gabriele","0","77467","2018-11-27 15:18:02" +"2422","2018-01-01","","6","PRTLSN69L49G702V","Pierotti","Alessandra","0","76030","2018-11-08 09:15:04" +"2755","2020-04-01","2021-03-31","3","PRVPVN84M42Z224D","Parvin","Parvaneh","0","104603","2020-04-17 11:07:19" +"2476","2018-12-27","","3","PSCMNT81R46D862B","Pascali","Maria Antonietta","0","78937","2018-12-06 16:27:22" +"2637","2019-09-30","","3","PTNMRC81L10A269S","Potenziani","Marco","0","95190","2019-09-26 10:57:51" +"564","2001-12-31","","1","PTRFBA60P05L628J","Paternò","Fabio","0","115128","2020-11-12 15:28:12" +"565","2001-09-04","","2","PTRFNC63D04H224K","Potortì","Francesco","0","27783","2016-09-30 12:23:35" +"1000","2013-01-01","2020-11-01","6","PVNLCU65P53F839H","Pavoni","Lucia","0","117555","2021-01-20 15:18:26" +"581","2006-01-01","","6","RCCGNN58S30G702P","Ricci","Giovanni","0","770","2012-11-29 12:47:13" +"2498","2018-12-27","","3","RGGMRC81H12L833E","Reggiannini","Marco","0","78937","2018-12-06 17:04:33" +"2499","2018-12-27","","3","RGHMRC75M21E625Z","Righi","Marco","0","78937","2018-12-06 17:06:27" +"2477","2018-12-27","","3","RMNVTR73C05H224X","Romano","Vittorio Enrico Carlo","0","115128","2020-11-12 20:56:28" +"2772","2020-06-01","","2","RNSCHR68R57L781C","Renso","Chiara","0","106709","2020-05-29 14:20:22" +"607","2010-07-01","","3","RNZSVT76R29H163J","Rinzivillo","Salvatore","0","12860","2015-01-23 10:21:42" +"2478","2018-12-27","","3","RSSDRA76D18E625D","Russo","Dario","0","78937","2018-12-06 16:30:34" +"2515","2018-12-28","","3","RSSGLI84R06E202M","Rossetti","Giulio","0","80739","2018-12-27 16:01:38" +"2256","2018-02-01","","6","RTNMHL76B25A390M","Artini","Michele","0","55267","2018-01-15 10:15:55" +"934","2010-01-01","","4","RVLCLD56M41L219R","Raviolo","Claudia","0","1545","2013-01-29 15:05:10" +"2644","2019-10-01","","1","SBSFRZ60B15C372J","Sebastiani","Fabrizio","0","95695","2019-10-07 13:17:49" +"625","2001-12-31","","1","SCPRRT60L20H501Z","Scopigno","Roberto","0","12860","2015-01-23 10:10:25" +"626","2009-02-16","","3","SCZNDR66L09E625Y","Scozzari","Andrea","0","12471","2015-01-16 13:08:59" +"2768","2020-06-01","","2","SLENDR77T26G702W","Esuli","Andrea","0","106709","2020-05-29 14:12:05" +"635","2001-12-31","","2","SLRMNL58S04A494N","Salerno","Emanuele","0","12471","2015-01-16 12:34:42" +"636","2009-02-16","2013-10-21","3","SLVFRZ75R10A561S","Silvestri","Fabrizio","0","113934","2020-10-21 11:18:18" +"2479","2018-12-27","","3","SNBFBA84A05B354L","Sinibaldi","Fabio","0","115866","2020-12-01 10:04:26" +"2773","2020-06-01","","2","SNTCML69R48F205S","Santoro","Carmelina","0","106709","2020-05-29 14:21:37" +"2480","2018-12-27","","3","SPGGGR81B09I119V","Spagnolo","Giorgio Oronzo","0","112257","2020-10-01 14:38:56" +"2482","2018-12-27","","3","SSNMSM82C05E625D","Assante","Massimiliano","0","78937","2018-12-06 16:35:38" +"2738","2020-02-17","","1","STRMRT65P29Z133D","Straccia","Umberto","0","102011","2020-02-11 14:52:03" +"2638","2019-09-30","","3","STTLNE73R57I726Q","Siotto","Eliana","0","96041","2019-10-10 11:41:28" +"677","2001-10-18","","2","SVNPQL55S26L505N","Savino","Pasquale","0","12471","2015-01-16 11:52:05" +"687","2011-02-16","","6","TMPMRC81H17G702E","Tampucci","Marco","0","770","2012-11-29 12:47:13" +"1125","2010-01-01","","2","TNZNNA57M53E463D","Tonazzini","Anna","0","12471","2015-01-16 13:41:28" +"2501","2018-12-27","","3","TRPLCU65S05H501F","Trupiano","Luca","0","78937","2018-12-06 17:09:23" +"711","2011-09-16","","3","TRSRRT79M08M082U","Trasarti","Roberto","0","14092","2015-02-26 16:25:27" +"2500","2018-12-27","","7","TRVMLS69M70D612T","Trivella","Maria Luisa Rita","0","78937","2018-12-06 17:07:47" +"2458","2018-12-27","","3","TZRCLD83E02B354K","Atzori","Claudio","0","78937","2018-12-06 15:03:51" +"2599","2019-07-01","","3","VDCLCU87A64D086T","Vadicamo","Lucia","0","92313","2019-07-08 14:57:31" +"722","2011-02-16","","6","VLPFRC79R18G843Z","Volpini","Federico","0","113441","2020-10-15 10:41:04" +"2481","2018-12-27","","3","VRACDF82P22H926O","Vairo","Claudio Francesco","0","78937","2018-12-06 16:34:14" +"2502","2018-12-27","","3","VRSLDN72D68I119V","Versienti","Loredana","0","78937","2018-12-06 17:10:42" +"2503","2018-12-27","","3","ZPPFNC56P29G702L","Zoppi","Franco","0","115748","2020-11-30 16:37:50" \ No newline at end of file diff --git a/src/test/resources/logback.xml b/src/test/resources/logback.xml new file mode 100644 index 0000000..3db6c3f --- /dev/null +++ b/src/test/resources/logback.xml @@ -0,0 +1,47 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + WARN + + + + diff --git a/src/test/resources/templates/mail/testEmail.html b/src/test/resources/templates/mail/testEmail.html new file mode 100644 index 0000000..a4ca16a --- /dev/null +++ b/src/test/resources/templates/mail/testEmail.html @@ -0,0 +1 @@ + diff --git a/tsconfig.app.json b/tsconfig.app.json new file mode 100644 index 0000000..5c33c96 --- /dev/null +++ b/tsconfig.app.json @@ -0,0 +1,4 @@ +{ + "extends": "./tsconfig.base.json", + "files": ["src/main/webapp/app/app.main.ts"] +} diff --git a/tsconfig.base.json b/tsconfig.base.json new file mode 100644 index 0000000..afc0bb2 --- /dev/null +++ b/tsconfig.base.json @@ -0,0 +1,31 @@ +{ + "compilerOptions": { + "target": "es6", + "module": "esnext", + "moduleResolution": "node", + "sourceMap": true, + "emitDecoratorMetadata": true, + "experimentalDecorators": true, + "strict": true, + "noImplicitReturns": true, + "noFallthroughCasesInSwitch": true, + "suppressImplicitAnyIndexErrors": true, + "skipLibCheck": true, + "outDir": "target/classes/static/app", + "lib": ["es7", "dom"], + "baseUrl": "./", + "paths": { + "app/*": ["src/main/webapp/app/*"] + }, + "importHelpers": true, + "downlevelIteration": true, + "declaration": false, + "forceConsistentCasingInFileNames": true + }, + "angularCompilerOptions": { + "strictInjectionParameters": true, + "fullTemplateTypeCheck": true, + "strictTemplates": true, + "preserveWhitespaces": true + } +} diff --git a/tsconfig.json b/tsconfig.json new file mode 100644 index 0000000..dda3146 --- /dev/null +++ b/tsconfig.json @@ -0,0 +1,11 @@ +{ + "files": [], + "references": [ + { + "path": "./tsconfig.app.json" + }, + { + "path": "./tsconfig.e2e.json" + } + ] +} diff --git a/tslint.json b/tslint.json new file mode 100644 index 0000000..52fb925 --- /dev/null +++ b/tslint.json @@ -0,0 +1,17 @@ +{ + "rulesDirectory": ["node_modules/codelyzer"], + "rules": { + "directive-selector": [true, "attribute", "jhi", "camelCase"], + "component-selector": [true, "element", "jhi", "kebab-case"], + "no-inputs-metadata-property": true, + "no-outputs-metadata-property": true, + "no-host-metadata-property": true, + "no-input-rename": true, + "no-output-rename": true, + "use-lifecycle-interface": true, + "use-pipe-transform-interface": false, + "component-class-suffix": true, + "directive-class-suffix": true, + "typedef": [true, "call-signature"] + } +} diff --git a/webpack/ePASMedLogo.png b/webpack/ePASMedLogo.png new file mode 100644 index 0000000000000000000000000000000000000000..0384e97809045bed05f7d9dcfeb15a721c2a4c03 GIT binary patch literal 7181 zcmV+o9P;CdP)1-mw<|000McNliruWFU8GbZ8()Nlj2>E@cM*02^^hL_t(|+U=cploa=s z??1Pyx_eUhMDh&7AQ*B+L;(VU5Q~CQw1WK{*S74pa$0-6+UMPO*6CS(^4ibx%Ce-j zCCieK4HiN+NFpO45JM6fh8zZ%96MCieScJUPil~m7zFm#IepGdSNhfc)(zi#@9)>b zRj{clz(8OyFf{&mF}V4ztC@h$FhpHfV}@Zi0_T7uzyV-4un#yaB3Jyg$`ua6FklvN zJun*R38VqW`@D<`Gy|uA&A=*PHE{4s1xKz>AZ`L~1?GC1fO&~7AB$_vGGGa?=4#NI zj|>P^y+Ku>8@4tWKTkqL%zz6Vy|An;$n53dAmkt+a< zslbgpgGL2SGI<5(sd2zbC06~nL*Fy>kqImQD? zB-KMZz%#&OB2xA@3y7-T3w#Y2uaJB{6~MM(=XO~68mwCdR8tLqcpm=v0&=+g z(g1Nja7qWm9Uu2T!BA2d>V2vW_;=txF1KI#C;<6K;D0Nmw*zDIM)=*+u>4iTb>E*^ z`mmswCd|JL9{3E58r#v^*DCzGCGzNBEg-7;CE$AsdF^cd(#7zbCy<>x+KlP~L(}&d zd2;^+4s7dV#*Bkcf59_laT69lvkG`vM1J*G2Z*ZP0sK^&WLOkj+0;UVb= z#&f@c#~wwht1c|G_}i@c1zs2%4`KD`$C0B)+8=t&K=|b6pyCv|eGBZ_F>-RdL}#fobh_EM5qY|12)73z!uF9XJSXm4J;4irR?iqq&X|HC7gx7?mET_P|W zcu-Zp($!Ah0{F%ce>rRWwmoB09bg)Ud)=f-FaNKH?k)d-_xQ2_pJ+$+)~tjle~Fws zn_4&tjp)`L7Tk^Aayv4hv>oH7wg#G;JD=aw2=#T|_gO)>b`p%60P}A{mn?#p|8PN# zlv5|sr~VTPx}%e?YYWP!94L#k`itJa^`l=8*}8SdJWY`pHGqNFl&pU22VdW8sVaB< z{daEPyYIk*s+}+7_xU0@IoT&w^@9L12KY1(NPcs;9G>|#Qhp${VkE%0ZU)@5%$R@Prl1Ni#!ytZAXutte|=OPKLQoXGrZB zJGN}soT35Rr&o6|O;ZxFmYOEqbPJSRgFgK$u{-9G{!F; z|1CS0yeZSCjA!l*)0jG841T|_^`JpR;TzvU)6_YL#hVQlOX6b%IClmXFGLy}&Z`;R|0($L z*C9Kn)ikN{cG%$vo1cizm{ z5kp#eM)#iZm2aT7jlA^x&SMfmYO2x23*p*{XpZ-STxypt0zT801*0~8jyk#4!mce& z6E_ojcTJR)_~S3HARE?qX5R@jHSZSq+=K1Fa9x*YUVND+pZNnP4)4cjD!Bn~Uj`0p zr@mj7-ys)or`ue~!i6ufW&18Z_ldi>_l`E?ub>-z{>x}}C9>kR6pmhywJXu(e};P= z=+fva-K@(eYW24+3p+f03jVSL94BRTqeF&y+@V*WBp9)n&Ci~AmR~&aEKazQ?_PU? z2|XIUohAg&Sy&s>LEB%?Z@N4*@f4rWvy*Sbx>F&3P-gJ2Kl*=E)zs>jK7FrRmT3r} z#U=2G&!LCPy|B2QlaKqoWv{^Oo6!7%E|*o8@{pu z{``q&<z^9Ps zLN8zlPyz)xHn*2l6EIy`T3Y0>C!g~@x9C+Q@y>|A$L@sbvpT*?Ac{nz9(QV2kGWa_ z*}W6Cyp!_1(qIs7Uf>b9__lu2R(;|(3u$a>5`ejVYnd~u4hG-}IWgi)7t-9$_sei0 z(_{GffWBVxW8UyuW);?XoU*pgc>L)Xbko+I5lI-m+@0Rw?^zNV=BrXePMtijnrpxQ zFkwnkUQI1w_myR(Sjf@B-z2cs>XV zv!G6r#4oeZWxm5y~*Zh$+OdnI!)!)g-lc)w+%B+0;p zAhBqQ6td}+Of4%>!*Qlg*VjWm^`LIHW``vcQ;Jm@xa};Y#%#vNL9>sjI|p! z_k3>QD9k+^;5R0LKK$j52E(1iuVXugkYCBVbeD~X4OK4*QMd7gZNK=zw2t<|4 zbphDy^Y}!r-SDH`qBB2LI4mu z2-xl=>jsU8O+a0m8k^HvTEdc%mX<2{sKilI8TG&D5&s;g=*&Z##vW17|l`WUpG#c4~rJ11eh zogp0hxDobMSND8<_1ir@zTl?Qq?ldx=Qq~$t*x!g9_L0#Q`d6^@q4y<5Zcwsl7y0W z4X$*oe}hn6TZibyS^N6B2Gn)DPsb+|CwWT?z>bB9G*n`{DKfGVKe1@I)8O873#6gx znMjHXf~2S84j%3`S2xrnu_#Rq4G7>0yB5sfHF^Aqs%X?PR;=C7^AAgw4coKtK)<D@$viaB&T~fG(8ioJd63mhvW5+ z4nhy*AeR5)&9JJc9gf|(^|g&Zvf8DX&sXvLAe+w*`Q20RYp93X8nmz&4dsxL*PW)K z&Frmgf>=7s&V-qnRgV=fljM(gIJ3s3>9|3zI@}hkB8DL=*1R>iZ2PYM-STr=9M_d| zmDQPz4UOh>yOx_{)gD(D2=#Z#8|z^B!0>?+hGl1S*lY*E5OVuoL;8?msL$t_t^<3J z=EhX+2@1L+J|7>ZZYUS{qU%y8riu3IlX~Fc7UXDovN0{-_fa~aBzf$s`*K;kv$bfI zUhL9ymYp2!U?Y<)OZ{XUtDTD)?07482@0S7n$=&sj0j zdTDYfD|TeF>~L<zTmUG>1iEa zG&!&rjm5z79RJY41IZ2rscUGUD(qv?rhKN3Y$PpH5ry72#t2skCzksWptl?2A(WFca*_jk&@-o!$M!2OWW?}6$uMi#ySc0^3nlj(=}>%kjH17b zB38TSMQee{m#dU~GD_)ZBjgTo(75nL1d;d)ElS=9htkhSD0v`?*(R>2J7TbSLq5&6 z7bypWSxgu|8pAYNf!|h!QPa62Ujet@HE3S`)u|x@kd}@P9^Sbr(4(#xWpATR4N0;z zdgL%BO>Auie(himufLT8wXJl-2tbdUYzBTILRL{{!k1m*Qu@UR-EXjwwg&ke{Q1pX zRvybvel>OSwTu`#C^)HF5$)c@!H0(E^x{d!)5KD2IJ73do7R2fxvolb+$71pBJR*+w z$fX1OGjsYROhb}D6)Pf)YT-* z;R8#GxoOS}fBw)Ib3Mg{y}g2zDx5omv3@1uIGv*_^`K8+F+IeM3HCYhz}o{Hg)?H!QDG72wzVb_2!iH!MfeaWX8(+D#+=R7=L;Z z*|oiMeo4LDu#TUMJ82qSsvHVcO}WSMXO>xXzyGr@Z`=11$KCsxOHbBY<44WT!JRo5 z*|!^MYG{YO2Z~^)@?i4xqBb}<1J_;<+G8V zp2mQpeheQrh_R!GGoYjxLyY!m^%F-i{`ef`TdPy%@c_ul!<{+ToArw%Q}mb8bSX;I zHuTr8_kZev`->j2==Fn|=E!}~3nUY@xw(0>NHXwz->DNp@>O-5h?j+TCr(2S?88{{ zVy9R?3CgNv5N^S}r>vb4i;q zn@}j&W{IITH{7)iW64YLkXuqjRq<&Nv*&vONg}P{rL#7H)?TbNi9{krc>VaI9SM%W z>1t_`!e^Vk2|I?kVu(tL*wj^xys>iaqc=<*^}IckZpY(`JF?!<+8t2`?uUM2ZT%*|{A6!LsOC(hm&<+c3KLNIS+rDG##CLYf>Xr70Wa>WMyQ0!EUbId9k9{xBtL5P1F28nxgb8 z(kgjNMEJy){%LVTedCeygs$#@;^NIyrhK5ej+@qO+WxfT*gsZPpNOE{d*aT&1EcaR z96i)29W9C3xd-=Ro;Zr^+l5ZM0e8r7#4uZr&dNf4KE#f70>m>hw3->mMs{t-Si1t_ ztyww|}^zsBhu74N|9yd-mwF z=D?O!T|b-ca7~SIbl>6MMq~E<0+M(@S@#C!b5D6~{weHoqROi%nwyV~pNx*32&F@C zL)plw6Zrn^n^0Tbe#&CE9>o6bCs5D>>T8j`JCQ9Lk+O|QU2SqUww%YF)2j&k{?DQl zr?r9sYWb{RU(>tb3w`_cy}HBRlH6SP?d`iCQ;iI=?bx*mEzs$6a3WF6g};UB%FaF| z85avuQw6UtMc1vs?Nfx=zZCNFQPV^^R)`G)W9W6##eYn$SdjUzjF~YR0<37LdtJB7fxGc;5 zUDoxoR=vITD~|1aU)2qH=D;&e#>Tanue^lp-SwWJT!39-cOC_%AM(j4t;AW~_Z@`*&>H{;un|X>I+5l<&j*%gY`Yaoh{O)gOKw@q1bp z&h^ek5xyu6}##KRK@Ru)3O+unh{((u}cc z8Rp7m$dNZU)g&7_FdCr zEs-zVj&mojnyq#`(fA2cUxyspkL=%rQNABepYVFSB4Ka24sElxqGdrK0KshNU5I;4 zDOx%d?O%$<=fg`I_8mXZ_F3i&=^1IioH%mmhL2JnSH#-ydrzJWo;y|bpzXvy8;dyu zB!%Nh3NIQ#8tS30#sjLc0iy9NV;~L93ZkKG$jkm6Eh))$KWbTAQNPu&_1%M08^X=^+P1yGvF%>%#wXH76KGv@M$SuC+AJxTXbwsi zt<<#4lcs6Bm>Uc}KXS-`vMWUfSLE-9Y}Zu1?J?V(Yuk1Ss#a@Zh4-BT*LDDc z7=~?_#zCLu`%6Z;|4(Cwmae_hem~@j{)L^geMbg|n;Ncn9Cw!ExMLm1>8Yw2N!@6( z=y56yQBXsq*|3aLKEvGX^95F?1^lZ=4e7uCzkgxpLu|*69iy@#IxrFr4~aR>a8>sj z7cEg&vjxd@TyGwQX&CCN)|!S4lA1M&4q{bHr36i4RW(*6 z+NQf!M2(8^rB+JuMceOv|A6niuk)O9?%#9H@BHrPx}N8H(6%rOK5j{F006)Tu{5;@ z063U?*I`aJ0$qrNvo~jLtr2GItx;H`QACqrL=)Q^Mc@qg?03Eg{>j6CA>mE#sF%>h zZde8dmQIFckZm&d{2%*;{ImTV-%BVq=@hoNOz8TD^S={~!kcVUyJ33;6CvarjvgOZnB7JW7o5Blel)Ha2&hkZw>4McK+M56E>AiI+<($iKCp{m>Xv;ghTjch9m|y~Rb$6N~zhBbS>tXM=MWr8~ zob~*=7QR;WYVsLI9`@G?e6wP9eK}!cTP7-81UcX0(A{i^%NWOJOE2y2?mlkgnw|jw zctjwk7ZA}L3(pS;0^|Sy*^Li7q0jndj4Z?`6vTj8q3i8e#LeZBU2vKxogx3rW}gQqAsoT zGaPX0Q@Wa%j=JcbP;Kk@%;1BZ;@SDN-3NUhbWSfdoi?9#=#s3t@ctsM(LL0l_eOy# z3kYSUc^yj_fBYt|vOcr1SR-<9!9z5T$LwdWJ^0y|#|399fA9vig*n*zjW&#RF>oQ#67?FmZfre?(-{Tm~D^%=gP^_W=Z!UFp1iM&3|E_r6DU%x%cr@ zVdjxE>CvqG6@D$=zW%-!9UB=(u4Ce*s+fm8W&M?;?tXndWC02ecn0%!81gt09`qRK zKTNxr>F6n0FTzVp%61Kz=NqjRsiP8ladYlTq1`{XvHNv2gf23#ea?+b4yA|9Tu8C< zkr6-(tnqG6`~DoDFlg?w1oNK+}fx zS*a_CJRMR*c$#d`BECVL7V10Ij+7q`#t1}_4}`?M5*KLbGzaDh z)?_(LZisnb8vZyal6er*lgHruaLZ}*^bc`j-)7!h!u)6}iGv6=Pdz(s zHVJ!?A!6bgRtmaa{AUX%C0GwN}D0#>F%9CQ`wUDQg zybVO;$;^SY(f<8;_fRPnsx#hSQC!R+LhtI5mq$O|4tLmXz)DHjbZUn;<@Ig;ffFdX z1dn~ha&eEJIR$1njJqhQ_9cay%Ld#(9F&J!WA*Sc-7_3R+hMh!vG z_g&A${jl~B2Oa~d5{fLrWlX5U5>8ssf|5p9mE%q`bIPMDUqVt}a3BK=NbMK5N{Bc7)LU zEmT4W`5hi0>B&5k{sBbeZHaOcIJusz&5TqSgJ40>(X zOc8aMN4f;+f{j&oPzmK~@%&K?b=f!J8DozgS|dAVJ|azn4elb)R-Sga%pnzK6zhdE zdwzp6OcD4oR=ws1yVVu1`9P?c(fzK9tuHiQrU0%26XL$Ko1h@^*3oz?uetzl0L!!5 zk)Ps$_YANfqTzh|{1mTT!il(;^-t(~pS zWFhH@wvlZ2QdMqL1m=mU;+DFiqI19BEo@5tJ|nxX9Ew-uzl5|IC^Y!#(jB})8gHVW zt{w=YM+nyq0Vb?XMTRXr?=0{!lHr?cqceo^P_ju|C*W?2{+S(@2 zQ=vw0zl=4r33h@7C?HO31CPBP&CTC>f51{-hbtYykDyOQ=^~T8Tdw`3xP(1}Y-Kt{ zMa_a{)2Gg!)(C5o%(u6p1%jUtqofnSRCVAYpBUU9Ugs%g^vyeUG=MW;#(2h>$NNEh zgvEoKx-$mN?KGpeMQVJHCmX&-N#C%Gfn%F*O0>QJrPinrt))kRdM7lyMJU0+TW52= zwoRSbK3*!UPngrh>4@JB94Y=c*@ZqX5TZR6J{>7|V{nuytIjN&MI)|a;Nv|Pf;ldF81pSTCw=qkA3NUXX1x6D%zew+=iC- zknUgSG7PFm1s~0#9ZY}A5_qMuZ0FlGJQ374Vpy{f&3oAN%1Ysn3XbueI#7JArw?Lg zf|+iAlZj~r2nH;k3zfUwTR!udH3m2Oi~X@P*aV?aUmtuFs5pat7eKDeY^3o>)mlwR z9aak$sc3TQb(hcufpkW|)waC%sVtHxv)~X&DZsx^LmOjA7jpw0%--VRvw!pDke28q z098@jXWJDeR{`yU4O`K6txL#uMCIf8fv)NIC*r{c6F+Hmb@c|`}z85ggK-LW%)ONWtx#9+fhw*x{yBtOr}X1X1S zgZQPaz5{Mn9ZkVU;>^xeR|qtGlheF<6fw$4O{v$Gytv(zX3$yM<&)CJly1fFVq5T$ zP?L-XAN54)*fEB?{9?O0hS#dz~?f2*!k77A+P;evuFsMsJ~X@V&HUcgs(dQ$~YIjeg4^{ zv2)p`UTXqU%+~E`T2o+|!uREMn$fR+&4JxkK?L&uu|)W^*Y7*RsVaNC5Hpx5*4QKY EKQ4Pe { + // use Typescript alias in Webpack only if this has value + return Boolean(value.length); + }) + .map(([key, value]) => { + // if Typescript alias ends with /* then remove this for Webpack + const regexToReplace = /\/\*$/; + const aliasKey = key.replace(regexToReplace, ''); + const aliasValue = value[0].replace(regexToReplace, ''); + return [aliasKey, root(aliasValue)]; + }) + .reduce((aliases, [key, value]) => { + aliases[key] = value; + return aliases; + }, webpackAliases); + return webpackAliases; +} diff --git a/webpack/webpack.common.js b/webpack/webpack.common.js new file mode 100644 index 0000000..ca84250 --- /dev/null +++ b/webpack/webpack.common.js @@ -0,0 +1,100 @@ +const webpack = require('webpack'); +const CopyWebpackPlugin = require('copy-webpack-plugin'); +const HtmlWebpackPlugin = require('html-webpack-plugin'); +const AngularCompilerPlugin = require('@ngtools/webpack').AngularCompilerPlugin; + +const utils = require('./utils.js'); + +module.exports = (options) => ({ + resolve: { + extensions: ['.ts', '.js'], + modules: ['node_modules'], + mainFields: [ 'es2015', 'browser', 'module', 'main'], + alias: utils.mapTypescriptAliasToWebpackAlias() + }, + stats: { + children: false + }, + module: { + rules: [ + { + test: /(?:\.ngfactory\.js|\.ngstyle\.js|\.ts)$/, + loader: '@ngtools/webpack' + }, + { + test: /\.html$/, + loader: 'html-loader', + options: { + minimize: { + caseSensitive: true, + removeAttributeQuotes:false, + minifyJS:false, + minifyCSS:false + } + }, + exclude: utils.root('src/main/webapp/index.html') + }, + { + test: /\.(jpe?g|png|gif|svg|woff2?|ttf|eot)$/i, + loader: 'file-loader', + options: { + digest: 'hex', + hash: 'sha512', + // For fixing src attr of image + // See https://github.com/jhipster/generator-jhipster/issues/11209 + name: 'content/[hash].[ext]', + esModule: false + } + }, + { + test: /manifest.webapp$/, + loader: 'file-loader', + options: { + name: 'manifest.webapp' + } + }, + // Ignore warnings about System.import in Angular + { test: /[\/\\]@angular[\/\\].+\.js$/, parser: { system: true } }, + ] + }, + plugins: [ + new webpack.DefinePlugin({ + 'process.env': { + NODE_ENV: `'${options.env}'`, + BUILD_TIMESTAMP: `'${new Date().getTime()}'`, + // APP_VERSION is passed as an environment variable from the Gradle / Maven build tasks. + VERSION: `'${process.env.hasOwnProperty('APP_VERSION') ? process.env.APP_VERSION : 'DEV'}'`, + DEBUG_INFO_ENABLED: options.env === 'development', + // The root URL for API calls, ending with a '/' - for example: `"https://www.jhipster.tech:8081/myservice/"`. + // If this URL is left empty (""), then it will be relative to the current context. + // If you use an API server, in `prod` mode, you will need to enable CORS + // (see the `jhipster.cors` common JHipster property in the `application-*.yml` configurations) + SERVER_API_URL: `''` + } + }), + new CopyWebpackPlugin({ + patterns: [ + { from: './node_modules/swagger-ui-dist/*.{js,css,html,png}', to: 'swagger-ui', flatten: true, globOptions: { ignore: ['**/index.html'] }}, + { from: './node_modules/axios/dist/axios.min.js', to: 'swagger-ui' }, + { from: './src/main/webapp/swagger-ui/', to: 'swagger-ui' }, + { from: './src/main/webapp/content/', to: 'content' }, + { from: './src/main/webapp/favicon.ico', to: 'favicon.ico' }, + { from: './src/main/webapp/manifest.webapp', to: 'manifest.webapp' }, + // jhipster-needle-add-assets-to-webpack - JHipster will add/remove third-party resources in this array + { from: './src/main/webapp/robots.txt', to: 'robots.txt' } + ], + }), + new HtmlWebpackPlugin({ + template: './src/main/webapp/index.html', + chunks: ['polyfills', 'main', 'global'], + chunksSortMode: 'manual', + inject: 'body', + base: '/', + }), + new AngularCompilerPlugin({ + mainPath: utils.root('src/main/webapp/app/app.main.ts'), + tsConfigPath: utils.root('tsconfig.app.json'), + sourceMap: true + }) + ] +}); diff --git a/webpack/webpack.dev.js b/webpack/webpack.dev.js new file mode 100644 index 0000000..476a187 --- /dev/null +++ b/webpack/webpack.dev.js @@ -0,0 +1,120 @@ +const webpack = require('webpack'); +const writeFilePlugin = require('write-file-webpack-plugin'); +const webpackMerge = require('webpack-merge'); +const BrowserSyncPlugin = require('browser-sync-webpack-plugin'); +const FriendlyErrorsWebpackPlugin = require('friendly-errors-webpack-plugin'); +const SimpleProgressWebpackPlugin = require('simple-progress-webpack-plugin'); +const WebpackNotifierPlugin = require('webpack-notifier'); +const path = require('path'); +const sass = require('sass'); + +const utils = require('./utils.js'); +const commonConfig = require('./webpack.common.js'); + +const ENV = 'development'; + +module.exports = (options) => webpackMerge(commonConfig({ env: ENV }), { + devtool: 'eval-source-map', + devServer: { + contentBase: './target/classes/static/', + proxy: [{ + context: [ + '/api', + '/services', + '/management', + '/swagger-resources', + '/v2/api-docs', + '/h2-console', + '/auth' + ], + target: `http${options.tls ? 's' : ''}://localhost:8080`, + secure: false, + changeOrigin: options.tls + }], + stats: options.stats, + watchOptions: { + ignored: /node_modules/ + }, + https: options.tls, + historyApiFallback: true + }, + entry: { + global: './src/main/webapp/content/scss/global.scss', + main: './src/main/webapp/app/app.main' + }, + output: { + path: utils.root('target/classes/static/'), + filename: 'app/[name].bundle.js', + chunkFilename: 'app/[id].chunk.js' + }, + module: { + rules: [{ + test: /\.(j|t)s$/, + enforce: 'pre', + loader: 'eslint-loader', + exclude: /node_modules/ + }, + { + test: /\.scss$/, + use: ['to-string-loader', 'css-loader', 'postcss-loader', { + loader: 'sass-loader', + options: { implementation: sass } + }], + exclude: /(vendor\.scss|global\.scss)/ + }, + { + test: /(vendor\.scss|global\.scss)/, + use: ['style-loader', 'css-loader', 'postcss-loader', { + loader: 'sass-loader', + options: { implementation: sass } + }] + }] + }, + stats: process.env.JHI_DISABLE_WEBPACK_LOGS ? 'none' : options.stats, + plugins: [ + process.env.JHI_DISABLE_WEBPACK_LOGS + ? null + : new SimpleProgressWebpackPlugin({ + format: options.stats === 'minimal' ? 'compact' : 'expanded' + }), + new FriendlyErrorsWebpackPlugin(), + new BrowserSyncPlugin({ + https: options.tls, + host: 'localhost', + port: 9000, + proxy: { + target: `http${options.tls ? 's' : ''}://localhost:9060`, + proxyOptions: { + changeOrigin: false //pass the Host header to the backend unchanged https://github.com/Browsersync/browser-sync/issues/430 + } + }, + socket: { + clients: { + heartbeatTimeout: 60000 + } + } + /* + ,ghostMode: { // uncomment this part to disable BrowserSync ghostMode; https://github.com/jhipster/generator-jhipster/issues/11116 + clicks: false, + location: false, + forms: false, + scroll: false + } */ + }, { + reload: false + }), + new webpack.ContextReplacementPlugin( + /angular(\\|\/)core(\\|\/)/, + path.resolve(__dirname, './src/main/webapp/') + ), + new writeFilePlugin(), + new webpack.WatchIgnorePlugin([ + utils.root('src/test'), + ]), + new WebpackNotifierPlugin({ + title: 'ePASMed', + contentImage: path.join(__dirname, 'ePASMedLogo.png') + }) + ].filter(Boolean), + mode: 'development' +}); diff --git a/webpack/webpack.prod.js b/webpack/webpack.prod.js new file mode 100644 index 0000000..7e36306 --- /dev/null +++ b/webpack/webpack.prod.js @@ -0,0 +1,147 @@ +const webpack = require('webpack'); +const webpackMerge = require('webpack-merge'); +const MiniCssExtractPlugin = require('mini-css-extract-plugin'); +const OptimizeCSSAssetsPlugin = require("optimize-css-assets-webpack-plugin"); +const BundleAnalyzerPlugin = require('webpack-bundle-analyzer').BundleAnalyzerPlugin; +const MomentLocalesPlugin = require('moment-locales-webpack-plugin'); +const TerserPlugin = require('terser-webpack-plugin'); +const WorkboxPlugin = require('workbox-webpack-plugin'); + +const utils = require('./utils.js'); +const commonConfig = require('./webpack.common.js'); + +const ENV = 'production'; +const sass = require('sass'); + +module.exports = webpackMerge(commonConfig({ env: ENV }), { + // Enable source maps. Please note that this will slow down the build. + // You have to enable it in Terser config below and in tsconfig.json as well + // devtool: 'source-map', + entry: { + global: './src/main/webapp/content/scss/global.scss', + main: './src/main/webapp/app/app.main' + }, + output: { + path: utils.root('target/classes/static/'), + filename: 'app/[name].[hash].bundle.js', + chunkFilename: 'app/[id].[hash].chunk.js' + }, + module: { + rules: [ + { + test: /\.scss$/, + use: ['to-string-loader', 'css-loader', 'postcss-loader', { + loader: 'sass-loader', + options: { implementation: sass } + }], + exclude: /(vendor\.scss|global\.scss)/ + }, + { + test: /(vendor\.scss|global\.scss)/, + use: [ + { + loader: MiniCssExtractPlugin.loader, + options: { + publicPath: '../' + } + }, + 'css-loader', + 'postcss-loader', + { + loader: 'sass-loader', + options: { implementation: sass } + } + ] + }, + { + test: /\.css$/, + use: ['to-string-loader', 'css-loader'], + exclude: /(vendor\.css|global\.css)/ + }, + { + test: /(vendor\.css|global\.css)/, + use: [ + { + loader: MiniCssExtractPlugin.loader, + options: { + publicPath: '../' + } + }, + 'css-loader', + 'postcss-loader' + ] + }] + }, + optimization: { + runtimeChunk: false, + minimizer: [ + new TerserPlugin({ + parallel: true, + cache: true, + // sourceMap: true, // Enable source maps. Please note that this will slow down the build + terserOptions: { + ecma: 6, + ie8: false, + toplevel: true, + module: true, + compress: { + dead_code: true, + warnings: false, + properties: true, + drop_debugger: true, + conditionals: true, + booleans: true, + loops: true, + unused: true, + toplevel: true, + if_return: true, + inline: true, + join_vars: true, + ecma: 6, + module: true + }, + output: { + comments: false, + beautify: false, + indent_level: 2, + ecma: 6 + }, + mangle: { + module: true, + toplevel: true + } + } + }), + new OptimizeCSSAssetsPlugin({}) + ] + }, + plugins: [ + new MiniCssExtractPlugin({ + // Options similar to the same options in webpackOptions.output + // both options are optional + filename: 'content/[name].[contenthash].css', + chunkFilename: 'content/[id].css' + }), + new MomentLocalesPlugin({ + localesToKeep: [ + // jhipster-needle-i18n-language-moment-webpack - JHipster will add/remove languages in this array + ] + }), + new BundleAnalyzerPlugin({ + analyzerMode: 'static', + openAnalyzer: false, + // Webpack statistics in target folder + reportFilename: '../stats.html' + }), + new webpack.LoaderOptionsPlugin({ + minimize: true, + debug: false + }), + new WorkboxPlugin.GenerateSW({ + clientsClaim: true, + skipWaiting: true, + exclude: [/swagger-ui/] + }) + ], + mode: 'production' +});